Skip to content

Commit f9956dc

Browse files
Changes as per review
LF-2985
1 parent 90a4c64 commit f9956dc

12 files changed

+264
-166
lines changed

.eslintrc.yml

+3
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,9 @@ parserOptions:
88
ecmaVersion: 2016
99
requireConfigFile: false
1010
sourceType: module
11+
globals:
12+
Headers: true
13+
fetch: true
1114
rules:
1215
indent:
1316
- error

README.md

+7-5
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,8 @@ let tracefunction = function (x, label) {
153153
const res = fhirpath.evaluate(contextNode, path, environment, fhirpath_r4_model, { traceFn: tracefunction });
154154
```
155155
156+
### Asynchronous functions
157+
156158
Some FHIRPath functions may be asynchronous. These functions throw exceptions by default.
157159
To enable these functions, we need to pass the `async` option to `evaluate` or `compile`.
158160
`async=true` enables return of a Promise only for expressions containing asynchronous functions.
@@ -163,15 +165,15 @@ For example, using the `memberOf` function might look like this:
163165
fhirpath.evaluate(
164166
resource,
165167
"Observation.code.coding.where(memberOf('http://hl7.org/fhir/ValueSet/observation-vitalsignresult'))",
166-
{ terminologies: { validateVS }},
168+
{},
167169
model,
168-
{ async: true }
170+
{ async: true, serverUrl: 'https://lforms-fhir.nlm.nih.gov/baseR4' }
169171
)
170172
```
171173
172-
Please note that for the `memberOf` function to work we must define a general
173-
%terminologies object. See https://build.fhir.org/fhirpath.html#txapi and example
174-
in `test/async-functions.test.js`.
174+
Please note that for the `memberOf` function to work we must define a serverURL
175+
option that will be used by the %terminologies.validateVS function.
176+
(see https://build.fhir.org/fhirpath.html#txapi).
175177
176178
### User-defined functions
177179

bin/fhirpath

+4-1
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,10 @@ else {
4545
throw new Error('FHIR model must be one of '+supportedVersions);
4646
model = require('../fhir-context/'+options.model);
4747
}
48-
let res = fp.evaluate(resource, base ? {base, expression} : expression, context, model);
48+
let res = fp.evaluate(
49+
resource, base ? {base, expression} : expression, context, model,
50+
{ async: true }
51+
);
4952

5053
if (res instanceof Promise) {
5154
res.then((r) => printResult(expression, r))

index.d.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,8 @@ interface Options {
7777
resolveInternalTypes?: boolean
7878
traceFn?: (value: any, label: string) => void,
7979
userInvocationTable?: UserInvocationTable,
80-
async: false|true|'always'
80+
async: false|true|'always',
81+
serverUrl: string
8182
}
8283

8384
type Compile = (resource: any, context?: Context) => any[];

package-lock.json

+14-14
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/additional.js

+14-6
Original file line numberDiff line numberDiff line change
@@ -6,25 +6,33 @@ let engine = {};
66

77
/**
88
* Returns true if the code is a member of the given valueset.
9-
* @param {(string | Coding | CodeableConcept)[]} coll - input collection
9+
* @param {(string|Object)[]} coll - input collection with a single Coding,
10+
* CodeableConcept, or code element.
1011
* @param {string} valueset - value set URL
11-
* @return {Promise<{[p: string]: PromiseSettledResult<Awaited<*>>, [p: number]: PromiseSettledResult<Awaited<*>>, [p: symbol]: PromiseSettledResult<Awaited<*>>}>}
12+
* @return {Promise<boolean>|[]} - promise of a boolean value indicating that
13+
* there is one element in the input collection whose code is a member of the
14+
* specified value set.
1215
*/
1316
engine.memberOf = function (coll, valueset ) {
1417
if (!this.async) {
15-
throw new Error('The asynchronous function "memberOf" is not allowed. To enable asynchronous functions, use the async=true or async="always" option.');
18+
throw new Error('The asynchronous function "memberOf" is not allowed. ' +
19+
'To enable asynchronous functions, use the async=true or async="always"' +
20+
' option.');
1621
}
1722
// If the input is empty or has more than one value, the return value is empty
1823
if (coll.length !== 1 || coll[0] == null) {
1924
return [];
2025
}
2126

2227
if (typeof valueset === 'string' && /^https?:\/\/.*/.test(valueset)) {
23-
const terminologies = this.vars.terminologies || this.processedVars.terminologies;
24-
return terminologies.validateVS(valueset, util.valData(coll[0]), '');
28+
return this.terminologies.validateVS(valueset, util.valData(coll[0]), '')
29+
.then(params => {
30+
return params.parameter.find((p) => p.name === "result").valueBoolean;
31+
});
2532
}
2633

27-
// If the valueset cannot be resolved as an uri to a value set, the return value is empty.
34+
// If the valueset cannot be resolved as an uri to a value set,
35+
// the return value is empty.
2836
return [];
2937
};
3038

src/existence.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -27,18 +27,18 @@ engine.existsMacro = function(coll, expr) {
2727
};
2828

2929
engine.allMacro = function(coll, expr) {
30-
const promisses = [];
30+
const promises = [];
3131
for (let i=0, len=coll.length; i<len; ++i) {
3232
this.$index = i;
3333
const exprRes = expr(coll[i]);
3434
if (exprRes instanceof Promise) {
35-
promisses.push(exprRes);
35+
promises.push(exprRes);
3636
} else if(!util.isTrue(exprRes)){
3737
return [false];
3838
}
3939
}
40-
if (promisses.length) {
41-
return Promise.all(promisses).then(r => r.some(i => !util.isTrue(i)) ? [false] : [true]);
40+
if (promises.length) {
41+
return Promise.all(promises).then(r => r.some(i => !util.isTrue(i)) ? [false] : [true]);
4242
}
4343
return [true];
4444
};

src/fhirpath.js

+12-11
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@ const {
5454
FP_Type, ResourceNode, TypeInfo
5555
} = types;
5656
let makeResNode = ResourceNode.makeResNode;
57+
const Terminologies = require('./terminologies');
5758

5859
// * fn: handler
5960
// * arity: is index map with type signature
@@ -494,9 +495,7 @@ function doInvoke(ctx, fnName, data, rawParams){
494495
if(!invoc.arity){
495496
if(!rawParams){
496497
res = invoc.fn.call(ctx, data);
497-
return res instanceof Promise
498-
? res.then(r => util.arraify(r))
499-
: util.arraify(res);
498+
return util.resolveAndArraify(res);
500499
} else {
501500
throw new Error(fnName + " expects no params");
502501
}
@@ -519,15 +518,11 @@ function doInvoke(ctx, fnName, data, rawParams){
519518
if (params.some(p => p instanceof Promise)) {
520519
return Promise.all(params).then(p => {
521520
res = invoc.fn.apply(ctx, p);
522-
return res instanceof Promise
523-
? res.then(r => util.arraify(r))
524-
: util.arraify(res);
521+
return util.resolveAndArraify(res);
525522
});
526523
}
527524
res = invoc.fn.apply(ctx, params);
528-
return res instanceof Promise
529-
? res.then(r => util.arraify(r))
530-
: util.arraify(res);
525+
return util.resolveAndArraify(res);
531526
} else {
532527
console.log(fnName + " wrong arity: got " + paramsNumber );
533528
return [];
@@ -702,9 +697,10 @@ function parse(path) {
702697
* @param {object} [options.userInvocationTable] - a user invocation table used
703698
* to replace any existing or define new functions.
704699
* @param {boolean|string} [options.async] - defines how to support asynchronous functions:
705-
* false or similar to false, e.g. undefined, null, or 0 (default) - throw an exception,
706-
* true or similar to true - return Promise, only for asynchronous functions,
700+
* false or similar to false, e.g. undefined, null, or 0 (default) - throw an exception;
701+
* true or similar to true - return Promise only for asynchronous functions;
707702
* "always" - return Promise always.
703+
* @param {string} [options.serverUrl] - a URL that points to a FHIR RESTful API.
708704
*/
709705
function applyParsedPath(resource, parsedPath, context, model, options) {
710706
constants.reset();
@@ -728,6 +724,9 @@ function applyParsedPath(resource, parsedPath, context, model, options) {
728724
if (options.async) {
729725
ctx.async = options.async;
730726
}
727+
if (options.serverUrl) {
728+
ctx.terminologies = new Terminologies(options.serverUrl);
729+
}
731730
const res = engine.doEval(ctx, dataRoot, parsedPath.children[0]);
732731
return res instanceof Promise
733732
? res.then(r => prepareEvalResult(r, options))
@@ -822,6 +821,7 @@ function resolveInternalTypes(val) {
822821
* false or similar to false, e.g. undefined, null, or 0 (default) - throw an exception,
823822
* true or similar to true - return Promise, only for asynchronous functions,
824823
* "always" - return Promise always.
824+
* @param {string} [options.serverUrl] - a URL that points to a FHIR RESTful API.
825825
*/
826826
function evaluate(fhirData, path, context, model, options) {
827827
return compile(path, model, options)(fhirData, context);
@@ -849,6 +849,7 @@ function evaluate(fhirData, path, context, model, options) {
849849
* false or similar to false, e.g. undefined, null, or 0 (default) - throw an exception,
850850
* true or similar to true - return Promise, only for asynchronous functions,
851851
* "always" - return Promise always.
852+
* @param {string} [options.serverUrl] - a URL that points to a FHIR RESTful API.
852853
*/
853854
function compile(path, model, options) {
854855
options = {

src/filtering.js

+11-3
Original file line numberDiff line numberDiff line change
@@ -54,18 +54,26 @@ engine.repeatMacro = function(parentData, expr, res = [], unique = {}) {
5454
return Promise.all(newItems).then(items => {
5555
items = [].concat(...items);
5656
if (items.length) {
57-
return engine.repeatMacro(getUniqItems(items, unique, res), expr, res, unique);
57+
return engine.repeatMacro(getNewItems(items, unique, res), expr, res, unique);
5858
}
5959
return res;
6060
});
6161
} else if (newItems.length) {
62-
return engine.repeatMacro(getUniqItems(newItems, unique, res), expr, res, unique);
62+
return engine.repeatMacro(getNewItems(newItems, unique, res), expr, res, unique);
6363
} else {
6464
return res;
6565
}
6666
};
6767

68-
function getUniqItems(items, unique, res) {
68+
/**
69+
* Returns new items from the input array that are not in the hash of existing
70+
* unique items and adds them to the result array.
71+
* @param {Array<*>} items - inout array.
72+
* @param {{[key: string]: *}} unique - hash of existing unique items
73+
* @param {Array<*>} res - result array.
74+
* @return {Array<*>}
75+
*/
76+
function getNewItems(items, unique, res) {
6977
const newItems = items.filter(item => {
7078
const key = hashObject(item);
7179
const isUnique = !unique[key];

0 commit comments

Comments
 (0)