@openhi/types 0.0.24 → 0.0.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/lib/index.d.ts CHANGED
@@ -14,3 +14,4 @@
14
14
  export * from "./data";
15
15
  export * from "./control";
16
16
  export * from "./summary";
17
+ export * from "./sort-key";
package/lib/index.js CHANGED
@@ -30,4 +30,5 @@ Object.defineProperty(exports, "__esModule", { value: true });
30
30
  __exportStar(require("./data"), exports);
31
31
  __exportStar(require("./control"), exports);
32
32
  __exportStar(require("./summary"), exports);
33
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOzs7Ozs7Ozs7Ozs7R0FZRztBQUNILHlDQUF1QjtBQUN2Qiw0Q0FBMEI7QUFDMUIsNENBQTBCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAb3BlbmhpL3R5cGVzIOKAlCBGSElSIGFuZCBPcGVuSEkgdHlwZXMgYnkgZG9tYWluLlxuICpcbiAqIERhdGEgcGxhbmUgaXMgZ2VuZXJhdG9yIG91dHB1dCAoYHNyYy9kYXRhL2AsIG9uZSBtb2R1bGUgcGVyIEZISVIgUjRcbiAqIFN0cnVjdHVyZURlZmluaXRpb24sIGJhcnJlbCBpbiBgLi9kYXRhL2luZGV4LnRzYCkuIEJhc2UgcmVzb3VyY2VzXG4gKiAoUmVzb3VyY2UsIERvbWFpblJlc291cmNlLCBFbGVtZW50LCBCYWNrYm9uZUVsZW1lbnQsIEV4dGVuc2lvbiwgZXRjLilcbiAqIGxpdmUgaW4gdGhhdCBzYW1lIHRyZWUg4oCUIHRoZSBnZW5lcmF0b3IgdHJlYXRzIHRoZW0gbGlrZSBhbnkgb3RoZXIgU0QuXG4gKlxuICogQ29udHJvbCBwbGFuZSBpcyBnZW5lcmF0b3Igb3V0cHV0IChgc3JjL2NvbnRyb2wvYCk6IFVzZXIsIFRlbmFudCxcbiAqIFdvcmtzcGFjZSwgTWVtYmVyc2hpcCwgUm9sZUFzc2lnbm1lbnQsIFJvbGUsIHBsdXMgdGhlXG4gKiBgQ29udHJvbFBsYW5lUm9sZUNvZGVgIHRlcm1pbm9sb2d5LiBBdXRob3JlZCBpbiBGU0ggdW5kZXJcbiAqIGBmc2gvaW5wdXQvZnNoL2NvbnRyb2wtcGxhbmUvYCBhbmQgZW1pdHRlZCBieSB0aGUgc2FtZSBwaXBlbGluZS5cbiAqL1xuZXhwb3J0ICogZnJvbSBcIi4vZGF0YVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29udHJvbFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vc3VtbWFyeVwiO1xuIl19
33
+ __exportStar(require("./sort-key"), exports);
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOzs7Ozs7Ozs7Ozs7R0FZRztBQUNILHlDQUF1QjtBQUN2Qiw0Q0FBMEI7QUFDMUIsNENBQTBCO0FBQzFCLDZDQUEyQiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQG9wZW5oaS90eXBlcyDigJQgRkhJUiBhbmQgT3BlbkhJIHR5cGVzIGJ5IGRvbWFpbi5cbiAqXG4gKiBEYXRhIHBsYW5lIGlzIGdlbmVyYXRvciBvdXRwdXQgKGBzcmMvZGF0YS9gLCBvbmUgbW9kdWxlIHBlciBGSElSIFI0XG4gKiBTdHJ1Y3R1cmVEZWZpbml0aW9uLCBiYXJyZWwgaW4gYC4vZGF0YS9pbmRleC50c2ApLiBCYXNlIHJlc291cmNlc1xuICogKFJlc291cmNlLCBEb21haW5SZXNvdXJjZSwgRWxlbWVudCwgQmFja2JvbmVFbGVtZW50LCBFeHRlbnNpb24sIGV0Yy4pXG4gKiBsaXZlIGluIHRoYXQgc2FtZSB0cmVlIOKAlCB0aGUgZ2VuZXJhdG9yIHRyZWF0cyB0aGVtIGxpa2UgYW55IG90aGVyIFNELlxuICpcbiAqIENvbnRyb2wgcGxhbmUgaXMgZ2VuZXJhdG9yIG91dHB1dCAoYHNyYy9jb250cm9sL2ApOiBVc2VyLCBUZW5hbnQsXG4gKiBXb3Jrc3BhY2UsIE1lbWJlcnNoaXAsIFJvbGVBc3NpZ25tZW50LCBSb2xlLCBwbHVzIHRoZVxuICogYENvbnRyb2xQbGFuZVJvbGVDb2RlYCB0ZXJtaW5vbG9neS4gQXV0aG9yZWQgaW4gRlNIIHVuZGVyXG4gKiBgZnNoL2lucHV0L2ZzaC9jb250cm9sLXBsYW5lL2AgYW5kIGVtaXR0ZWQgYnkgdGhlIHNhbWUgcGlwZWxpbmUuXG4gKi9cbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2NvbnRyb2xcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N1bW1hcnlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3NvcnQta2V5XCI7XG4iXX0=
@@ -0,0 +1,44 @@
1
+ /**
2
+ * Pure projection of a FHIR R4 resource into its GSI1 sort key per DR-004.
3
+ *
4
+ * - Labeled types (entries in `LABEL_PATHS`): `<normalizedLabel>#<id>`
5
+ * so list endpoints sort alphabetically and `BEGINS_WITH` serves
6
+ * prefix queries like `?name=Sm`.
7
+ * - Unlabeled types: `<ISO-8601 lastUpdated>#<id>` so list endpoints
8
+ * serve reverse-chronological with `ScanIndexForward=false`.
9
+ *
10
+ * Pure: no I/O, no DynamoDB access, deterministic over the resource and
11
+ * the committed `LABEL_PATHS` table. Consumed by the data-plane writer
12
+ * before each `PutItem` / `UpdateItem`.
13
+ */
14
+ import type { FhirResourceLike } from "../summary/extract-summary";
15
+ /**
16
+ * Returns the GSI1 sort-key value for `resource`. The id is the
17
+ * tie-breaker suffix; callers must pass a resource that carries an `id`
18
+ * for an `<id>` segment to be present, but `extractSortKey` returns
19
+ * `<label>#` (or `<lastUpdated>#`) with an empty trailing segment when id
20
+ * is missing rather than throwing — the caller validates id at the
21
+ * write-path entry.
22
+ *
23
+ * Resolution order:
24
+ * 1. `LABEL_PATHS[resourceType]` — the curated, type-aware extraction.
25
+ * 2. Introspection fallback per ADR-011 / DR-004: try `resource.name`
26
+ * (when it's a non-empty string), then `resource.title` (same), so
27
+ * spec types with a string `name`/`title` participate in alphabetical
28
+ * sort and `BEGINS_WITH` even if they aren't in `LABEL_PATHS` yet.
29
+ * 3. Unlabeled fallback — `<ISO-8601 lastUpdated>#<id>`.
30
+ */
31
+ export declare function extractSortKey(resource: FhirResourceLike): string;
32
+ /**
33
+ * Normalize a natural-language label per DR-004:
34
+ * 1. ASCII-fold common Latin diacritics (ñ→n, é→e, ø→o, …).
35
+ * 2. Lowercase (DynamoDB byte-order sort).
36
+ * 3. Strip apostrophes and periods (so `O'Connor` → `oconnor`).
37
+ * 4. Replace hyphens with spaces (so `García-López` → `garcia lopez`).
38
+ * 5. Strip other punctuation, collapse runs of whitespace, trim.
39
+ *
40
+ * Non-Latin scripts pass through unchanged after ASCII-fold so same-script
41
+ * items sort together. Locale-aware collation is a downstream client
42
+ * concern.
43
+ */
44
+ export declare function normalizeLabel(input: string): string;
@@ -0,0 +1,130 @@
1
+ "use strict";
2
+ /**
3
+ * Pure projection of a FHIR R4 resource into its GSI1 sort key per DR-004.
4
+ *
5
+ * - Labeled types (entries in `LABEL_PATHS`): `<normalizedLabel>#<id>`
6
+ * so list endpoints sort alphabetically and `BEGINS_WITH` serves
7
+ * prefix queries like `?name=Sm`.
8
+ * - Unlabeled types: `<ISO-8601 lastUpdated>#<id>` so list endpoints
9
+ * serve reverse-chronological with `ScanIndexForward=false`.
10
+ *
11
+ * Pure: no I/O, no DynamoDB access, deterministic over the resource and
12
+ * the committed `LABEL_PATHS` table. Consumed by the data-plane writer
13
+ * before each `PutItem` / `UpdateItem`.
14
+ */
15
+ Object.defineProperty(exports, "__esModule", { value: true });
16
+ exports.extractSortKey = extractSortKey;
17
+ exports.normalizeLabel = normalizeLabel;
18
+ const label_paths_1 = require("./label-paths");
19
+ /**
20
+ * Returns the GSI1 sort-key value for `resource`. The id is the
21
+ * tie-breaker suffix; callers must pass a resource that carries an `id`
22
+ * for an `<id>` segment to be present, but `extractSortKey` returns
23
+ * `<label>#` (or `<lastUpdated>#`) with an empty trailing segment when id
24
+ * is missing rather than throwing — the caller validates id at the
25
+ * write-path entry.
26
+ *
27
+ * Resolution order:
28
+ * 1. `LABEL_PATHS[resourceType]` — the curated, type-aware extraction.
29
+ * 2. Introspection fallback per ADR-011 / DR-004: try `resource.name`
30
+ * (when it's a non-empty string), then `resource.title` (same), so
31
+ * spec types with a string `name`/`title` participate in alphabetical
32
+ * sort and `BEGINS_WITH` even if they aren't in `LABEL_PATHS` yet.
33
+ * 3. Unlabeled fallback — `<ISO-8601 lastUpdated>#<id>`.
34
+ */
35
+ function extractSortKey(resource) {
36
+ const id = typeof resource.id === "string" ? resource.id : "";
37
+ const source = label_paths_1.LABEL_PATHS[resource.resourceType];
38
+ if (source) {
39
+ const label = extractLabel(resource, source);
40
+ if (label.length > 0) {
41
+ return `${normalizeLabel(label)}#${id}`;
42
+ }
43
+ }
44
+ const introspected = introspectStringField(resource, "name") ??
45
+ introspectStringField(resource, "title");
46
+ if (introspected !== undefined) {
47
+ return `${normalizeLabel(introspected)}#${id}`;
48
+ }
49
+ const lastUpdated = typeof resource.meta?.lastUpdated === "string"
50
+ ? resource.meta.lastUpdated
51
+ : "";
52
+ return `${lastUpdated}#${id}`;
53
+ }
54
+ /**
55
+ * Returns the value of `field` on `resource` only if it's a non-empty string.
56
+ * Guards against HumanName arrays (Patient.name) and other non-string shapes
57
+ * so the introspection fallback never produces a key from a typed field it
58
+ * doesn't understand — that case stays on the lastUpdated fallback.
59
+ */
60
+ function introspectStringField(resource, field) {
61
+ const value = resource[field];
62
+ if (typeof value === "string" && value.length > 0)
63
+ return value;
64
+ return undefined;
65
+ }
66
+ function extractLabel(resource, source) {
67
+ if (source.kind === "humanName") {
68
+ const names = resource[source.path];
69
+ if (!Array.isArray(names))
70
+ return "";
71
+ const first = names[0];
72
+ if (!first || typeof first !== "object")
73
+ return "";
74
+ const name = first;
75
+ const family = typeof name.family === "string" ? name.family : "";
76
+ const given = Array.isArray(name.given) && typeof name.given[0] === "string"
77
+ ? name.given[0]
78
+ : "";
79
+ if (family.length === 0 && given.length === 0)
80
+ return "";
81
+ return `${family},${given}`;
82
+ }
83
+ if (source.kind === "string") {
84
+ const value = resource[source.path];
85
+ return typeof value === "string" ? value : "";
86
+ }
87
+ if (source.kind === "codeDisplay") {
88
+ const code = resource[source.path];
89
+ if (!code || typeof code !== "object")
90
+ return "";
91
+ const cc = code;
92
+ if (Array.isArray(cc.coding)) {
93
+ const first = cc.coding[0];
94
+ if (first && typeof first === "object") {
95
+ const display = first.display;
96
+ if (typeof display === "string" && display.length > 0)
97
+ return display;
98
+ }
99
+ }
100
+ return typeof cc.text === "string" ? cc.text : "";
101
+ }
102
+ return "";
103
+ }
104
+ /**
105
+ * Normalize a natural-language label per DR-004:
106
+ * 1. ASCII-fold common Latin diacritics (ñ→n, é→e, ø→o, …).
107
+ * 2. Lowercase (DynamoDB byte-order sort).
108
+ * 3. Strip apostrophes and periods (so `O'Connor` → `oconnor`).
109
+ * 4. Replace hyphens with spaces (so `García-López` → `garcia lopez`).
110
+ * 5. Strip other punctuation, collapse runs of whitespace, trim.
111
+ *
112
+ * Non-Latin scripts pass through unchanged after ASCII-fold so same-script
113
+ * items sort together. Locale-aware collation is a downstream client
114
+ * concern.
115
+ */
116
+ function normalizeLabel(input) {
117
+ return asciiFold(input)
118
+ .toLowerCase()
119
+ .replace(/['’.]/g, "")
120
+ .replace(/-/g, " ")
121
+ .replace(/[^\p{L}\p{N},\s]/gu, " ")
122
+ .replace(/\s+/g, " ")
123
+ .trim();
124
+ }
125
+ function asciiFold(input) {
126
+ // Decompose accents and strip combining marks; covers the common Latin
127
+ // diacritics (é, ñ, ü, …). Non-Latin scripts are unaffected.
128
+ return input.normalize("NFKD").replace(/\p{M}+/gu, "");
129
+ }
130
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extract-sort-key.js","sourceRoot":"","sources":["../../src/sort-key/extract-sort-key.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Pure projection of a FHIR R4 resource into its GSI1 sort key per DR-004.\n *\n * - Labeled types (entries in `LABEL_PATHS`): `<normalizedLabel>#<id>`\n *   so list endpoints sort alphabetically and `BEGINS_WITH` serves\n *   prefix queries like `?name=Sm`.\n * - Unlabeled types: `<ISO-8601 lastUpdated>#<id>` so list endpoints\n *   serve reverse-chronological with `ScanIndexForward=false`.\n *\n * Pure: no I/O, no DynamoDB access, deterministic over the resource and\n * the committed `LABEL_PATHS` table. Consumed by the data-plane writer\n * before each `PutItem` / `UpdateItem`.\n */\n\nimport { LABEL_PATHS, type LabelSource } from \"./label-paths\";\nimport type { FhirResourceLike } from \"../summary/extract-summary\";\n\n/**\n * Returns the GSI1 sort-key value for `resource`. The id is the\n * tie-breaker suffix; callers must pass a resource that carries an `id`\n * for an `<id>` segment to be present, but `extractSortKey` returns\n * `<label>#` (or `<lastUpdated>#`) with an empty trailing segment when id\n * is missing rather than throwing — the caller validates id at the\n * write-path entry.\n *\n * Resolution order:\n * 1. `LABEL_PATHS[resourceType]` — the curated, type-aware extraction.\n * 2. Introspection fallback per ADR-011 / DR-004: try `resource.name`\n *    (when it's a non-empty string), then `resource.title` (same), so\n *    spec types with a string `name`/`title` participate in alphabetical\n *    sort and `BEGINS_WITH` even if they aren't in `LABEL_PATHS` yet.\n * 3. Unlabeled fallback — `<ISO-8601 lastUpdated>#<id>`.\n */\nexport function extractSortKey(resource: FhirResourceLike): string {\n  const id = typeof resource.id === \"string\" ? resource.id : \"\";\n\n  const source = LABEL_PATHS[resource.resourceType];\n  if (source) {\n    const label = extractLabel(resource, source);\n    if (label.length > 0) {\n      return `${normalizeLabel(label)}#${id}`;\n    }\n  }\n\n  const introspected =\n    introspectStringField(resource, \"name\") ??\n    introspectStringField(resource, \"title\");\n  if (introspected !== undefined) {\n    return `${normalizeLabel(introspected)}#${id}`;\n  }\n\n  const lastUpdated =\n    typeof resource.meta?.lastUpdated === \"string\"\n      ? resource.meta.lastUpdated\n      : \"\";\n  return `${lastUpdated}#${id}`;\n}\n\n/**\n * Returns the value of `field` on `resource` only if it's a non-empty string.\n * Guards against HumanName arrays (Patient.name) and other non-string shapes\n * so the introspection fallback never produces a key from a typed field it\n * doesn't understand — that case stays on the lastUpdated fallback.\n */\nfunction introspectStringField(\n  resource: FhirResourceLike,\n  field: string,\n): string | undefined {\n  const value = resource[field];\n  if (typeof value === \"string\" && value.length > 0) return value;\n  return undefined;\n}\n\nfunction extractLabel(resource: FhirResourceLike, source: LabelSource): string {\n  if (source.kind === \"humanName\") {\n    const names = resource[source.path];\n    if (!Array.isArray(names)) return \"\";\n    const first = names[0];\n    if (!first || typeof first !== \"object\") return \"\";\n    const name = first as { family?: unknown; given?: unknown };\n    const family = typeof name.family === \"string\" ? name.family : \"\";\n    const given =\n      Array.isArray(name.given) && typeof name.given[0] === \"string\"\n        ? name.given[0]\n        : \"\";\n    if (family.length === 0 && given.length === 0) return \"\";\n    return `${family},${given}`;\n  }\n\n  if (source.kind === \"string\") {\n    const value = resource[source.path];\n    return typeof value === \"string\" ? value : \"\";\n  }\n\n  if (source.kind === \"codeDisplay\") {\n    const code = resource[source.path];\n    if (!code || typeof code !== \"object\") return \"\";\n    const cc = code as {\n      coding?: unknown;\n      text?: unknown;\n    };\n    if (Array.isArray(cc.coding)) {\n      const first = cc.coding[0];\n      if (first && typeof first === \"object\") {\n        const display = (first as { display?: unknown }).display;\n        if (typeof display === \"string\" && display.length > 0) return display;\n      }\n    }\n    return typeof cc.text === \"string\" ? cc.text : \"\";\n  }\n\n  return \"\";\n}\n\n/**\n * Normalize a natural-language label per DR-004:\n * 1. ASCII-fold common Latin diacritics (ñ→n, é→e, ø→o, …).\n * 2. Lowercase (DynamoDB byte-order sort).\n * 3. Strip apostrophes and periods (so `O'Connor` → `oconnor`).\n * 4. Replace hyphens with spaces (so `García-López` → `garcia lopez`).\n * 5. Strip other punctuation, collapse runs of whitespace, trim.\n *\n * Non-Latin scripts pass through unchanged after ASCII-fold so same-script\n * items sort together. Locale-aware collation is a downstream client\n * concern.\n */\nexport function normalizeLabel(input: string): string {\n  return asciiFold(input)\n    .toLowerCase()\n    .replace(/['’.]/g, \"\")\n    .replace(/-/g, \" \")\n    .replace(/[^\\p{L}\\p{N},\\s]/gu, \" \")\n    .replace(/\\s+/g, \" \")\n    .trim();\n}\n\nfunction asciiFold(input: string): string {\n  // Decompose accents and strip combining marks; covers the common Latin\n  // diacritics (é, ñ, ü, …). Non-Latin scripts are unaffected.\n  return input.normalize(\"NFKD\").replace(/\\p{M}+/gu, \"\");\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export * from "./label-paths";
2
+ export * from "./extract-sort-key";
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./label-paths"), exports);
18
+ __exportStar(require("./extract-sort-key"), exports);
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc29ydC1rZXkvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLGdEQUE4QjtBQUM5QixxREFBbUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9sYWJlbC1wYXRoc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZXh0cmFjdC1zb3J0LWtleVwiO1xuIl19
@@ -0,0 +1,29 @@
1
+ /**
2
+ * `LABEL_PATHS` — per-resource-type label-extraction rule for `extractSortKey`.
3
+ * Hand-curated platform-tier table per DR-004.
4
+ *
5
+ * Resource types listed here emit GSI1 sort keys of the form
6
+ * `<normalizedLabel>#<id>` so list endpoints return alphabetically sorted by
7
+ * natural label and `BEGINS_WITH` queries serve prefix searches like
8
+ * `?name=Sm`. Resource types absent from this map fall back to
9
+ * `<ISO-8601 lastUpdated>#<id>`.
10
+ *
11
+ * The minimum platform-curated set per DR-004 is Patient, Practitioner,
12
+ * Organization, Location, Medication. New entries here are one-line additions
13
+ * reviewed at PR time — no engine change required.
14
+ *
15
+ * @see openhi-planning DR-004 — FHIR Summary-Attribute and Sort-Key Projection Strategy
16
+ */
17
+ /** Source of the natural label for a labeled resource type. */
18
+ export type LabelSource = {
19
+ kind: "humanName";
20
+ path: "name";
21
+ } | {
22
+ kind: "string";
23
+ path: "name";
24
+ } | {
25
+ kind: "codeDisplay";
26
+ path: "code";
27
+ };
28
+ /** Resource types with curated natural labels (DR-004 Tier 1). */
29
+ export declare const LABEL_PATHS: Record<string, LabelSource>;
@@ -0,0 +1,28 @@
1
+ "use strict";
2
+ /**
3
+ * `LABEL_PATHS` — per-resource-type label-extraction rule for `extractSortKey`.
4
+ * Hand-curated platform-tier table per DR-004.
5
+ *
6
+ * Resource types listed here emit GSI1 sort keys of the form
7
+ * `<normalizedLabel>#<id>` so list endpoints return alphabetically sorted by
8
+ * natural label and `BEGINS_WITH` queries serve prefix searches like
9
+ * `?name=Sm`. Resource types absent from this map fall back to
10
+ * `<ISO-8601 lastUpdated>#<id>`.
11
+ *
12
+ * The minimum platform-curated set per DR-004 is Patient, Practitioner,
13
+ * Organization, Location, Medication. New entries here are one-line additions
14
+ * reviewed at PR time — no engine change required.
15
+ *
16
+ * @see openhi-planning DR-004 — FHIR Summary-Attribute and Sort-Key Projection Strategy
17
+ */
18
+ Object.defineProperty(exports, "__esModule", { value: true });
19
+ exports.LABEL_PATHS = void 0;
20
+ /** Resource types with curated natural labels (DR-004 Tier 1). */
21
+ exports.LABEL_PATHS = {
22
+ Patient: { kind: "humanName", path: "name" },
23
+ Practitioner: { kind: "humanName", path: "name" },
24
+ Organization: { kind: "string", path: "name" },
25
+ Location: { kind: "string", path: "name" },
26
+ Medication: { kind: "codeDisplay", path: "code" },
27
+ };
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGFiZWwtcGF0aHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc29ydC1rZXkvbGFiZWwtcGF0aHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7Ozs7Ozs7R0FlRzs7O0FBUUgsa0VBQWtFO0FBQ3JELFFBQUEsV0FBVyxHQUFnQztJQUN0RCxPQUFPLEVBQUUsRUFBRSxJQUFJLEVBQUUsV0FBVyxFQUFFLElBQUksRUFBRSxNQUFNLEVBQUU7SUFDNUMsWUFBWSxFQUFFLEVBQUUsSUFBSSxFQUFFLFdBQVcsRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFO0lBQ2pELFlBQVksRUFBRSxFQUFFLElBQUksRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLE1BQU0sRUFBRTtJQUM5QyxRQUFRLEVBQUUsRUFBRSxJQUFJLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxNQUFNLEVBQUU7SUFDMUMsVUFBVSxFQUFFLEVBQUUsSUFBSSxFQUFFLGFBQWEsRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFO0NBQ2xELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIGBMQUJFTF9QQVRIU2Ag4oCUIHBlci1yZXNvdXJjZS10eXBlIGxhYmVsLWV4dHJhY3Rpb24gcnVsZSBmb3IgYGV4dHJhY3RTb3J0S2V5YC5cbiAqIEhhbmQtY3VyYXRlZCBwbGF0Zm9ybS10aWVyIHRhYmxlIHBlciBEUi0wMDQuXG4gKlxuICogUmVzb3VyY2UgdHlwZXMgbGlzdGVkIGhlcmUgZW1pdCBHU0kxIHNvcnQga2V5cyBvZiB0aGUgZm9ybVxuICogYDxub3JtYWxpemVkTGFiZWw+IzxpZD5gIHNvIGxpc3QgZW5kcG9pbnRzIHJldHVybiBhbHBoYWJldGljYWxseSBzb3J0ZWQgYnlcbiAqIG5hdHVyYWwgbGFiZWwgYW5kIGBCRUdJTlNfV0lUSGAgcXVlcmllcyBzZXJ2ZSBwcmVmaXggc2VhcmNoZXMgbGlrZVxuICogYD9uYW1lPVNtYC4gUmVzb3VyY2UgdHlwZXMgYWJzZW50IGZyb20gdGhpcyBtYXAgZmFsbCBiYWNrIHRvXG4gKiBgPElTTy04NjAxIGxhc3RVcGRhdGVkPiM8aWQ+YC5cbiAqXG4gKiBUaGUgbWluaW11bSBwbGF0Zm9ybS1jdXJhdGVkIHNldCBwZXIgRFItMDA0IGlzIFBhdGllbnQsIFByYWN0aXRpb25lcixcbiAqIE9yZ2FuaXphdGlvbiwgTG9jYXRpb24sIE1lZGljYXRpb24uIE5ldyBlbnRyaWVzIGhlcmUgYXJlIG9uZS1saW5lIGFkZGl0aW9uc1xuICogcmV2aWV3ZWQgYXQgUFIgdGltZSDigJQgbm8gZW5naW5lIGNoYW5nZSByZXF1aXJlZC5cbiAqXG4gKiBAc2VlIG9wZW5oaS1wbGFubmluZyBEUi0wMDQg4oCUIEZISVIgU3VtbWFyeS1BdHRyaWJ1dGUgYW5kIFNvcnQtS2V5IFByb2plY3Rpb24gU3RyYXRlZ3lcbiAqL1xuXG4vKiogU291cmNlIG9mIHRoZSBuYXR1cmFsIGxhYmVsIGZvciBhIGxhYmVsZWQgcmVzb3VyY2UgdHlwZS4gKi9cbmV4cG9ydCB0eXBlIExhYmVsU291cmNlID1cbiAgfCB7IGtpbmQ6IFwiaHVtYW5OYW1lXCI7IHBhdGg6IFwibmFtZVwiIH1cbiAgfCB7IGtpbmQ6IFwic3RyaW5nXCI7IHBhdGg6IFwibmFtZVwiIH1cbiAgfCB7IGtpbmQ6IFwiY29kZURpc3BsYXlcIjsgcGF0aDogXCJjb2RlXCIgfTtcblxuLyoqIFJlc291cmNlIHR5cGVzIHdpdGggY3VyYXRlZCBuYXR1cmFsIGxhYmVscyAoRFItMDA0IFRpZXIgMSkuICovXG5leHBvcnQgY29uc3QgTEFCRUxfUEFUSFM6IFJlY29yZDxzdHJpbmcsIExhYmVsU291cmNlPiA9IHtcbiAgUGF0aWVudDogeyBraW5kOiBcImh1bWFuTmFtZVwiLCBwYXRoOiBcIm5hbWVcIiB9LFxuICBQcmFjdGl0aW9uZXI6IHsga2luZDogXCJodW1hbk5hbWVcIiwgcGF0aDogXCJuYW1lXCIgfSxcbiAgT3JnYW5pemF0aW9uOiB7IGtpbmQ6IFwic3RyaW5nXCIsIHBhdGg6IFwibmFtZVwiIH0sXG4gIExvY2F0aW9uOiB7IGtpbmQ6IFwic3RyaW5nXCIsIHBhdGg6IFwibmFtZVwiIH0sXG4gIE1lZGljYXRpb246IHsga2luZDogXCJjb2RlRGlzcGxheVwiLCBwYXRoOiBcImNvZGVcIiB9LFxufTtcbiJdfQ==
package/package.json CHANGED
@@ -36,7 +36,7 @@
36
36
  "publishConfig": {
37
37
  "access": "public"
38
38
  },
39
- "version": "0.0.24",
39
+ "version": "0.0.25",
40
40
  "types": "lib/index.d.ts",
41
41
  "//": "~~ Generated by projen. To modify, edit .projenrc.js and run \"pnpm exec projen\".",
42
42
  "scripts": {