Skip to content

Commit cc86e57

Browse files
Update as per discussion
LF-3081
1 parent 96ccd17 commit cc86e57

File tree

3 files changed

+778
-199
lines changed

3 files changed

+778
-199
lines changed

src/sdc-ig-supplements.js

+204-54
Original file line numberDiff line numberDiff line change
@@ -62,17 +62,21 @@ engine.weight = function (coll) {
6262
res.push(score);
6363
} else if (qItem.answerValueSet || valueCoding.system) {
6464
// Otherwise, check corresponding value set and code system
65-
hasPromise = true;
66-
res.push(getWeightFromTerminologyServer(
67-
this, qItem.answerValueSet, valueCoding.code,
68-
valueCoding.system, checkExtUrl));
65+
const score = getWeightFromCorrespondingResources(this, questionnaire,
66+
qItem.answerValueSet, valueCoding.code, valueCoding.system);
67+
if (score !== undefined) {
68+
res.push(score);
69+
}
70+
hasPromise = hasPromise || score instanceof Promise;
6971
}
7072
} else if (qItem?.answerValueSet) {
7173
// Otherwise, check corresponding value set and code system
72-
hasPromise = true;
73-
res.push(getWeightFromTerminologyServer(
74-
this, qItem.answerValueSet, valueCoding.code,
75-
valueCoding.system, checkExtUrl));
74+
const score = getWeightFromCorrespondingResources(this, questionnaire,
75+
qItem.answerValueSet, valueCoding.code, valueCoding.system);
76+
if (score !== undefined) {
77+
res.push(score);
78+
}
79+
hasPromise = hasPromise || score instanceof Promise;
7680
} else {
7781
throw new Error(
7882
'Questionnaire answerOption/answerValueSet with this linkId was not found: ' +
@@ -82,12 +86,13 @@ engine.weight = function (coll) {
8286
throw new Error('%questionnaire is needed but not specified.');
8387
}
8488
} else if (valueCoding.system) {
85-
// If there are no questionnaire (no linkId) check corresponding value
86-
// set and code system
87-
hasPromise = true;
88-
res.push(getWeightFromTerminologyServer(
89-
this, null, valueCoding.code, valueCoding.system,
90-
checkExtUrl));
89+
// If there are no questionnaire (no linkId) check corresponding code system
90+
const score = getWeightFromCorrespondingResources(this, null,
91+
null, valueCoding.code, valueCoding.system);
92+
if (score !== undefined) {
93+
res.push(score);
94+
}
95+
hasPromise = hasPromise || score instanceof Promise;
9196
}
9297
}
9398
}
@@ -96,18 +101,148 @@ engine.weight = function (coll) {
96101
return hasPromise ? Promise.all(res) : res;
97102
};
98103

104+
99105
/**
100-
* Returns a promise of score value received from the terminology server.
106+
* Returns the value of score or its promise received from a corresponding value
107+
* set or code system.
101108
* @param {Object} ctx - object describing the context of expression
102109
* evaluation (see the "applyParsedPath" function).
110+
* @param {Object} questionnaire - object containing questionnaire resource data
103111
* @param {string} vsURL - value set URL specified in the Questionnaire item.
104112
* @param {string} code - symbol in syntax defined by the system.
105113
* @param {string} system - code system.
106-
* @param {Function} checkExtUrl - function to check if an extension has a URL
107-
* to store a score.
108-
* @return {Promise<number|null|undefined>}
114+
* @return {number|undefined|Promise<number|undefined>}
115+
*/
116+
function getWeightFromCorrespondingResources(ctx, questionnaire, vsURL, code, system) {
117+
let result;
118+
119+
if (code) {
120+
const contextResource = ctx.processedVars.context?.[0].data || ctx.vars.context?.[0];
121+
122+
if (vsURL) {
123+
const vsId = /#(.*)/.test(vsURL) ? RegExp.$1 : null;
124+
const isAnswerValueSet = vsId
125+
? (r) => r.id === vsId && r.resourceType === 'ValueSet'
126+
: (r) => r.url === vsURL && r.resourceType === 'ValueSet';
127+
128+
const containedVS = contextResource?.contained?.find(isAnswerValueSet)
129+
|| questionnaire?.contained?.find(isAnswerValueSet);
130+
131+
if (containedVS) {
132+
if (!containedVS.expansion) {
133+
result = fetch(`${getTerminologyUrl(ctx)}/ValueSet/$expand`, {
134+
method: 'POST',
135+
headers: {
136+
'Accept': 'application/fhir+json',
137+
'Content-Type': 'application/fhir+json'
138+
},
139+
body: JSON.stringify({
140+
"resourceType": "Parameters",
141+
"parameter": [{
142+
"name": "valueSet",
143+
"resource": containedVS
144+
}, {
145+
"name": "property",
146+
"valueString": "itemWeight"
147+
}]
148+
})
149+
})
150+
.then(r => r.ok ? r.json() : Promise.reject(r.json()))
151+
.then((terminologyVS) => {
152+
return getItemWeightFromProperty(
153+
getValueSetItem(terminologyVS.expansion?.contains, code, system)
154+
);
155+
});
156+
} else {
157+
result = getItemWeightFromProperty(
158+
getValueSetItem(containedVS.expansion.contains, code, system)
159+
);
160+
}
161+
} else {
162+
result = fetch(`${getTerminologyUrl(ctx)}/ValueSet?` + new URLSearchParams({
163+
url: vsURL
164+
}, {
165+
headers: {
166+
'Accept': 'application/fhir+json'
167+
}
168+
}).toString())
169+
.then(r => r.ok ? r.json() : Promise.reject(r.json()))
170+
.then((bundle) => {
171+
const terminologyVS = bundle?.entry?.[0]?.resource;
172+
if (!terminologyVS) {
173+
return Promise.reject(
174+
`Cannot resolve the corresponding value set: ${vsURL}`
175+
);
176+
}
177+
return getItemWeightFromProperty(
178+
getValueSetItem(terminologyVS?.expansion?.contains, code, system)
179+
);
180+
});
181+
}
182+
}
183+
184+
if (system) {
185+
if (result === undefined) {
186+
const isCodeSystem = (r) => r.url === system && r.resourceType === 'CodeSystem';
187+
const containedCS = contextResource?.contained?.find(isCodeSystem)
188+
|| questionnaire?.contained?.find(isCodeSystem);
189+
190+
if (containedCS) {
191+
result = getItemWeightFromProperty(
192+
getCodeSystemItem(containedCS?.concept, code)
193+
);
194+
} else {
195+
result = getWeightFromTerminologyCodeSet(ctx, code, system);
196+
}
197+
} else if (result instanceof Promise) {
198+
result = result.then(weightFromVS => {
199+
if (weightFromVS !== undefined) {
200+
return weightFromVS;
201+
}
202+
return getWeightFromTerminologyCodeSet(ctx, code, system);
203+
});
204+
}
205+
}
206+
}
207+
208+
return result;
209+
}
210+
211+
212+
/**
213+
* Returns the promised score value from the code system obtained from the
214+
* terminology server.
215+
* @param {Object} ctx - object describing the context of expression
216+
* evaluation (see the "applyParsedPath" function).
217+
* @param {string} code - symbol in syntax defined by the system.
218+
* @param {string} system - code system.
219+
* @return {Promise<number|undefined>}
220+
*/
221+
function getWeightFromTerminologyCodeSet(ctx, code, system) {
222+
return fetch(`${getTerminologyUrl(ctx)}/CodeSystem/$lookup?` + new URLSearchParams({
223+
code, system, property: 'itemWeight'
224+
}, {
225+
headers: {
226+
'Accept': 'application/fhir+json'
227+
}
228+
}).toString())
229+
.then(r => r.ok ? r.json() : Promise.reject(r.json()))
230+
.then((parameters) => {
231+
return parameters.parameter
232+
.find(p => p.name === 'property'&& p.part
233+
.find(part => part.name === 'code' && part.valueCode === 'itemWeight'))
234+
?.part?.find(p => p.name === 'value')?.valueDecimal;
235+
});
236+
}
237+
238+
239+
/**
240+
* Returns the URL of the terminology server.
241+
* @param {Object} ctx - object describing the context of expression
242+
* evaluation (see the "applyParsedPath" function).
243+
* @return {string}
109244
*/
110-
function getWeightFromTerminologyServer(ctx, vsURL, code, system, checkExtUrl) {
245+
function getTerminologyUrl(ctx) {
111246
if (!ctx.async) {
112247
throw new Error('The asynchronous function "weight"/"ordinal" is not allowed. ' +
113248
'To enable asynchronous functions, use the async=true or async="always"' +
@@ -119,45 +254,59 @@ function getWeightFromTerminologyServer(ctx, vsURL, code, system, checkExtUrl) {
119254
throw new Error('Option "terminologyUrl" is not specified.');
120255
}
121256

122-
// Searching for value sets by item code is extremely onerous for a server
123-
// (see https://www.hl7.org/fhir/valueset.html#search for details); therefore,
124-
// we only check the value set whose URL is specified in the Questionnaire item.
125-
return (vsURL
126-
? fetch(`${terminologyUrl}/ValueSet?` + new URLSearchParams({
127-
url: vsURL
128-
}).toString()).then(r => r.json()).then((bundle) => {
129-
const vs = bundle?.entry?.[0]?.resource;
130-
if (!vs) {
131-
return Promise.reject(`Cannot resolve the corresponding value set: ${vsURL}`);
257+
return terminologyUrl;
258+
}
259+
260+
/**
261+
* Returns an item from "ValueSet.expansion.contains" that has the specified
262+
* code and system.
263+
* @param {Array<Object>} contains - value of "ValueSet.expansion.contains".
264+
* @param {string} code - symbol in syntax defined by the system.
265+
* @param {string} system - code system.
266+
* @return {Object| undefined}
267+
*/
268+
function getValueSetItem(contains, code, system) {
269+
let result;
270+
if (contains) {
271+
for(let i = 0; i < contains.length && !result; i++) {
272+
const item = contains[i];
273+
if (item.code === code && item.system === system) {
274+
result = item;
275+
} else {
276+
result = getValueSetItem(item.contains, code, system);
132277
}
133-
return (
134-
vs.compose?.include?.find(c => c.system === system)?.concept
135-
.find(c => c.code === code)
136-
||
137-
vs.expansion?.contains
138-
?.find(c => c.system === system && c.code === code)
139-
)?.extension?.find(checkExtUrl)?.valueDecimal;
140-
})
141-
: Promise.resolve(null)
142-
).then(weightFromVS => {
143-
if (weightFromVS !== null && weightFromVS !== undefined) {
144-
return weightFromVS;
145278
}
146-
return system
147-
? fetch(`${terminologyUrl}/CodeSystem?` + new URLSearchParams({
148-
code, system
149-
}).toString()).then(r => r.json()).then((bundle) => {
150-
const cs = bundle?.entry?.[0]?.resource;
151-
if (!cs) {
152-
return Promise.reject(`Cannot resolve the corresponding code system: ${system}`);
153-
}
154-
return cs.concept?.find(c => c.code === code)
155-
?.extension?.find(checkExtUrl)?.valueDecimal;
156-
})
157-
: Promise.resolve(null);
158-
});
279+
}
280+
return result;
159281
}
160282

283+
284+
/**
285+
* Returns an item from "CodeSystem.concept" that has the specified code.
286+
* @param {Array<Object>} concept - value of "CodeSystem.concept".
287+
* @param {string} code - symbol in syntax defined by the system.
288+
* @return {Object| undefined}
289+
*/
290+
function getCodeSystemItem(concept, code) {
291+
let result;
292+
if (concept) {
293+
for(let i = 0; i < concept.length && !result; i++) {
294+
const item = concept[i];
295+
if (item.code === code) {
296+
result = item;
297+
} else {
298+
result = getCodeSystemItem(item.concept, code);
299+
}
300+
}
301+
}
302+
return result;
303+
}
304+
305+
function getItemWeightFromProperty(item) {
306+
return item?.property?.find(p => p.code === 'itemWeight')?.valueDecimal;
307+
}
308+
309+
161310
/**
162311
* Returns array of linkIds of ancestor ResourceNodes and source ResourceNode
163312
* starting with the linkId of the given node and ending with the topmost item's
@@ -176,6 +325,7 @@ function getLinkIds(node) {
176325
return res;
177326
}
178327

328+
179329
/**
180330
* Returns a questionnaire item based on the linkIds array of the ancestor
181331
* ResourceNodes and the target ResourceNode. If the questionnaire item is not

test/mock-fetch-results.js

+8-4
Original file line numberDiff line numberDiff line change
@@ -9,12 +9,16 @@ let fetchSpy;
99
function mockFetchResults(results) {
1010
fetchSpy = jest.spyOn(global, 'fetch').mockImplementation(
1111
(url) => new Promise((resolve, _) => {
12-
const mockedResult = results?.find(
12+
const mockedItem = results?.find(
1313
r => r[0] instanceof RegExp ? r[0].test(url) : url.indexOf(r[0]) !== -1
14-
)?.[1];
14+
);
15+
const okResult = mockedItem?.[1];
16+
const badResult = mockedItem?.[2];
1517

16-
if(mockedResult) {
17-
resolve({ json: () => mockedResult });
18+
if(okResult) {
19+
resolve({ json: () => okResult, ok: true });
20+
} else if(badResult) {
21+
resolve({ json: () => badResult, ok: false });
1822
} else {
1923
console.error(`"${url}" is not mocked.`)
2024
}

0 commit comments

Comments
 (0)