@@ -62,17 +62,21 @@ engine.weight = function (coll) {
62
62
res . push ( score ) ;
63
63
} else if ( qItem . answerValueSet || valueCoding . system ) {
64
64
// 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 ;
69
71
}
70
72
} else if ( qItem ?. answerValueSet ) {
71
73
// 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 ;
76
80
} else {
77
81
throw new Error (
78
82
'Questionnaire answerOption/answerValueSet with this linkId was not found: ' +
@@ -82,12 +86,13 @@ engine.weight = function (coll) {
82
86
throw new Error ( '%questionnaire is needed but not specified.' ) ;
83
87
}
84
88
} 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 ;
91
96
}
92
97
}
93
98
}
@@ -96,18 +101,148 @@ engine.weight = function (coll) {
96
101
return hasPromise ? Promise . all ( res ) : res ;
97
102
} ;
98
103
104
+
99
105
/**
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.
101
108
* @param {Object } ctx - object describing the context of expression
102
109
* evaluation (see the "applyParsedPath" function).
110
+ * @param {Object } questionnaire - object containing questionnaire resource data
103
111
* @param {string } vsURL - value set URL specified in the Questionnaire item.
104
112
* @param {string } code - symbol in syntax defined by the system.
105
113
* @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 }
109
244
*/
110
- function getWeightFromTerminologyServer ( ctx , vsURL , code , system , checkExtUrl ) {
245
+ function getTerminologyUrl ( ctx ) {
111
246
if ( ! ctx . async ) {
112
247
throw new Error ( 'The asynchronous function "weight"/"ordinal" is not allowed. ' +
113
248
'To enable asynchronous functions, use the async=true or async="always"' +
@@ -119,45 +254,59 @@ function getWeightFromTerminologyServer(ctx, vsURL, code, system, checkExtUrl) {
119
254
throw new Error ( 'Option "terminologyUrl" is not specified.' ) ;
120
255
}
121
256
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 ) ;
132
277
}
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 ;
145
278
}
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 ;
159
281
}
160
282
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
+
161
310
/**
162
311
* Returns array of linkIds of ancestor ResourceNodes and source ResourceNode
163
312
* starting with the linkId of the given node and ending with the topmost item's
@@ -176,6 +325,7 @@ function getLinkIds(node) {
176
325
return res ;
177
326
}
178
327
328
+
179
329
/**
180
330
* Returns a questionnaire item based on the linkIds array of the ancestor
181
331
* ResourceNodes and the target ResourceNode. If the questionnaire item is not
0 commit comments