Skip to content

Commit

Permalink
Merge branch 'master' into other-dep-resource-types-2
Browse files Browse the repository at this point in the history
  • Loading branch information
cmoesel authored Dec 30, 2024
2 parents 34c31f3 + 97e5efd commit 9c3c838
Show file tree
Hide file tree
Showing 11 changed files with 396 additions and 44 deletions.
1 change: 1 addition & 0 deletions src/app.ts
Original file line number Diff line number Diff line change
Expand Up @@ -269,6 +269,7 @@ async function runBuild(input: string, program: OptionValues, helpText: string)
config = readConfig(originalInput);
updateConfig(config, program);
tank = fillTank(rawFSH, config);
tank.checkDuplicateNameEntities();
} catch (e) {
// If no errors have been logged yet, log this exception so the user knows why we're exiting
if (stats.numError === 0) {
Expand Down
6 changes: 3 additions & 3 deletions src/fhirdefs/impliedExtensions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ export function isImpliedExtension(url: string): boolean {
* @returns {any} a JSON StructureDefinition representing the implied extension
*/
export function materializeImpliedExtension(url: string, defs: FHIRDefinitions): any {
const match = url.match(IMPLIED_EXTENSION_REGEX);
const match = decodeURI(url).match(IMPLIED_EXTENSION_REGEX);
if (match == null) {
logger.error(
`Unsupported extension URL: ${url}. Extension URLs for converting between versions of ` +
Expand Down Expand Up @@ -260,7 +260,7 @@ function applyMetadataToExtension(
toExt: StructureDefinition
): void {
const elementId = fromED.id ?? fromED.path;
toExt.id = `extension-${elementId}`;
toExt.id = encodeURIComponent(`extension-${elementId}`);
toExt.url = `http://hl7.org/fhir/${fromVersion}/StructureDefinition/${toExt.id}`;
toExt.version = fromSD.fhirVersion;
toExt.name = `Extension_${elementId.replace(/[^A-Za-z0-9]/g, '_')}`;
Expand Down Expand Up @@ -457,7 +457,7 @@ function applyToExtensionElement(
const id: string = e.id ?? e.path;
const tail = id.slice(edId.length + 1);
if (tail.indexOf('.') === -1 && !IGNORED_CHILDREN.includes(tail)) {
const slice = toED.addSlice(tail);
const slice = toED.addSlice(encodeURIComponent(tail));
applyMetadataToElement(e, slice);
slice.type = [new ElementDefinitionType('Extension')];
slice.unfold(defs);
Expand Down
9 changes: 9 additions & 0 deletions src/fhirtypes/StructureDefinition.ts
Original file line number Diff line number Diff line change
Expand Up @@ -909,6 +909,15 @@ export class StructureDefinition {
);
}
}
if (
!matchingSlice &&
pathPart.brackets?.every(p => p === decodeURIComponent(p)) &&
pathPart.brackets?.some(p => p !== encodeURIComponent(p))
) {
const encodedPathPart = cloneDeep(pathPart);
encodedPathPart.brackets = encodedPathPart.brackets.map(p => encodeURIComponent(p));
return this.findMatchingSlice(fhirPathString, encodedPathPart, elements, fisher);
}
// NOTE: This function will assume the 'brackets' field contains information about slices. Even
// if you search for foo[sliceName][refName], this will try to find a re-slice
// sliceName/refName. To find the matching element for foo[sliceName][refName], you must
Expand Down
45 changes: 45 additions & 0 deletions src/import/FSHTank.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import {
} from '../fhirtypes/common';
import flatMap from 'lodash/flatMap';
import { getNonInstanceValueFromRules } from '../fshtypes/common';
import { logger } from '../utils/FSHLogger';

export class FSHTank implements Fishable {
constructor(
Expand Down Expand Up @@ -139,6 +140,50 @@ export class FSHTank implements Fishable {
return undefined;
}

checkDuplicateNameEntities(): undefined {
const allEntities = [
...this.getAllStructureDefinitions(),
...this.getAllInstances(),
...this.getAllMappings(),
...this.getAllInvariants(),
...this.getAllValueSets(),
...this.getAllCodeSystems(),
...this.getAllRuleSets(),
...this.getAllExtensions()
];

const duplicateEntities = new Set();
allEntities.forEach(entity => {
if (
this.docs.some(
doc =>
(doc.profiles.has(entity.name) && doc.profiles.get(entity.name) != entity) ||
(doc.extensions.has(entity.name) && doc.extensions.get(entity.name) != entity) ||
(doc.logicals.has(entity.name) && doc.logicals.get(entity.name) != entity) ||
(doc.resources.has(entity.name) && doc.resources.get(entity.name) != entity) ||
(doc.instances.has(entity.name) && doc.instances.get(entity.name) != entity) ||
(doc.mappings.has(entity.name) && doc.mappings.get(entity.name) != entity) ||
(doc.invariants.has(entity.name) && doc.invariants.get(entity.name) != entity) ||
(doc.valueSets.has(entity.name) && doc.valueSets.get(entity.name) != entity) ||
(doc.codeSystems.has(entity.name) && doc.codeSystems.get(entity.name) != entity) ||
(doc.ruleSets.has(entity.name) && doc.ruleSets.get(entity.name) != entity)
)
) {
duplicateEntities.add(entity.name);
}
});

if (duplicateEntities.size > 0) {
logger.warn(
'Detected FSH entity definitions with duplicate names. While FSH allows for duplicate ' +
'names across entity types, they can lead to ambiguous results when referring to these ' +
'entities by name elsewhere (e.g., in references). Consider using unique names in FSH ' +
'declarations and assigning duplicated names using caret assignment rules instead. ' +
`Detected duplicate names: ${Array.from(duplicateEntities).join(', ')}.`
);
}
}

fish(
item: string,
...types: Type[]
Expand Down
1 change: 1 addition & 0 deletions src/run/FshToFhir.ts
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ export async function fshToFhir(
rawFSHes.push(new RawFSH(input));
}
const tank = fillTank(rawFSHes, config);
tank.checkDuplicateNameEntities();

// process FSH text into FHIR
const outPackage = exportFHIR(tank, defs);
Expand Down
58 changes: 30 additions & 28 deletions src/utils/PathUtils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,48 +2,50 @@ import { flatten } from 'lodash';
import { InstanceDefinition, PathPart } from '../fhirtypes';
import { splitOnPathPeriods } from '../fhirtypes/common';
import { CaretValueRule, Rule } from '../fshtypes/rules';
import { SourceInfo } from '../fshtypes/FshEntity';
import { logger } from './FSHLogger';

/**
* Parses a FSH Path into a more easily usable form
* @param {string} fshPath - A syntactically valid path in FSH
* @returns {PathPart[]} an array of PathParts that is easier to work with
*/
export function parseFSHPath(fshPath: string): PathPart[] {
export function parseFSHPath(fshPath: string, sourceInfo?: SourceInfo): PathPart[] {
const pathParts: PathPart[] = [];
const seenSlices: string[] = [];
const indexRegex = /^[0-9]+$/;
const splitPath = fshPath === '.' ? [fshPath] : splitOnPathPeriods(fshPath);
for (const pathPart of splitPath) {
const splitPathPart = pathPart.split('[');
if (splitPathPart.length === 1 || pathPart.endsWith('[x]')) {
// There are no brackets, or the brackets are for a choice, so just push on the name
pathParts.push({ base: pathPart });
} else {
// We have brackets, let's save the bracket info
let fhirPathBase = splitPathPart[0];
// Get the bracket elements and slice off the trailing ']'
let brackets = splitPathPart.slice(1).map(s => s.slice(0, -1));
// Get rid of any remaining [x] elements in the brackets
if (brackets[0] === 'x') {
fhirPathBase += '[x]';
brackets = brackets.slice(1);
const parsedPart: { base: string; brackets?: string[]; slices?: string[] } = {
base: pathPart.match(/^([^\[]+(\[x\])?)/)?.[0] ?? ''
};
if (pathPart.length > parsedPart.base.length) {
// Get the content from the outermost bracket pairs. (?:[^\[\]]*) ensures we don't
// match nested closing brackets (thank you, claude.ai)
parsedPart.brackets = Array.from(
pathPart.slice(parsedPart.base.length).matchAll(/\[([^\[\]]|\[(?:[^\[\]]*)\])*\]/g)
).map(match => match[0].slice(1, -1));
seenSlices.push(
...parsedPart.brackets.filter(b => !indexRegex.test(b) && !(b === '+' || b === '='))
);
if (seenSlices.length > 0) {
parsedPart.slices = [...seenSlices];
}
brackets.forEach(bracket => {
if (!indexRegex.test(bracket) && !(bracket === '+' || bracket === '=')) {
seenSlices.push(bracket);
const parsedPartLength =
parsedPart.base.length +
parsedPart.brackets
.map(b => b.length + 2)
.reduce((total: number, current: number) => total + current, 0);
if (pathPart.length !== parsedPartLength) {
const message = `Error processing path due to unmatched brackets: ${fshPath}. `;
if (sourceInfo) {
logger.error(message, sourceInfo);
} else {
logger.error(message);
}
});
if (seenSlices.length > 0) {
pathParts.push({
base: fhirPathBase,
brackets: brackets,
slices: [...seenSlices]
});
} else {
pathParts.push({ base: fhirPathBase, brackets: brackets });
}
}
pathParts.push(parsedPart);
}
return pathParts;
}
Expand Down Expand Up @@ -208,11 +210,11 @@ export function resolveSoftIndexing(rules: Array<Rule | CaretValueRule>, strict
// Parsing and separating rules by base name and bracket indexes
const parsedRules = rules.map(rule => {
const parsedPath: { path: PathPart[]; caretPath?: PathPart[] } = {
path: parseFSHPath(rule.path)
path: parseFSHPath(rule.path, rule.sourceInfo)
};
// If we have a CaretValueRule, we'll need a second round of parsing for the caret path
if (rule instanceof CaretValueRule) {
parsedPath.caretPath = parseFSHPath(rule.caretPath);
parsedPath.caretPath = parseFSHPath(rule.caretPath, rule.sourceInfo);
}
return parsedPath;
});
Expand Down
124 changes: 113 additions & 11 deletions test/fhirdefs/impliedExtension.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -814,6 +814,108 @@ describe('impliedExtensions', () => {
expect(loggerSpy.getAllLogs('error')).toHaveLength(0);
});

it('should materialize a simple R5 extension for a choice element and properly encode the brackets', () => {
const ext = materializeImpliedExtension(
'http://hl7.org/fhir/5.0/StructureDefinition/extension-Questionnaire.versionAlgorithm[x]',
defs
);
expect(ext).toBeDefined();
expect(ext).toMatchObject({
resourceType: 'StructureDefinition',
id: 'extension-Questionnaire.versionAlgorithm%5Bx%5D',
url: 'http://hl7.org/fhir/5.0/StructureDefinition/extension-Questionnaire.versionAlgorithm%5Bx%5D',
version: '5.0.0',
name: 'Extension_Questionnaire_versionAlgorithm_x_',
title: 'Implied extension for Questionnaire.versionAlgorithm[x]',
status: 'active',
description: 'Implied extension for Questionnaire.versionAlgorithm[x]',
fhirVersion: '4.0.1',
kind: 'complex-type',
abstract: false,
context: [{ type: 'element', expression: 'Element' }],
type: 'Extension',
baseDefinition: 'http://hl7.org/fhir/StructureDefinition/Extension',
derivation: 'constraint'
});

const diffUrl = ext.differential?.element?.find((e: any) => e.id === 'Extension.url');
expect(diffUrl).toEqual({
id: 'Extension.url',
path: 'Extension.url',
fixedUri:
'http://hl7.org/fhir/5.0/StructureDefinition/extension-Questionnaire.versionAlgorithm%5Bx%5D'
});
const snapUrl = ext.snapshot?.element?.find((e: any) => e.id === 'Extension.url');
expect(snapUrl).toMatchObject(diffUrl);

const diffValue = ext.differential?.element?.find((e: any) => e.id === 'Extension.value[x]');
expect(diffValue).toEqual({
id: 'Extension.value[x]',
path: 'Extension.value[x]',
binding: {
strength: 'extensible',
valueSet: 'http://hl7.org/fhir/ValueSet/version-algorithm'
},
type: [{ code: 'string' }, { code: 'Coding' }]
});
const snapValue = ext.snapshot?.element?.find((e: any) => e.id === 'Extension.value[x]');
expect(snapValue).toMatchObject(diffValue);

expect(loggerSpy.getAllLogs('warn')).toHaveLength(0);
expect(loggerSpy.getAllLogs('error')).toHaveLength(0);
});

it('should materialize a simple R5 extension for a choice element with brackets already encoded', () => {
const ext = materializeImpliedExtension(
'http://hl7.org/fhir/5.0/StructureDefinition/extension-Questionnaire.versionAlgorithm%5Bx%5D',
defs
);
expect(ext).toBeDefined();
expect(ext).toMatchObject({
resourceType: 'StructureDefinition',
id: 'extension-Questionnaire.versionAlgorithm%5Bx%5D',
url: 'http://hl7.org/fhir/5.0/StructureDefinition/extension-Questionnaire.versionAlgorithm%5Bx%5D',
version: '5.0.0',
name: 'Extension_Questionnaire_versionAlgorithm_x_',
title: 'Implied extension for Questionnaire.versionAlgorithm[x]',
status: 'active',
description: 'Implied extension for Questionnaire.versionAlgorithm[x]',
fhirVersion: '4.0.1',
kind: 'complex-type',
abstract: false,
context: [{ type: 'element', expression: 'Element' }],
type: 'Extension',
baseDefinition: 'http://hl7.org/fhir/StructureDefinition/Extension',
derivation: 'constraint'
});

const diffUrl = ext.differential?.element?.find((e: any) => e.id === 'Extension.url');
expect(diffUrl).toEqual({
id: 'Extension.url',
path: 'Extension.url',
fixedUri:
'http://hl7.org/fhir/5.0/StructureDefinition/extension-Questionnaire.versionAlgorithm%5Bx%5D'
});
const snapUrl = ext.snapshot?.element?.find((e: any) => e.id === 'Extension.url');
expect(snapUrl).toMatchObject(diffUrl);

const diffValue = ext.differential?.element?.find((e: any) => e.id === 'Extension.value[x]');
expect(diffValue).toEqual({
id: 'Extension.value[x]',
path: 'Extension.value[x]',
binding: {
strength: 'extensible',
valueSet: 'http://hl7.org/fhir/ValueSet/version-algorithm'
},
type: [{ code: 'string' }, { code: 'Coding' }]
});
const snapValue = ext.snapshot?.element?.find((e: any) => e.id === 'Extension.value[x]');
expect(snapValue).toMatchObject(diffValue);

expect(loggerSpy.getAllLogs('warn')).toHaveLength(0);
expect(loggerSpy.getAllLogs('error')).toHaveLength(0);
});

it('should materialize a complex R5 extension', () => {
const ext = materializeImpliedExtension(
'http://hl7.org/fhir/5.0/StructureDefinition/extension-MedicationRequest.substitution',
Expand Down Expand Up @@ -893,12 +995,12 @@ describe('impliedExtensions', () => {
});

const diffAllowed = ext.differential?.element?.find(
(e: any) => e.id === 'Extension.extension:allowed[x]'
(e: any) => e.id === 'Extension.extension:allowed%5Bx%5D'
);
expect(diffAllowed).toEqual({
id: 'Extension.extension:allowed[x]',
id: 'Extension.extension:allowed%5Bx%5D',
path: 'Extension.extension',
sliceName: 'allowed[x]',
sliceName: 'allowed%5Bx%5D',
short: 'Whether substitution is allowed or not',
definition:
'True if the prescriber allows a different drug to be dispensed from what was prescribed.',
Expand All @@ -910,28 +1012,28 @@ describe('impliedExtensions', () => {
type: [{ code: 'Extension' }]
});
const snapAllowed = ext.snapshot?.element?.find(
(e: any) => e.id === 'Extension.extension:allowed[x]'
(e: any) => e.id === 'Extension.extension:allowed%5Bx%5D'
);
expect(snapAllowed).toMatchObject(diffAllowed);

const diffAllowedURL = ext.differential?.element?.find(
(e: any) => e.id === 'Extension.extension:allowed[x].url'
(e: any) => e.id === 'Extension.extension:allowed%5Bx%5D.url'
);
expect(diffAllowedURL).toEqual({
id: 'Extension.extension:allowed[x].url',
id: 'Extension.extension:allowed%5Bx%5D.url',
path: 'Extension.extension.url',
fixedUri: 'allowed[x]'
fixedUri: 'allowed%5Bx%5D'
});
const snapAllowedURL = ext.snapshot?.element?.find(
(e: any) => e.id === 'Extension.extension:allowed[x].url'
(e: any) => e.id === 'Extension.extension:allowed%5Bx%5D.url'
);
expect(snapAllowedURL).toMatchObject(diffAllowedURL);

const diffAllowedValue = ext.differential?.element?.find(
(e: any) => e.id === 'Extension.extension:allowed[x].value[x]'
(e: any) => e.id === 'Extension.extension:allowed%5Bx%5D.value[x]'
);
expect(diffAllowedValue).toEqual({
id: 'Extension.extension:allowed[x].value[x]',
id: 'Extension.extension:allowed%5Bx%5D.value[x]',
path: 'Extension.extension.value[x]',
type: [{ code: 'boolean' }, { code: 'CodeableConcept' }],
binding: {
Expand All @@ -941,7 +1043,7 @@ describe('impliedExtensions', () => {
}
});
const snapAllowedValue = ext.snapshot?.element?.find(
(e: any) => e.id === 'Extension.extension:allowed[x].value[x]'
(e: any) => e.id === 'Extension.extension:allowed%5Bx%5D.value[x]'
);
expect(snapAllowedValue).toMatchObject(diffAllowedValue);

Expand Down
Loading

0 comments on commit 9c3c838

Please sign in to comment.