@openhi/types 0.0.19 → 0.0.20
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 +1 -0
- package/lib/index.js +2 -1
- package/lib/summary/extract-summary.d.ts +56 -0
- package/lib/summary/extract-summary.js +173 -0
- package/lib/summary/index.d.ts +1 -0
- package/lib/summary/index.js +18 -0
- package/package.json +1 -1
package/lib/index.d.ts
CHANGED
package/lib/index.js
CHANGED
|
@@ -29,4 +29,5 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
29
29
|
*/
|
|
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
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
/** Summary exceeds the clinical-path p99; worth looking at if it's a pattern. */
|
|
2
|
+
export declare const DEFAULT_SUMMARY_WARN_BYTES: number;
|
|
3
|
+
/** Summary is outside any measured distribution; almost certainly inline attachment or unpacked Bundle. */
|
|
4
|
+
export declare const DEFAULT_SUMMARY_ERROR_BYTES: number;
|
|
5
|
+
export interface SummaryJson {
|
|
6
|
+
resourceType: string;
|
|
7
|
+
id?: string;
|
|
8
|
+
meta?: {
|
|
9
|
+
lastUpdated?: string;
|
|
10
|
+
versionId?: string;
|
|
11
|
+
};
|
|
12
|
+
[key: string]: unknown;
|
|
13
|
+
}
|
|
14
|
+
export interface FhirResourceLike {
|
|
15
|
+
resourceType: string;
|
|
16
|
+
id?: string;
|
|
17
|
+
meta?: {
|
|
18
|
+
lastUpdated?: string;
|
|
19
|
+
versionId?: string;
|
|
20
|
+
[key: string]: unknown;
|
|
21
|
+
};
|
|
22
|
+
[key: string]: unknown;
|
|
23
|
+
}
|
|
24
|
+
export interface ExtractSummaryLogger {
|
|
25
|
+
warn(...args: unknown[]): void;
|
|
26
|
+
error(...args: unknown[]): void;
|
|
27
|
+
}
|
|
28
|
+
export interface ExtractSummaryOptions {
|
|
29
|
+
/** Soft (info) warn threshold in UTF-8 bytes. Defaults to 4 KiB. */
|
|
30
|
+
warnThresholdBytes?: number;
|
|
31
|
+
/** Hard (error) warn threshold in UTF-8 bytes. Defaults to 8 KiB. */
|
|
32
|
+
errorThresholdBytes?: number;
|
|
33
|
+
/** Logger for size-guard warnings. Defaults to `console`. */
|
|
34
|
+
logger?: ExtractSummaryLogger;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Projects `resource` into its summary JSON using `SUMMARY_PATHS`.
|
|
38
|
+
*
|
|
39
|
+
* Always emits `resourceType`, `id` (if present), and
|
|
40
|
+
* `meta.{lastUpdated,versionId}` (if present). Then, for every path in
|
|
41
|
+
* `SUMMARY_PATHS[resourceType]`:
|
|
42
|
+
* - Simple paths (`name`) copy the whole value (arrays shallow-copied).
|
|
43
|
+
* - Dotted paths (`component.code`) project the named sub-field of each
|
|
44
|
+
* element of the container; if any sub-paths are listed for a
|
|
45
|
+
* container, only those sub-fields are copied (not the whole element).
|
|
46
|
+
* - `[x]` choice-type paths (`value[x]`) match whichever concrete
|
|
47
|
+
* variant (`valueQuantity`, `valueString`, …) is present.
|
|
48
|
+
*
|
|
49
|
+
* Missing fields are omitted — no nulls.
|
|
50
|
+
*
|
|
51
|
+
* If the serialized summary exceeds `warnThresholdBytes` /
|
|
52
|
+
* `errorThresholdBytes`, a warn / error is logged with `{ resourceType,
|
|
53
|
+
* id, bytes }`. The summary is always returned; the size guard never
|
|
54
|
+
* rejects.
|
|
55
|
+
*/
|
|
56
|
+
export declare function extractSummary(resource: FhirResourceLike, options?: ExtractSummaryOptions): SummaryJson;
|
|
@@ -0,0 +1,173 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.DEFAULT_SUMMARY_ERROR_BYTES = exports.DEFAULT_SUMMARY_WARN_BYTES = void 0;
|
|
4
|
+
exports.extractSummary = extractSummary;
|
|
5
|
+
/**
|
|
6
|
+
* Pure projection of a FHIR R4 resource into its summary JSON — the
|
|
7
|
+
* GSI-friendly shape defined by ADR 2026-04-17-02 and the generated
|
|
8
|
+
* `SUMMARY_PATHS` table (issue #812).
|
|
9
|
+
*
|
|
10
|
+
* Consumed by the write path in #814; no I/O, no DynamoDB access here.
|
|
11
|
+
*
|
|
12
|
+
* Threshold defaults come from the #811 size benchmark across every R4
|
|
13
|
+
* example resource. Both thresholds are warn-only — the guard is a
|
|
14
|
+
* diagnostic signal, never a reason to reject a store operation.
|
|
15
|
+
*/
|
|
16
|
+
const summary_paths_1 = require("../data/summary-paths");
|
|
17
|
+
/** Summary exceeds the clinical-path p99; worth looking at if it's a pattern. */
|
|
18
|
+
exports.DEFAULT_SUMMARY_WARN_BYTES = 4 * 1024;
|
|
19
|
+
/** Summary is outside any measured distribution; almost certainly inline attachment or unpacked Bundle. */
|
|
20
|
+
exports.DEFAULT_SUMMARY_ERROR_BYTES = 8 * 1024;
|
|
21
|
+
/**
|
|
22
|
+
* Projects `resource` into its summary JSON using `SUMMARY_PATHS`.
|
|
23
|
+
*
|
|
24
|
+
* Always emits `resourceType`, `id` (if present), and
|
|
25
|
+
* `meta.{lastUpdated,versionId}` (if present). Then, for every path in
|
|
26
|
+
* `SUMMARY_PATHS[resourceType]`:
|
|
27
|
+
* - Simple paths (`name`) copy the whole value (arrays shallow-copied).
|
|
28
|
+
* - Dotted paths (`component.code`) project the named sub-field of each
|
|
29
|
+
* element of the container; if any sub-paths are listed for a
|
|
30
|
+
* container, only those sub-fields are copied (not the whole element).
|
|
31
|
+
* - `[x]` choice-type paths (`value[x]`) match whichever concrete
|
|
32
|
+
* variant (`valueQuantity`, `valueString`, …) is present.
|
|
33
|
+
*
|
|
34
|
+
* Missing fields are omitted — no nulls.
|
|
35
|
+
*
|
|
36
|
+
* If the serialized summary exceeds `warnThresholdBytes` /
|
|
37
|
+
* `errorThresholdBytes`, a warn / error is logged with `{ resourceType,
|
|
38
|
+
* id, bytes }`. The summary is always returned; the size guard never
|
|
39
|
+
* rejects.
|
|
40
|
+
*/
|
|
41
|
+
function extractSummary(resource, options = {}) {
|
|
42
|
+
const { resourceType, id } = resource;
|
|
43
|
+
const logger = options.logger ?? console;
|
|
44
|
+
const warnBytes = options.warnThresholdBytes ?? exports.DEFAULT_SUMMARY_WARN_BYTES;
|
|
45
|
+
const errorBytes = options.errorThresholdBytes ?? exports.DEFAULT_SUMMARY_ERROR_BYTES;
|
|
46
|
+
const result = { resourceType };
|
|
47
|
+
if (id !== undefined)
|
|
48
|
+
result.id = id;
|
|
49
|
+
const meta = resource.meta;
|
|
50
|
+
if (meta && typeof meta === "object") {
|
|
51
|
+
const projectedMeta = {};
|
|
52
|
+
if (meta.lastUpdated !== undefined) {
|
|
53
|
+
projectedMeta.lastUpdated = meta.lastUpdated;
|
|
54
|
+
}
|
|
55
|
+
if (meta.versionId !== undefined) {
|
|
56
|
+
projectedMeta.versionId = meta.versionId;
|
|
57
|
+
}
|
|
58
|
+
if (Object.keys(projectedMeta).length > 0) {
|
|
59
|
+
result.meta = projectedMeta;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
const paths = summary_paths_1.SUMMARY_PATHS[resourceType];
|
|
63
|
+
if (paths && paths.length > 0) {
|
|
64
|
+
// `id` and `meta` are handled above with bespoke rules (meta is projected
|
|
65
|
+
// to just lastUpdated / versionId, not the whole Meta datatype).
|
|
66
|
+
const applicable = paths.filter((p) => p !== "id" && p !== "meta");
|
|
67
|
+
applyPaths(resource, result, applicable);
|
|
68
|
+
}
|
|
69
|
+
const bytes = utf8ByteLength(JSON.stringify(result));
|
|
70
|
+
if (bytes > errorBytes) {
|
|
71
|
+
logger.error("extractSummary: summary exceeds error threshold", {
|
|
72
|
+
resourceType,
|
|
73
|
+
id,
|
|
74
|
+
bytes,
|
|
75
|
+
thresholdBytes: errorBytes,
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
else if (bytes > warnBytes) {
|
|
79
|
+
logger.warn("extractSummary: summary exceeds warn threshold", {
|
|
80
|
+
resourceType,
|
|
81
|
+
id,
|
|
82
|
+
bytes,
|
|
83
|
+
thresholdBytes: warnBytes,
|
|
84
|
+
});
|
|
85
|
+
}
|
|
86
|
+
return result;
|
|
87
|
+
}
|
|
88
|
+
function applyPaths(source, target, paths) {
|
|
89
|
+
const groups = groupByHead(paths);
|
|
90
|
+
for (const [head, groupPaths] of groups) {
|
|
91
|
+
if (head.endsWith("[x]")) {
|
|
92
|
+
applyChoicePath(source, target, head);
|
|
93
|
+
continue;
|
|
94
|
+
}
|
|
95
|
+
const value = source[head];
|
|
96
|
+
if (value === undefined || value === null)
|
|
97
|
+
continue;
|
|
98
|
+
const children = [];
|
|
99
|
+
for (const p of groupPaths) {
|
|
100
|
+
if (p === head)
|
|
101
|
+
continue;
|
|
102
|
+
children.push(p.slice(head.length + 1));
|
|
103
|
+
}
|
|
104
|
+
if (children.length === 0) {
|
|
105
|
+
target[head] = shallowCopy(value);
|
|
106
|
+
continue;
|
|
107
|
+
}
|
|
108
|
+
if (Array.isArray(value)) {
|
|
109
|
+
const projected = [];
|
|
110
|
+
for (const el of value) {
|
|
111
|
+
if (el && typeof el === "object" && !Array.isArray(el)) {
|
|
112
|
+
const sub = {};
|
|
113
|
+
applyPaths(el, sub, children);
|
|
114
|
+
if (Object.keys(sub).length > 0)
|
|
115
|
+
projected.push(sub);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
if (projected.length > 0)
|
|
119
|
+
target[head] = projected;
|
|
120
|
+
continue;
|
|
121
|
+
}
|
|
122
|
+
if (typeof value === "object") {
|
|
123
|
+
const sub = {};
|
|
124
|
+
applyPaths(value, sub, children);
|
|
125
|
+
if (Object.keys(sub).length > 0)
|
|
126
|
+
target[head] = sub;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
function applyChoicePath(source, target, segment) {
|
|
131
|
+
const prefix = segment.slice(0, -"[x]".length);
|
|
132
|
+
for (const key of Object.keys(source)) {
|
|
133
|
+
if (!key.startsWith(prefix) || key.length <= prefix.length)
|
|
134
|
+
continue;
|
|
135
|
+
const nextCharCode = key.charCodeAt(prefix.length);
|
|
136
|
+
// Concrete FHIR variants are always `<prefix><CapitalizedType>`, e.g.
|
|
137
|
+
// `valueQuantity`, `effectiveDateTime`. Require an uppercase ASCII
|
|
138
|
+
// character immediately after the prefix to avoid false-positive
|
|
139
|
+
// matches against unrelated fields that happen to share the prefix.
|
|
140
|
+
if (nextCharCode < 65 || nextCharCode > 90)
|
|
141
|
+
continue;
|
|
142
|
+
const value = source[key];
|
|
143
|
+
if (value === undefined)
|
|
144
|
+
continue;
|
|
145
|
+
target[key] = shallowCopy(value);
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
function groupByHead(paths) {
|
|
149
|
+
const groups = new Map();
|
|
150
|
+
for (const p of paths) {
|
|
151
|
+
const dot = p.indexOf(".");
|
|
152
|
+
const head = dot === -1 ? p : p.slice(0, dot);
|
|
153
|
+
let bucket = groups.get(head);
|
|
154
|
+
if (!bucket) {
|
|
155
|
+
bucket = [];
|
|
156
|
+
groups.set(head, bucket);
|
|
157
|
+
}
|
|
158
|
+
bucket.push(p);
|
|
159
|
+
}
|
|
160
|
+
return groups;
|
|
161
|
+
}
|
|
162
|
+
function shallowCopy(value) {
|
|
163
|
+
if (Array.isArray(value))
|
|
164
|
+
return value.slice();
|
|
165
|
+
if (value !== null && typeof value === "object") {
|
|
166
|
+
return { ...value };
|
|
167
|
+
}
|
|
168
|
+
return value;
|
|
169
|
+
}
|
|
170
|
+
function utf8ByteLength(s) {
|
|
171
|
+
return new TextEncoder().encode(s).length;
|
|
172
|
+
}
|
|
173
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extract-summary.js","sourceRoot":"","sources":["../../src/summary/extract-summary.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Pure projection of a FHIR R4 resource into its summary JSON — the\n * GSI-friendly shape defined by ADR 2026-04-17-02 and the generated\n * `SUMMARY_PATHS` table (issue #812).\n *\n * Consumed by the write path in #814; no I/O, no DynamoDB access here.\n *\n * Threshold defaults come from the #811 size benchmark across every R4\n * example resource. Both thresholds are warn-only — the guard is a\n * diagnostic signal, never a reason to reject a store operation.\n */\nimport { SUMMARY_PATHS } from \"../data/summary-paths\";\n\n/** Summary exceeds the clinical-path p99; worth looking at if it's a pattern. */\nexport const DEFAULT_SUMMARY_WARN_BYTES = 4 * 1024;\n\n/** Summary is outside any measured distribution; almost certainly inline attachment or unpacked Bundle. */\nexport const DEFAULT_SUMMARY_ERROR_BYTES = 8 * 1024;\n\nexport interface SummaryJson {\n  resourceType: string;\n  id?: string;\n  meta?: {\n    lastUpdated?: string;\n    versionId?: string;\n  };\n  [key: string]: unknown;\n}\n\nexport interface FhirResourceLike {\n  resourceType: string;\n  id?: string;\n  meta?: {\n    lastUpdated?: string;\n    versionId?: string;\n    [key: string]: unknown;\n  };\n  [key: string]: unknown;\n}\n\nexport interface ExtractSummaryLogger {\n  warn(...args: unknown[]): void;\n  error(...args: unknown[]): void;\n}\n\nexport interface ExtractSummaryOptions {\n  /** Soft (info) warn threshold in UTF-8 bytes. Defaults to 4 KiB. */\n  warnThresholdBytes?: number;\n  /** Hard (error) warn threshold in UTF-8 bytes. Defaults to 8 KiB. */\n  errorThresholdBytes?: number;\n  /** Logger for size-guard warnings. Defaults to `console`. */\n  logger?: ExtractSummaryLogger;\n}\n\n/**\n * Projects `resource` into its summary JSON using `SUMMARY_PATHS`.\n *\n * Always emits `resourceType`, `id` (if present), and\n * `meta.{lastUpdated,versionId}` (if present). Then, for every path in\n * `SUMMARY_PATHS[resourceType]`:\n *   - Simple paths (`name`) copy the whole value (arrays shallow-copied).\n *   - Dotted paths (`component.code`) project the named sub-field of each\n *     element of the container; if any sub-paths are listed for a\n *     container, only those sub-fields are copied (not the whole element).\n *   - `[x]` choice-type paths (`value[x]`) match whichever concrete\n *     variant (`valueQuantity`, `valueString`, …) is present.\n *\n * Missing fields are omitted — no nulls.\n *\n * If the serialized summary exceeds `warnThresholdBytes` /\n * `errorThresholdBytes`, a warn / error is logged with `{ resourceType,\n * id, bytes }`. The summary is always returned; the size guard never\n * rejects.\n */\nexport function extractSummary(\n  resource: FhirResourceLike,\n  options: ExtractSummaryOptions = {},\n): SummaryJson {\n  const { resourceType, id } = resource;\n  const logger: ExtractSummaryLogger = options.logger ?? console;\n  const warnBytes = options.warnThresholdBytes ?? DEFAULT_SUMMARY_WARN_BYTES;\n  const errorBytes = options.errorThresholdBytes ?? DEFAULT_SUMMARY_ERROR_BYTES;\n\n  const result: SummaryJson = { resourceType };\n  if (id !== undefined) result.id = id;\n\n  const meta = resource.meta;\n  if (meta && typeof meta === \"object\") {\n    const projectedMeta: { lastUpdated?: string; versionId?: string } = {};\n    if (meta.lastUpdated !== undefined) {\n      projectedMeta.lastUpdated = meta.lastUpdated;\n    }\n    if (meta.versionId !== undefined) {\n      projectedMeta.versionId = meta.versionId;\n    }\n    if (Object.keys(projectedMeta).length > 0) {\n      result.meta = projectedMeta;\n    }\n  }\n\n  const paths = SUMMARY_PATHS[resourceType];\n  if (paths && paths.length > 0) {\n    // `id` and `meta` are handled above with bespoke rules (meta is projected\n    // to just lastUpdated / versionId, not the whole Meta datatype).\n    const applicable = paths.filter((p) => p !== \"id\" && p !== \"meta\");\n    applyPaths(\n      resource as unknown as Record<string, unknown>,\n      result as unknown as Record<string, unknown>,\n      applicable,\n    );\n  }\n\n  const bytes = utf8ByteLength(JSON.stringify(result));\n  if (bytes > errorBytes) {\n    logger.error(\"extractSummary: summary exceeds error threshold\", {\n      resourceType,\n      id,\n      bytes,\n      thresholdBytes: errorBytes,\n    });\n  } else if (bytes > warnBytes) {\n    logger.warn(\"extractSummary: summary exceeds warn threshold\", {\n      resourceType,\n      id,\n      bytes,\n      thresholdBytes: warnBytes,\n    });\n  }\n\n  return result;\n}\n\nfunction applyPaths(\n  source: Record<string, unknown>,\n  target: Record<string, unknown>,\n  paths: readonly string[],\n): void {\n  const groups = groupByHead(paths);\n  for (const [head, groupPaths] of groups) {\n    if (head.endsWith(\"[x]\")) {\n      applyChoicePath(source, target, head);\n      continue;\n    }\n\n    const value = source[head];\n    if (value === undefined || value === null) continue;\n\n    const children: string[] = [];\n    for (const p of groupPaths) {\n      if (p === head) continue;\n      children.push(p.slice(head.length + 1));\n    }\n\n    if (children.length === 0) {\n      target[head] = shallowCopy(value);\n      continue;\n    }\n\n    if (Array.isArray(value)) {\n      const projected: unknown[] = [];\n      for (const el of value) {\n        if (el && typeof el === \"object\" && !Array.isArray(el)) {\n          const sub: Record<string, unknown> = {};\n          applyPaths(el as Record<string, unknown>, sub, children);\n          if (Object.keys(sub).length > 0) projected.push(sub);\n        }\n      }\n      if (projected.length > 0) target[head] = projected;\n      continue;\n    }\n\n    if (typeof value === \"object\") {\n      const sub: Record<string, unknown> = {};\n      applyPaths(value as Record<string, unknown>, sub, children);\n      if (Object.keys(sub).length > 0) target[head] = sub;\n    }\n  }\n}\n\nfunction applyChoicePath(\n  source: Record<string, unknown>,\n  target: Record<string, unknown>,\n  segment: string,\n): void {\n  const prefix = segment.slice(0, -\"[x]\".length);\n  for (const key of Object.keys(source)) {\n    if (!key.startsWith(prefix) || key.length <= prefix.length) continue;\n    const nextCharCode = key.charCodeAt(prefix.length);\n    // Concrete FHIR variants are always `<prefix><CapitalizedType>`, e.g.\n    // `valueQuantity`, `effectiveDateTime`. Require an uppercase ASCII\n    // character immediately after the prefix to avoid false-positive\n    // matches against unrelated fields that happen to share the prefix.\n    if (nextCharCode < 65 || nextCharCode > 90) continue;\n    const value = source[key];\n    if (value === undefined) continue;\n    target[key] = shallowCopy(value);\n  }\n}\n\nfunction groupByHead(paths: readonly string[]): Map<string, string[]> {\n  const groups = new Map<string, string[]>();\n  for (const p of paths) {\n    const dot = p.indexOf(\".\");\n    const head = dot === -1 ? p : p.slice(0, dot);\n    let bucket = groups.get(head);\n    if (!bucket) {\n      bucket = [];\n      groups.set(head, bucket);\n    }\n    bucket.push(p);\n  }\n  return groups;\n}\n\nfunction shallowCopy(value: unknown): unknown {\n  if (Array.isArray(value)) return value.slice();\n  if (value !== null && typeof value === \"object\") {\n    return { ...(value as Record<string, unknown>) };\n  }\n  return value;\n}\n\nfunction utf8ByteLength(s: string): number {\n  return new TextEncoder().encode(s).length;\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from "./extract-summary";
|
|
@@ -0,0 +1,18 @@
|
|
|
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("./extract-summary"), exports);
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc3VtbWFyeS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsb0RBQWtDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vZXh0cmFjdC1zdW1tYXJ5XCI7XG4iXX0=
|
package/package.json
CHANGED