@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 CHANGED
@@ -13,3 +13,4 @@
13
13
  */
14
14
  export * from "./data";
15
15
  export * from "./control";
16
+ export * from "./summary";
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
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBOzs7Ozs7Ozs7Ozs7R0FZRztBQUNILHlDQUF1QjtBQUN2Qiw0Q0FBMEIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBvcGVuaGkvdHlwZXMg4oCUIEZISVIgYW5kIE9wZW5ISSB0eXBlcyBieSBkb21haW4uXG4gKlxuICogRGF0YSBwbGFuZSBpcyBnZW5lcmF0b3Igb3V0cHV0IChgc3JjL2RhdGEvYCwgb25lIG1vZHVsZSBwZXIgRkhJUiBSNFxuICogU3RydWN0dXJlRGVmaW5pdGlvbiwgYmFycmVsIGluIGAuL2RhdGEvaW5kZXgudHNgKS4gQmFzZSByZXNvdXJjZXNcbiAqIChSZXNvdXJjZSwgRG9tYWluUmVzb3VyY2UsIEVsZW1lbnQsIEJhY2tib25lRWxlbWVudCwgRXh0ZW5zaW9uLCBldGMuKVxuICogbGl2ZSBpbiB0aGF0IHNhbWUgdHJlZSDigJQgdGhlIGdlbmVyYXRvciB0cmVhdHMgdGhlbSBsaWtlIGFueSBvdGhlciBTRC5cbiAqXG4gKiBDb250cm9sIHBsYW5lIGlzIGdlbmVyYXRvciBvdXRwdXQgKGBzcmMvY29udHJvbC9gKTogVXNlciwgVGVuYW50LFxuICogV29ya3NwYWNlLCBNZW1iZXJzaGlwLCBSb2xlQXNzaWdubWVudCwgUm9sZSwgcGx1cyB0aGVcbiAqIGBDb250cm9sUGxhbmVSb2xlQ29kZWAgdGVybWlub2xvZ3kuIEF1dGhvcmVkIGluIEZTSCB1bmRlclxuICogYGZzaC9pbnB1dC9mc2gvY29udHJvbC1wbGFuZS9gIGFuZCBlbWl0dGVkIGJ5IHRoZSBzYW1lIHBpcGVsaW5lLlxuICovXG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9jb250cm9sXCI7XG4iXX0=
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
@@ -36,7 +36,7 @@
36
36
  "publishConfig": {
37
37
  "access": "public"
38
38
  },
39
- "version": "0.0.19",
39
+ "version": "0.0.20",
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": {