@blue-labs/repository-contract 2.0.0-rc.14

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.
@@ -0,0 +1,5 @@
1
+ export * from './types.js';
2
+ export * from './pointers.js';
3
+ export * from './refs.js';
4
+ export * from './validation.js';
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,YAAY,CAAC;AAC3B,cAAc,eAAe,CAAC;AAC9B,cAAc,WAAW,CAAC;AAC1B,cAAc,iBAAiB,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,4 @@
1
+ export * from './types.js';
2
+ export * from './pointers.js';
3
+ export * from './refs.js';
4
+ export * from './validation.js';
@@ -0,0 +1,10 @@
1
+ export declare class InvalidRepositoryPointerError extends Error {
2
+ readonly pointer: string;
3
+ constructor(pointer: string, message?: string);
4
+ }
5
+ export declare const RESERVED_ATTRIBUTES_POINTER_SEGMENTS: ReadonlySet<string>;
6
+ export declare function unescapePointerToken(token: string): string;
7
+ export declare function parsePointer(pointer: string): string[];
8
+ export declare function validatePointer(pointer: string): void;
9
+ export declare function validateAttributesAddedPointer(pointer: string): void;
10
+ //# sourceMappingURL=pointers.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pointers.d.ts","sourceRoot":"","sources":["../src/pointers.ts"],"names":[],"mappings":"AAAA,qBAAa,6BAA8B,SAAQ,KAAK;aAEpC,OAAO,EAAE,MAAM;gBAAf,OAAO,EAAE,MAAM,EAC/B,OAAO,CAAC,EAAE,MAAM;CAKnB;AAED,eAAO,MAAM,oCAAoC,EAAE,WAAW,CAAC,MAAM,CAajE,CAAC;AAEL,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAuB1D;AAED,wBAAgB,YAAY,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,EAAE,CAoBtD;AAED,wBAAgB,eAAe,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,CAErD;AAED,wBAAgB,8BAA8B,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,CAyBpE"}
@@ -0,0 +1,75 @@
1
+ export class InvalidRepositoryPointerError extends Error {
2
+ pointer;
3
+ constructor(pointer, message) {
4
+ super(message ?? `Invalid repository pointer: ${pointer}`);
5
+ this.pointer = pointer;
6
+ this.name = 'InvalidRepositoryPointerError';
7
+ }
8
+ }
9
+ export const RESERVED_ATTRIBUTES_POINTER_SEGMENTS = new Set([
10
+ 'name',
11
+ 'description',
12
+ 'type',
13
+ 'keyType',
14
+ 'value',
15
+ 'items',
16
+ 'blueId',
17
+ 'blue',
18
+ 'schema',
19
+ 'mergePolicy',
20
+ 'contracts',
21
+ ]);
22
+ export function unescapePointerToken(token) {
23
+ let result = '';
24
+ for (let i = 0; i < token.length; i++) {
25
+ const ch = token[i];
26
+ if (ch !== '~') {
27
+ result += ch;
28
+ continue;
29
+ }
30
+ const next = token[i + 1];
31
+ if (next === '0') {
32
+ result += '~';
33
+ i++;
34
+ }
35
+ else if (next === '1') {
36
+ result += '/';
37
+ i++;
38
+ }
39
+ else {
40
+ throw new InvalidRepositoryPointerError(token, `Invalid escape sequence in pointer token: ~${next ?? ''}`);
41
+ }
42
+ }
43
+ return result;
44
+ }
45
+ export function parsePointer(pointer) {
46
+ if (pointer === '') {
47
+ return [];
48
+ }
49
+ if (!pointer.startsWith('/')) {
50
+ throw new InvalidRepositoryPointerError(pointer, 'Pointer must start with "/" or be empty');
51
+ }
52
+ const segments = pointer.split('/').slice(1);
53
+ if (segments.some((segment) => segment.length === 0)) {
54
+ throw new InvalidRepositoryPointerError(pointer, 'Pointer must not contain empty segments');
55
+ }
56
+ return segments.map(unescapePointerToken);
57
+ }
58
+ export function validatePointer(pointer) {
59
+ parsePointer(pointer);
60
+ }
61
+ export function validateAttributesAddedPointer(pointer) {
62
+ if (pointer === '' || !pointer.startsWith('/')) {
63
+ throw new InvalidRepositoryPointerError(pointer, 'Pointer must start with "/"');
64
+ }
65
+ const segments = parsePointer(pointer);
66
+ for (const segment of segments) {
67
+ if (RESERVED_ATTRIBUTES_POINTER_SEGMENTS.has(segment)) {
68
+ throw new InvalidRepositoryPointerError(pointer, `attributesAdded pointers must not reference reserved field '${segment}'`);
69
+ }
70
+ }
71
+ const terminal = segments.at(-1);
72
+ if (terminal === 'itemType' || terminal === 'valueType') {
73
+ throw new InvalidRepositoryPointerError(pointer, `attributesAdded pointers must not end with '${terminal}'`);
74
+ }
75
+ }
package/dist/refs.d.ts ADDED
@@ -0,0 +1,7 @@
1
+ import type { BlueRepository, TypeAlias, TypeBlueId } from './types.js';
2
+ type Ref = TypeBlueId | TypeAlias;
3
+ export declare function collectTypeRefsFromContent(content: unknown): Set<Ref>;
4
+ export declare function validateNoCycles(repository: BlueRepository): void;
5
+ export declare function validateStableDoesNotDependOnDev(repository: BlueRepository): void;
6
+ export {};
7
+ //# sourceMappingURL=refs.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"refs.d.ts","sourceRoot":"","sources":["../src/refs.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,cAAc,EAGd,SAAS,EACT,UAAU,EACX,MAAM,YAAY,CAAC;AAEpB,KAAK,GAAG,GAAG,UAAU,GAAG,SAAS,CAAC;AAgBlC,wBAAgB,0BAA0B,CAAC,OAAO,EAAE,OAAO,GAAG,GAAG,CAAC,GAAG,CAAC,CAqCrE;AA+DD,wBAAgB,gBAAgB,CAAC,UAAU,EAAE,cAAc,GAAG,IAAI,CA6CjE;AAED,wBAAgB,gCAAgC,CAC9C,UAAU,EAAE,cAAc,GACzB,IAAI,CAsBN"}
package/dist/refs.js ADDED
@@ -0,0 +1,142 @@
1
+ function extractRef(value) {
2
+ if (typeof value === 'string') {
3
+ return value;
4
+ }
5
+ if (value &&
6
+ typeof value === 'object' &&
7
+ typeof value.blueId === 'string') {
8
+ return value.blueId;
9
+ }
10
+ return null;
11
+ }
12
+ export function collectTypeRefsFromContent(content) {
13
+ const refs = new Set();
14
+ const seen = new WeakSet();
15
+ const visit = (node) => {
16
+ if (!node || typeof node !== 'object') {
17
+ return;
18
+ }
19
+ const obj = node;
20
+ if (seen.has(obj)) {
21
+ return;
22
+ }
23
+ seen.add(obj);
24
+ for (const key of ['type', 'itemType', 'keyType', 'valueType']) {
25
+ if (key in obj) {
26
+ const ref = extractRef(obj[key]);
27
+ if (ref) {
28
+ refs.add(ref);
29
+ }
30
+ }
31
+ }
32
+ for (const value of Object.values(obj)) {
33
+ if (value && typeof value === 'object') {
34
+ visit(value);
35
+ }
36
+ }
37
+ };
38
+ if (Array.isArray(content)) {
39
+ content.forEach(visit);
40
+ }
41
+ else {
42
+ visit(content);
43
+ }
44
+ return refs;
45
+ }
46
+ function getTypesMeta(pkg) {
47
+ const fromTypesMeta = pkg.typesMeta;
48
+ if (fromTypesMeta) {
49
+ return fromTypesMeta;
50
+ }
51
+ const fromLegacyTypeMetas = pkg.typeMetas;
52
+ if (fromLegacyTypeMetas) {
53
+ return fromLegacyTypeMetas;
54
+ }
55
+ const direct = pkg.typesMeta;
56
+ return direct ?? {};
57
+ }
58
+ function buildResolver(packages) {
59
+ const aliasToBlueId = new Map();
60
+ const allTypeMetas = new Map();
61
+ Object.values(packages).forEach((pkg) => {
62
+ Object.entries(pkg.aliases || {}).forEach(([alias, blueId]) => {
63
+ aliasToBlueId.set(alias, blueId);
64
+ aliasToBlueId.set(alias.toLowerCase(), blueId);
65
+ });
66
+ Object.entries(getTypesMeta(pkg)).forEach(([blueId, meta]) => {
67
+ allTypeMetas.set(blueId, meta);
68
+ });
69
+ });
70
+ return { allTypeMetas, aliasToBlueId };
71
+ }
72
+ function resolveRef(ref, aliasToBlueId, allTypeMetas) {
73
+ if (allTypeMetas.has(ref)) {
74
+ return ref;
75
+ }
76
+ const resolved = aliasToBlueId.get(ref) ?? aliasToBlueId.get(ref.toLowerCase());
77
+ if (resolved) {
78
+ return resolved;
79
+ }
80
+ return null;
81
+ }
82
+ export function validateNoCycles(repository) {
83
+ const { aliasToBlueId, allTypeMetas } = buildResolver(repository.packages);
84
+ const adjacency = new Map();
85
+ for (const pkg of Object.values(repository.packages)) {
86
+ Object.entries(pkg.contents || {}).forEach(([blueId, content]) => {
87
+ const refs = collectTypeRefsFromContent(content);
88
+ refs.forEach((ref) => {
89
+ const resolved = resolveRef(ref, aliasToBlueId, allTypeMetas);
90
+ if (resolved) {
91
+ const set = adjacency.get(blueId) ?? new Set();
92
+ set.add(resolved);
93
+ adjacency.set(blueId, set);
94
+ }
95
+ });
96
+ });
97
+ }
98
+ const visiting = new Set();
99
+ const visited = new Set();
100
+ const dfs = (node, path) => {
101
+ if (visiting.has(node)) {
102
+ const cyclePath = [...path, node].join(' -> ');
103
+ throw new Error(`Repository contains a type reference cycle: ${cyclePath}`);
104
+ }
105
+ if (visited.has(node)) {
106
+ return;
107
+ }
108
+ visiting.add(node);
109
+ const next = adjacency.get(node);
110
+ if (next) {
111
+ next.forEach((neighbor) => dfs(neighbor, [...path, neighbor]));
112
+ }
113
+ visiting.delete(node);
114
+ visited.add(node);
115
+ };
116
+ adjacency.forEach((_neighbors, node) => {
117
+ if (!visited.has(node)) {
118
+ dfs(node, [node]);
119
+ }
120
+ });
121
+ }
122
+ export function validateStableDoesNotDependOnDev(repository) {
123
+ const { aliasToBlueId, allTypeMetas } = buildResolver(repository.packages);
124
+ const resolve = (ref) => resolveRef(ref, aliasToBlueId, allTypeMetas);
125
+ for (const [pkgName, pkg] of Object.entries(repository.packages)) {
126
+ const typesMeta = getTypesMeta(pkg);
127
+ for (const [blueId, meta] of Object.entries(typesMeta)) {
128
+ if (meta.status !== 'stable')
129
+ continue;
130
+ const refs = collectTypeRefsFromContent(pkg.contents[blueId]);
131
+ for (const ref of refs) {
132
+ const resolved = resolve(ref);
133
+ if (!resolved)
134
+ continue;
135
+ const targetMeta = allTypeMetas.get(resolved);
136
+ if (targetMeta?.status === 'dev') {
137
+ throw new Error(`Stable type ${pkgName}/${meta.name} depends on dev type ${resolved}`);
138
+ }
139
+ }
140
+ }
141
+ }
142
+ }
@@ -0,0 +1 @@
1
+ {"fileNames":["../../../node_modules/typescript/lib/lib.es5.d.ts","../../../node_modules/typescript/lib/lib.es2015.d.ts","../../../node_modules/typescript/lib/lib.es2016.d.ts","../../../node_modules/typescript/lib/lib.es2017.d.ts","../../../node_modules/typescript/lib/lib.es2018.d.ts","../../../node_modules/typescript/lib/lib.es2019.d.ts","../../../node_modules/typescript/lib/lib.es2020.d.ts","../../../node_modules/typescript/lib/lib.es2021.d.ts","../../../node_modules/typescript/lib/lib.es2022.d.ts","../../../node_modules/typescript/lib/lib.es2015.core.d.ts","../../../node_modules/typescript/lib/lib.es2015.collection.d.ts","../../../node_modules/typescript/lib/lib.es2015.generator.d.ts","../../../node_modules/typescript/lib/lib.es2015.iterable.d.ts","../../../node_modules/typescript/lib/lib.es2015.promise.d.ts","../../../node_modules/typescript/lib/lib.es2015.proxy.d.ts","../../../node_modules/typescript/lib/lib.es2015.reflect.d.ts","../../../node_modules/typescript/lib/lib.es2015.symbol.d.ts","../../../node_modules/typescript/lib/lib.es2015.symbol.wellknown.d.ts","../../../node_modules/typescript/lib/lib.es2016.array.include.d.ts","../../../node_modules/typescript/lib/lib.es2016.intl.d.ts","../../../node_modules/typescript/lib/lib.es2017.arraybuffer.d.ts","../../../node_modules/typescript/lib/lib.es2017.date.d.ts","../../../node_modules/typescript/lib/lib.es2017.object.d.ts","../../../node_modules/typescript/lib/lib.es2017.sharedmemory.d.ts","../../../node_modules/typescript/lib/lib.es2017.string.d.ts","../../../node_modules/typescript/lib/lib.es2017.intl.d.ts","../../../node_modules/typescript/lib/lib.es2017.typedarrays.d.ts","../../../node_modules/typescript/lib/lib.es2018.asyncgenerator.d.ts","../../../node_modules/typescript/lib/lib.es2018.asynciterable.d.ts","../../../node_modules/typescript/lib/lib.es2018.intl.d.ts","../../../node_modules/typescript/lib/lib.es2018.promise.d.ts","../../../node_modules/typescript/lib/lib.es2018.regexp.d.ts","../../../node_modules/typescript/lib/lib.es2019.array.d.ts","../../../node_modules/typescript/lib/lib.es2019.object.d.ts","../../../node_modules/typescript/lib/lib.es2019.string.d.ts","../../../node_modules/typescript/lib/lib.es2019.symbol.d.ts","../../../node_modules/typescript/lib/lib.es2019.intl.d.ts","../../../node_modules/typescript/lib/lib.es2020.bigint.d.ts","../../../node_modules/typescript/lib/lib.es2020.date.d.ts","../../../node_modules/typescript/lib/lib.es2020.promise.d.ts","../../../node_modules/typescript/lib/lib.es2020.sharedmemory.d.ts","../../../node_modules/typescript/lib/lib.es2020.string.d.ts","../../../node_modules/typescript/lib/lib.es2020.symbol.wellknown.d.ts","../../../node_modules/typescript/lib/lib.es2020.intl.d.ts","../../../node_modules/typescript/lib/lib.es2020.number.d.ts","../../../node_modules/typescript/lib/lib.es2021.promise.d.ts","../../../node_modules/typescript/lib/lib.es2021.string.d.ts","../../../node_modules/typescript/lib/lib.es2021.weakref.d.ts","../../../node_modules/typescript/lib/lib.es2021.intl.d.ts","../../../node_modules/typescript/lib/lib.es2022.array.d.ts","../../../node_modules/typescript/lib/lib.es2022.error.d.ts","../../../node_modules/typescript/lib/lib.es2022.intl.d.ts","../../../node_modules/typescript/lib/lib.es2022.object.d.ts","../../../node_modules/typescript/lib/lib.es2022.string.d.ts","../../../node_modules/typescript/lib/lib.es2022.regexp.d.ts","../../../node_modules/typescript/lib/lib.decorators.d.ts","../../../node_modules/typescript/lib/lib.decorators.legacy.d.ts","../../../node_modules/tslib/tslib.d.ts","../../../node_modules/tslib/modules/index.d.ts","../../../node_modules/zod/lib/helpers/typeAliases.d.ts","../../../node_modules/zod/lib/helpers/util.d.ts","../../../node_modules/zod/lib/ZodError.d.ts","../../../node_modules/zod/lib/locales/en.d.ts","../../../node_modules/zod/lib/errors.d.ts","../../../node_modules/zod/lib/helpers/parseUtil.d.ts","../../../node_modules/zod/lib/helpers/enumUtil.d.ts","../../../node_modules/zod/lib/helpers/errorUtil.d.ts","../../../node_modules/zod/lib/helpers/partialUtil.d.ts","../../../node_modules/zod/lib/types.d.ts","../../../node_modules/zod/lib/external.d.ts","../../../node_modules/zod/lib/index.d.ts","../../../node_modules/zod/index.d.ts","../src/types.ts","../src/pointers.ts","../src/refs.ts","../src/validation.ts","../src/index.ts","../../../node_modules/@types/node/compatibility/disposable.d.ts","../../../node_modules/@types/node/compatibility/indexable.d.ts","../../../node_modules/@types/node/compatibility/iterators.d.ts","../../../node_modules/@types/node/compatibility/index.d.ts","../../../node_modules/@types/node/globals.typedarray.d.ts","../../../node_modules/@types/node/buffer.buffer.d.ts","../../../node_modules/buffer/index.d.ts","../../../node_modules/undici-types/header.d.ts","../../../node_modules/undici-types/readable.d.ts","../../../node_modules/undici-types/file.d.ts","../../../node_modules/undici-types/fetch.d.ts","../../../node_modules/undici-types/formdata.d.ts","../../../node_modules/undici-types/connector.d.ts","../../../node_modules/undici-types/client.d.ts","../../../node_modules/undici-types/errors.d.ts","../../../node_modules/undici-types/dispatcher.d.ts","../../../node_modules/undici-types/global-dispatcher.d.ts","../../../node_modules/undici-types/global-origin.d.ts","../../../node_modules/undici-types/pool-stats.d.ts","../../../node_modules/undici-types/pool.d.ts","../../../node_modules/undici-types/handlers.d.ts","../../../node_modules/undici-types/balanced-pool.d.ts","../../../node_modules/undici-types/agent.d.ts","../../../node_modules/undici-types/mock-interceptor.d.ts","../../../node_modules/undici-types/mock-agent.d.ts","../../../node_modules/undici-types/mock-client.d.ts","../../../node_modules/undici-types/mock-pool.d.ts","../../../node_modules/undici-types/mock-errors.d.ts","../../../node_modules/undici-types/proxy-agent.d.ts","../../../node_modules/undici-types/env-http-proxy-agent.d.ts","../../../node_modules/undici-types/retry-handler.d.ts","../../../node_modules/undici-types/retry-agent.d.ts","../../../node_modules/undici-types/api.d.ts","../../../node_modules/undici-types/interceptors.d.ts","../../../node_modules/undici-types/util.d.ts","../../../node_modules/undici-types/cookies.d.ts","../../../node_modules/undici-types/patch.d.ts","../../../node_modules/undici-types/websocket.d.ts","../../../node_modules/undici-types/eventsource.d.ts","../../../node_modules/undici-types/filereader.d.ts","../../../node_modules/undici-types/diagnostics-channel.d.ts","../../../node_modules/undici-types/content-type.d.ts","../../../node_modules/undici-types/cache.d.ts","../../../node_modules/undici-types/index.d.ts","../../../node_modules/@types/node/globals.d.ts","../../../node_modules/@types/node/assert.d.ts","../../../node_modules/@types/node/assert/strict.d.ts","../../../node_modules/@types/node/async_hooks.d.ts","../../../node_modules/@types/node/buffer.d.ts","../../../node_modules/@types/node/child_process.d.ts","../../../node_modules/@types/node/cluster.d.ts","../../../node_modules/@types/node/console.d.ts","../../../node_modules/@types/node/constants.d.ts","../../../node_modules/@types/node/crypto.d.ts","../../../node_modules/@types/node/dgram.d.ts","../../../node_modules/@types/node/diagnostics_channel.d.ts","../../../node_modules/@types/node/dns.d.ts","../../../node_modules/@types/node/dns/promises.d.ts","../../../node_modules/@types/node/domain.d.ts","../../../node_modules/@types/node/dom-events.d.ts","../../../node_modules/@types/node/events.d.ts","../../../node_modules/@types/node/fs.d.ts","../../../node_modules/@types/node/fs/promises.d.ts","../../../node_modules/@types/node/http.d.ts","../../../node_modules/@types/node/http2.d.ts","../../../node_modules/@types/node/https.d.ts","../../../node_modules/@types/node/inspector.d.ts","../../../node_modules/@types/node/module.d.ts","../../../node_modules/@types/node/net.d.ts","../../../node_modules/@types/node/os.d.ts","../../../node_modules/@types/node/path.d.ts","../../../node_modules/@types/node/perf_hooks.d.ts","../../../node_modules/@types/node/process.d.ts","../../../node_modules/@types/node/punycode.d.ts","../../../node_modules/@types/node/querystring.d.ts","../../../node_modules/@types/node/readline.d.ts","../../../node_modules/@types/node/readline/promises.d.ts","../../../node_modules/@types/node/repl.d.ts","../../../node_modules/@types/node/sea.d.ts","../../../node_modules/@types/node/stream.d.ts","../../../node_modules/@types/node/stream/promises.d.ts","../../../node_modules/@types/node/stream/consumers.d.ts","../../../node_modules/@types/node/stream/web.d.ts","../../../node_modules/@types/node/string_decoder.d.ts","../../../node_modules/@types/node/test.d.ts","../../../node_modules/@types/node/timers.d.ts","../../../node_modules/@types/node/timers/promises.d.ts","../../../node_modules/@types/node/tls.d.ts","../../../node_modules/@types/node/trace_events.d.ts","../../../node_modules/@types/node/tty.d.ts","../../../node_modules/@types/node/url.d.ts","../../../node_modules/@types/node/util.d.ts","../../../node_modules/@types/node/v8.d.ts","../../../node_modules/@types/node/vm.d.ts","../../../node_modules/@types/node/wasi.d.ts","../../../node_modules/@types/node/worker_threads.d.ts","../../../node_modules/@types/node/zlib.d.ts","../../../node_modules/@types/node/index.d.ts"],"fileIdsList":[[59,73,74,75,76,83,126],[59,83,126],[59,73,83,126],[59,72,83,126],[59,74,75,83,126],[83,123,126],[83,125,126],[126],[83,126,131,160],[83,126,127,132,138,139,146,157,168],[83,126,127,128,138,146],[83,126],[78,79,80,83,126],[83,126,129,169],[83,126,130,131,139,147],[83,126,131,157,165],[83,126,132,134,138,146],[83,125,126,133],[83,126,134,135],[83,126,136,138],[83,125,126,138],[83,126,138,139,140,157,168],[83,126,138,139,140,153,157,160],[83,121,126],[83,126,134,138,141,146,157,168],[83,126,138,139,141,142,146,157,165,168],[83,126,141,143,157,165,168],[81,82,83,122,123,124,125,126,127,128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174],[83,126,138,144],[83,126,145,168,173],[83,126,134,138,146,157],[83,126,147],[83,126,148],[83,125,126,149],[83,123,124,125,126,127,128,129,130,131,132,133,134,135,136,138,139,140,141,142,143,144,145,146,147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,165,166,167,168,169,170,171,172,173,174],[83,126,151],[83,126,152],[83,126,138,153,154],[83,126,153,155,169,171],[83,126,138,157,158,160],[83,126,159,160],[83,126,157,158],[83,126,160],[83,126,161],[83,123,126,157,162],[83,126,138,163,164],[83,126,163,164],[83,126,131,146,157,165],[83,126,166],[83,126,146,167],[83,126,141,152,168],[83,126,131,169],[83,126,157,170],[83,126,145,171],[83,126,172],[83,126,138,140,149,157,160,168,171,173],[83,126,157,174],[58,83,126],[83,93,97,126,168],[83,93,126,157,168],[83,88,126],[83,90,93,126,165,168],[83,126,146,165],[83,126,175],[83,88,126,175],[83,90,93,126,146,168],[83,85,86,89,92,126,138,157,168],[83,93,100,126],[83,85,91,126],[83,93,114,115,126],[83,89,93,126,160,168,175],[83,114,126,175],[83,87,88,126,175],[83,93,126],[83,87,88,89,90,91,92,93,94,95,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,115,116,117,118,119,120,126],[83,93,108,126],[83,93,100,101,126],[83,91,93,101,102,126],[83,92,126],[83,85,88,93,126],[83,93,97,101,102,126],[83,97,126],[83,91,93,96,126,168],[83,85,90,93,100,126],[83,126,157],[83,88,93,114,126,173,175],[71,83,126],[60,61,71,83,126],[62,63,83,126],[60,61,62,64,65,69,83,126],[61,62,83,126],[70,83,126],[62,83,126],[60,61,62,65,66,67,68,83,126]],"fileInfos":[{"version":"c430d44666289dae81f30fa7b2edebf186ecc91a2d4c71266ea6ae76388792e1","affectsGlobalScope":true,"impliedFormat":1},{"version":"45b7ab580deca34ae9729e97c13cfd999df04416a79116c3bfb483804f85ded4","impliedFormat":1},{"version":"3facaf05f0c5fc569c5649dd359892c98a85557e3e0c847964caeb67076f4d75","impliedFormat":1},{"version":"e44bb8bbac7f10ecc786703fe0a6a4b952189f908707980ba8f3c8975a760962","impliedFormat":1},{"version":"5e1c4c362065a6b95ff952c0eab010f04dcd2c3494e813b493ecfd4fcb9fc0d8","impliedFormat":1},{"version":"68d73b4a11549f9c0b7d352d10e91e5dca8faa3322bfb77b661839c42b1ddec7","impliedFormat":1},{"version":"5efce4fc3c29ea84e8928f97adec086e3dc876365e0982cc8479a07954a3efd4","impliedFormat":1},{"version":"feecb1be483ed332fad555aff858affd90a48ab19ba7272ee084704eb7167569","impliedFormat":1},{"version":"ee7bad0c15b58988daa84371e0b89d313b762ab83cb5b31b8a2d1162e8eb41c2","impliedFormat":1},{"version":"c57796738e7f83dbc4b8e65132f11a377649c00dd3eee333f672b8f0a6bea671","affectsGlobalScope":true,"impliedFormat":1},{"version":"dc2df20b1bcdc8c2d34af4926e2c3ab15ffe1160a63e58b7e09833f616efff44","affectsGlobalScope":true,"impliedFormat":1},{"version":"515d0b7b9bea2e31ea4ec968e9edd2c39d3eebf4a2d5cbd04e88639819ae3b71","affectsGlobalScope":true,"impliedFormat":1},{"version":"0559b1f683ac7505ae451f9a96ce4c3c92bdc71411651ca6ddb0e88baaaad6a3","affectsGlobalScope":true,"impliedFormat":1},{"version":"0dc1e7ceda9b8b9b455c3a2d67b0412feab00bd2f66656cd8850e8831b08b537","affectsGlobalScope":true,"impliedFormat":1},{"version":"ce691fb9e5c64efb9547083e4a34091bcbe5bdb41027e310ebba8f7d96a98671","affectsGlobalScope":true,"impliedFormat":1},{"version":"8d697a2a929a5fcb38b7a65594020fcef05ec1630804a33748829c5ff53640d0","affectsGlobalScope":true,"impliedFormat":1},{"version":"4ff2a353abf8a80ee399af572debb8faab2d33ad38c4b4474cff7f26e7653b8d","affectsGlobalScope":true,"impliedFormat":1},{"version":"fb0f136d372979348d59b3f5020b4cdb81b5504192b1cacff5d1fbba29378aa1","affectsGlobalScope":true,"impliedFormat":1},{"version":"d15bea3d62cbbdb9797079416b8ac375ae99162a7fba5de2c6c505446486ac0a","affectsGlobalScope":true,"impliedFormat":1},{"version":"68d18b664c9d32a7336a70235958b8997ebc1c3b8505f4f1ae2b7e7753b87618","affectsGlobalScope":true,"impliedFormat":1},{"version":"eb3d66c8327153d8fa7dd03f9c58d351107fe824c79e9b56b462935176cdf12a","affectsGlobalScope":true,"impliedFormat":1},{"version":"38f0219c9e23c915ef9790ab1d680440d95419ad264816fa15009a8851e79119","affectsGlobalScope":true,"impliedFormat":1},{"version":"69ab18c3b76cd9b1be3d188eaf8bba06112ebbe2f47f6c322b5105a6fbc45a2e","affectsGlobalScope":true,"impliedFormat":1},{"version":"a680117f487a4d2f30ea46f1b4b7f58bef1480456e18ba53ee85c2746eeca012","affectsGlobalScope":true,"impliedFormat":1},{"version":"2f11ff796926e0832f9ae148008138ad583bd181899ab7dd768a2666700b1893","affectsGlobalScope":true,"impliedFormat":1},{"version":"4de680d5bb41c17f7f68e0419412ca23c98d5749dcaaea1896172f06435891fc","affectsGlobalScope":true,"impliedFormat":1},{"version":"954296b30da6d508a104a3a0b5d96b76495c709785c1d11610908e63481ee667","affectsGlobalScope":true,"impliedFormat":1},{"version":"ac9538681b19688c8eae65811b329d3744af679e0bdfa5d842d0e32524c73e1c","affectsGlobalScope":true,"impliedFormat":1},{"version":"0a969edff4bd52585473d24995c5ef223f6652d6ef46193309b3921d65dd4376","affectsGlobalScope":true,"impliedFormat":1},{"version":"9e9fbd7030c440b33d021da145d3232984c8bb7916f277e8ffd3dc2e3eae2bdb","affectsGlobalScope":true,"impliedFormat":1},{"version":"811ec78f7fefcabbda4bfa93b3eb67d9ae166ef95f9bff989d964061cbf81a0c","affectsGlobalScope":true,"impliedFormat":1},{"version":"717937616a17072082152a2ef351cb51f98802fb4b2fdabd32399843875974ca","affectsGlobalScope":true,"impliedFormat":1},{"version":"d7e7d9b7b50e5f22c915b525acc5a49a7a6584cf8f62d0569e557c5cfc4b2ac2","affectsGlobalScope":true,"impliedFormat":1},{"version":"71c37f4c9543f31dfced6c7840e068c5a5aacb7b89111a4364b1d5276b852557","affectsGlobalScope":true,"impliedFormat":1},{"version":"576711e016cf4f1804676043e6a0a5414252560eb57de9faceee34d79798c850","affectsGlobalScope":true,"impliedFormat":1},{"version":"89c1b1281ba7b8a96efc676b11b264de7a8374c5ea1e6617f11880a13fc56dc6","affectsGlobalScope":true,"impliedFormat":1},{"version":"74f7fa2d027d5b33eb0471c8e82a6c87216223181ec31247c357a3e8e2fddc5b","affectsGlobalScope":true,"impliedFormat":1},{"version":"d6d7ae4d1f1f3772e2a3cde568ed08991a8ae34a080ff1151af28b7f798e22ca","affectsGlobalScope":true,"impliedFormat":1},{"version":"063600664504610fe3e99b717a1223f8b1900087fab0b4cad1496a114744f8df","affectsGlobalScope":true,"impliedFormat":1},{"version":"934019d7e3c81950f9a8426d093458b65d5aff2c7c1511233c0fd5b941e608ab","affectsGlobalScope":true,"impliedFormat":1},{"version":"52ada8e0b6e0482b728070b7639ee42e83a9b1c22d205992756fe020fd9f4a47","affectsGlobalScope":true,"impliedFormat":1},{"version":"3bdefe1bfd4d6dee0e26f928f93ccc128f1b64d5d501ff4a8cf3c6371200e5e6","affectsGlobalScope":true,"impliedFormat":1},{"version":"59fb2c069260b4ba00b5643b907ef5d5341b167e7d1dbf58dfd895658bda2867","affectsGlobalScope":true,"impliedFormat":1},{"version":"639e512c0dfc3fad96a84caad71b8834d66329a1f28dc95e3946c9b58176c73a","affectsGlobalScope":true,"impliedFormat":1},{"version":"368af93f74c9c932edd84c58883e736c9e3d53cec1fe24c0b0ff451f529ceab1","affectsGlobalScope":true,"impliedFormat":1},{"version":"af3dd424cf267428f30ccfc376f47a2c0114546b55c44d8c0f1d57d841e28d74","affectsGlobalScope":true,"impliedFormat":1},{"version":"995c005ab91a498455ea8dfb63aa9f83fa2ea793c3d8aa344be4a1678d06d399","affectsGlobalScope":true,"impliedFormat":1},{"version":"959d36cddf5e7d572a65045b876f2956c973a586da58e5d26cde519184fd9b8a","affectsGlobalScope":true,"impliedFormat":1},{"version":"965f36eae237dd74e6cca203a43e9ca801ce38824ead814728a2807b1910117d","affectsGlobalScope":true,"impliedFormat":1},{"version":"3925a6c820dcb1a06506c90b1577db1fdbf7705d65b62b99dce4be75c637e26b","affectsGlobalScope":true,"impliedFormat":1},{"version":"0a3d63ef2b853447ec4f749d3f368ce642264246e02911fcb1590d8c161b8005","affectsGlobalScope":true,"impliedFormat":1},{"version":"8cdf8847677ac7d20486e54dd3fcf09eda95812ac8ace44b4418da1bbbab6eb8","affectsGlobalScope":true,"impliedFormat":1},{"version":"8444af78980e3b20b49324f4a16ba35024fef3ee069a0eb67616ea6ca821c47a","affectsGlobalScope":true,"impliedFormat":1},{"version":"3287d9d085fbd618c3971944b65b4be57859f5415f495b33a6adc994edd2f004","affectsGlobalScope":true,"impliedFormat":1},{"version":"b4b67b1a91182421f5df999988c690f14d813b9850b40acd06ed44691f6727ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e7f8264d0fb4c5339605a15daadb037bf238c10b654bb3eee14208f860a32ea","affectsGlobalScope":true,"impliedFormat":1},{"version":"782dec38049b92d4e85c1585fbea5474a219c6984a35b004963b00beb1aab538","affectsGlobalScope":true,"impliedFormat":1},{"version":"a6a5253138c5432c68a1510c70fe78a644fe2e632111ba778e1978010d6edfec","impliedFormat":1},{"version":"b8f34dd1757f68e03262b1ca3ddfa668a855b872f8bdd5224d6f993a7b37dc2c","impliedFormat":99},{"version":"5487b97cfa28b26b4a9ef0770f872bdbebd4c46124858de00f242c3eed7519f4","impliedFormat":1},{"version":"c2869c4f2f79fd2d03278a68ce7c061a5a8f4aed59efb655e25fe502e3e471d5","impliedFormat":1},{"version":"b8fe42dbf4b0efba2eb4dbfb2b95a3712676717ff8469767dc439e75d0c1a3b6","impliedFormat":1},{"version":"8485b6da53ec35637d072e516631d25dae53984500de70a6989058f24354666f","impliedFormat":1},{"version":"ebe80346928736532e4a822154eb77f57ef3389dbe2b3ba4e571366a15448ef2","impliedFormat":1},{"version":"83306c97a4643d78420f082547ea0d488a0d134c922c8e65fc0b4f08ef66d92b","impliedFormat":1},{"version":"f672c876c1a04a223cf2023b3d91e8a52bb1544c576b81bf64a8fec82be9969c","impliedFormat":1},{"version":"98a9cc18f661d28e6bd31c436e1984f3980f35e0f0aa9cf795c54f8ccb667ffe","impliedFormat":1},{"version":"c76b0c5727302341d0bdfa2cc2cee4b19ff185b554edb6e8543f0661d8487116","impliedFormat":1},{"version":"dccd26a5c85325a011aff40f401e0892bd0688d44132ba79e803c67e68fffea5","impliedFormat":1},{"version":"f5ef066942e4f0bd98200aa6a6694b831e73200c9b3ade77ad0aa2409e8fe1b1","impliedFormat":1},{"version":"b9e99cd94f4166a245f5158f7286c05406e2a4c694619bceb7a4f3519d1d768e","impliedFormat":1},{"version":"5568d7c32e5cf5f35e092649f4e5e168c3114c800b1d7545b7ae5e0415704802","impliedFormat":1},{"version":"bf9e372b10ad2149a7be4e10c8b898247ec4122e4bad0d902974091712b00ecf","signature":"7563347776a581a6c1f41701ce67a754cd77823d401d18c0092b7de976d3c24a","impliedFormat":99},{"version":"79fd1ba9d5a6d5bb82347528e0f0881d0661bfc9ba7903864406d0e87e510f59","signature":"a67c55ae7f374f6a416528623daba3414f80d628e0401e5fa73de46eccabf725","impliedFormat":99},{"version":"ab3a301990edad9b5a66e3933c4409d8a7c6b47f6fbfe64e2e9e280ab61d03ee","signature":"a920dcfd6e216624c50e454743507c595f840796e6ff4091a9761bea431cfa42","impliedFormat":99},{"version":"d654d500a05dcb2d746f86e97fe501ae03762f67e2207d5c35e5c10e908ab20f","signature":"a3fac0b5fa2e07a68fc0f1dbce9e34eb5399b93163a02e740b99cf3d77094dd1","impliedFormat":99},{"version":"a1f7eab5857531a8fba4ae102c7c0e80e3ed1b0bac5d0659bf51ac692624f962","impliedFormat":99},{"version":"70521b6ab0dcba37539e5303104f29b721bfb2940b2776da4cc818c07e1fefc1","affectsGlobalScope":true,"impliedFormat":1},{"version":"ab41ef1f2cdafb8df48be20cd969d875602483859dc194e9c97c8a576892c052","affectsGlobalScope":true,"impliedFormat":1},{"version":"d153a11543fd884b596587ccd97aebbeed950b26933ee000f94009f1ab142848","affectsGlobalScope":true,"impliedFormat":1},{"version":"21d819c173c0cf7cc3ce57c3276e77fd9a8a01d35a06ad87158781515c9a438a","impliedFormat":1},{"version":"a79e62f1e20467e11a904399b8b18b18c0c6eea6b50c1168bf215356d5bebfaf","affectsGlobalScope":true,"impliedFormat":1},{"version":"49a5a44f2e68241a1d2bd9ec894535797998841c09729e506a7cbfcaa40f2180","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e9c23ba78aabc2e0a27033f18737a6df754067731e69dc5f52823957d60a4b6","impliedFormat":1},{"version":"5929864ce17fba74232584d90cb721a89b7ad277220627cc97054ba15a98ea8f","impliedFormat":1},{"version":"763fe0f42b3d79b440a9b6e51e9ba3f3f91352469c1e4b3b67bfa4ff6352f3f4","impliedFormat":1},{"version":"25c8056edf4314820382a5fdb4bb7816999acdcb929c8f75e3f39473b87e85bc","impliedFormat":1},{"version":"c464d66b20788266e5353b48dc4aa6bc0dc4a707276df1e7152ab0c9ae21fad8","impliedFormat":1},{"version":"78d0d27c130d35c60b5e5566c9f1e5be77caf39804636bc1a40133919a949f21","impliedFormat":1},{"version":"c6fd2c5a395f2432786c9cb8deb870b9b0e8ff7e22c029954fabdd692bff6195","impliedFormat":1},{"version":"1d6e127068ea8e104a912e42fc0a110e2aa5a66a356a917a163e8cf9a65e4a75","impliedFormat":1},{"version":"5ded6427296cdf3b9542de4471d2aa8d3983671d4cac0f4bf9c637208d1ced43","impliedFormat":1},{"version":"7f182617db458e98fc18dfb272d40aa2fff3a353c44a89b2c0ccb3937709bfb5","impliedFormat":1},{"version":"cadc8aced301244057c4e7e73fbcae534b0f5b12a37b150d80e5a45aa4bebcbd","impliedFormat":1},{"version":"385aab901643aa54e1c36f5ef3107913b10d1b5bb8cbcd933d4263b80a0d7f20","impliedFormat":1},{"version":"9670d44354bab9d9982eca21945686b5c24a3f893db73c0dae0fd74217a4c219","impliedFormat":1},{"version":"0b8a9268adaf4da35e7fa830c8981cfa22adbbe5b3f6f5ab91f6658899e657a7","impliedFormat":1},{"version":"11396ed8a44c02ab9798b7dca436009f866e8dae3c9c25e8c1fbc396880bf1bb","impliedFormat":1},{"version":"ba7bc87d01492633cb5a0e5da8a4a42a1c86270e7b3d2dea5d156828a84e4882","impliedFormat":1},{"version":"4893a895ea92c85345017a04ed427cbd6a1710453338df26881a6019432febdd","impliedFormat":1},{"version":"c21dc52e277bcfc75fac0436ccb75c204f9e1b3fa5e12729670910639f27343e","impliedFormat":1},{"version":"13f6f39e12b1518c6650bbb220c8985999020fe0f21d818e28f512b7771d00f9","impliedFormat":1},{"version":"9b5369969f6e7175740bf51223112ff209f94ba43ecd3bb09eefff9fd675624a","impliedFormat":1},{"version":"4fe9e626e7164748e8769bbf74b538e09607f07ed17c2f20af8d680ee49fc1da","impliedFormat":1},{"version":"24515859bc0b836719105bb6cc3d68255042a9f02a6022b3187948b204946bd2","impliedFormat":1},{"version":"ea0148f897b45a76544ae179784c95af1bd6721b8610af9ffa467a518a086a43","impliedFormat":1},{"version":"24c6a117721e606c9984335f71711877293a9651e44f59f3d21c1ea0856f9cc9","impliedFormat":1},{"version":"dd3273ead9fbde62a72949c97dbec2247ea08e0c6952e701a483d74ef92d6a17","impliedFormat":1},{"version":"405822be75ad3e4d162e07439bac80c6bcc6dbae1929e179cf467ec0b9ee4e2e","impliedFormat":1},{"version":"0db18c6e78ea846316c012478888f33c11ffadab9efd1cc8bcc12daded7a60b6","impliedFormat":1},{"version":"e61be3f894b41b7baa1fbd6a66893f2579bfad01d208b4ff61daef21493ef0a8","impliedFormat":1},{"version":"bd0532fd6556073727d28da0edfd1736417a3f9f394877b6d5ef6ad88fba1d1a","impliedFormat":1},{"version":"89167d696a849fce5ca508032aabfe901c0868f833a8625d5a9c6e861ef935d2","impliedFormat":1},{"version":"615ba88d0128ed16bf83ef8ccbb6aff05c3ee2db1cc0f89ab50a4939bfc1943f","impliedFormat":1},{"version":"a4d551dbf8746780194d550c88f26cf937caf8d56f102969a110cfaed4b06656","impliedFormat":1},{"version":"8bd86b8e8f6a6aa6c49b71e14c4ffe1211a0e97c80f08d2c8cc98838006e4b88","impliedFormat":1},{"version":"317e63deeb21ac07f3992f5b50cdca8338f10acd4fbb7257ebf56735bf52ab00","impliedFormat":1},{"version":"4732aec92b20fb28c5fe9ad99521fb59974289ed1e45aecb282616202184064f","impliedFormat":1},{"version":"2e85db9e6fd73cfa3d7f28e0ab6b55417ea18931423bd47b409a96e4a169e8e6","impliedFormat":1},{"version":"c46e079fe54c76f95c67fb89081b3e399da2c7d109e7dca8e4b58d83e332e605","impliedFormat":1},{"version":"bf67d53d168abc1298888693338cb82854bdb2e69ef83f8a0092093c2d562107","impliedFormat":1},{"version":"1ca84b44ad1d8e4576f24904d8b95dd23b94ea67e1575f89614ac90062fc67f4","affectsGlobalScope":true,"impliedFormat":1},{"version":"6d586db0a09a9495ebb5dece28f54df9684bfbd6e1f568426ca153126dac4a40","impliedFormat":1},{"version":"7394959e5a741b185456e1ef5d64599c36c60a323207450991e7a42e08911419","impliedFormat":1},{"version":"8c0bcd6c6b67b4b503c11e91a1fb91522ed585900eab2ab1f61bba7d7caa9d6f","impliedFormat":1},{"version":"567b7f607f400873151d7bc63a049514b53c3c00f5f56e9e95695d93b66a138e","affectsGlobalScope":true,"impliedFormat":1},{"version":"f3e58c4c18a031cbb17abec7a4ad0bd5ae9fc70c1f4ba1e7fb921ad87c504aca","impliedFormat":1},{"version":"84c1930e33d1bb12ad01bcbe11d656f9646bd21b2fb2afd96e8e10615a021aef","impliedFormat":1},{"version":"35ec8b6760fd7138bbf5809b84551e31028fb2ba7b6dc91d95d098bf212ca8b4","affectsGlobalScope":true,"impliedFormat":1},{"version":"5524481e56c48ff486f42926778c0a3cce1cc85dc46683b92b1271865bcf015a","impliedFormat":1},{"version":"4b87f767c7bc841511113c876a6b8bf1fd0cb0b718c888ad84478b372ec486b1","affectsGlobalScope":true,"impliedFormat":1},{"version":"8d04e3640dd9eb67f7f1e5bd3d0bf96c784666f7aefc8ac1537af6f2d38d4c29","impliedFormat":1},{"version":"9d19808c8c291a9010a6c788e8532a2da70f811adb431c97520803e0ec649991","impliedFormat":1},{"version":"2bf469abae4cc9c0f340d4e05d9d26e37f936f9c8ca8f007a6534f109dcc77e4","impliedFormat":1},{"version":"4aacb0dd020eeaef65426153686cc639a78ec2885dc72ad220be1d25f1a439df","impliedFormat":1},{"version":"f0bd7e6d931657b59605c44112eaf8b980ba7f957a5051ed21cb93d978cf2f45","impliedFormat":1},{"version":"71450bbc2d82821d24ca05699a533e72758964e9852062c53b30f31c36978ab8","affectsGlobalScope":true,"impliedFormat":1},{"version":"0ada07543808f3b967624645a8e1ccd446f8b01ade47842acf1328aec899fed0","affectsGlobalScope":true,"impliedFormat":1},{"version":"4c21aaa8257d7950a5b75a251d9075b6a371208fc948c9c8402f6690ef3b5b55","impliedFormat":1},{"version":"b5895e6353a5d708f55d8685c38a235c3a6d8138e374dee8ceb8ffde5aa8002a","impliedFormat":1},{"version":"54c4f21f578864961efc94e8f42bc893a53509e886370ec7dd602e0151b9266c","impliedFormat":1},{"version":"de735eca2c51dd8b860254e9fdb6d9ec19fe402dfe597c23090841ce3937cfc5","impliedFormat":1},{"version":"4ff41188773cbf465807dd2f7059c7494cbee5115608efc297383832a1150c43","impliedFormat":1},{"version":"5650cf3dace09e7c25d384e3e6b818b938f68f4e8de96f52d9c5a1b3db068e86","impliedFormat":1},{"version":"1354ca5c38bd3fd3836a68e0f7c9f91f172582ba30ab15bb8c075891b91502b7","affectsGlobalScope":true,"impliedFormat":1},{"version":"5155da3047ef977944d791a2188ff6e6c225f6975cc1910ab7bb6838ab84cede","impliedFormat":1},{"version":"93f437e1398a4f06a984f441f7fa7a9f0535c04399619b5c22e0b87bdee182cb","impliedFormat":1},{"version":"afbe24ab0d74694372baa632ecb28bb375be53f3be53f9b07ecd7fc994907de5","impliedFormat":1},{"version":"e16d218a30f6a6810b57f7e968124eaa08c7bb366133ea34bbf01e7cd6b8c0ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"eb8692dea24c27821f77e397272d9ed2eda0b95e4a75beb0fdda31081d15a8ae","affectsGlobalScope":true,"impliedFormat":1},{"version":"9e043a1bc8fbf2a255bccf9bf27e0f1caf916c3b0518ea34aa72357c0afd42ec","impliedFormat":1},{"version":"b4f70ec656a11d570e1a9edce07d118cd58d9760239e2ece99306ee9dfe61d02","impliedFormat":1},{"version":"3bc2f1e2c95c04048212c569ed38e338873f6a8593930cf5a7ef24ffb38fc3b6","impliedFormat":1},{"version":"8145e07aad6da5f23f2fcd8c8e4c5c13fb26ee986a79d03b0829b8fce152d8b2","impliedFormat":1},{"version":"f9d9d753d430ed050dc1bf2667a1bab711ccbb1c1507183d794cc195a5b085cc","impliedFormat":1},{"version":"9eece5e586312581ccd106d4853e861aaaa1a39f8e3ea672b8c3847eedd12f6e","impliedFormat":1},{"version":"5b6844ad931dcc1d3aca53268f4bd671428421464b1286746027aede398094f2","impliedFormat":1},{"version":"37ba7b45141a45ce6e80e66f2a96c8a5ab1bcef0fc2d0f56bb58df96ec67e972","impliedFormat":1},{"version":"125d792ec6c0c0f657d758055c494301cc5fdb327d9d9d5960b3f129aff76093","impliedFormat":1},{"version":"0dbcebe2126d03936c70545e96a6e41007cf065be38a1ce4d32a39fcedefead4","affectsGlobalScope":true,"impliedFormat":1},{"version":"1851a3b4db78664f83901bb9cac9e45e03a37bb5933cc5bf37e10bb7e91ab4eb","impliedFormat":1},{"version":"461e54289e6287e8494a0178ba18182acce51a02bca8dea219149bf2cf96f105","impliedFormat":1},{"version":"12ed4559eba17cd977aa0db658d25c4047067444b51acfdcbf38470630642b23","affectsGlobalScope":true,"impliedFormat":1},{"version":"f3ffabc95802521e1e4bcba4c88d8615176dc6e09111d920c7a213bdda6e1d65","impliedFormat":1},{"version":"e31e51c55800014d926e3f74208af49cb7352803619855c89296074d1ecbb524","impliedFormat":1},{"version":"ae56f65caf3be91108707bd8dfbccc2a57a91feb5daabf7165a06a945545ed26","impliedFormat":1},{"version":"a136d5de521da20f31631a0a96bf712370779d1c05b7015d7019a9b2a0446ca9","impliedFormat":1},{"version":"dfb96ba5177b68003deec9e773c47257da5c4c8a74053d8956389d832df72002","affectsGlobalScope":true,"impliedFormat":1},{"version":"92d3070580cf72b4bb80959b7f16ede9a3f39e6f4ef2ac87cfa4561844fdc69f","affectsGlobalScope":true,"impliedFormat":1},{"version":"d3dffd70e6375b872f0b4e152de4ae682d762c61a24881ecc5eb9f04c5caf76f","impliedFormat":1},{"version":"613deebaec53731ff6b74fe1a89f094b708033db6396b601df3e6d5ab0ec0a47","impliedFormat":1},{"version":"d91a7d8b5655c42986f1bdfe2105c4408f472831c8f20cf11a8c3345b6b56c8c","impliedFormat":1},{"version":"e56eb632f0281c9f8210eb8c86cc4839a427a4ffffcfd2a5e40b956050b3e042","affectsGlobalScope":true,"impliedFormat":1},{"version":"e8a979b8af001c9fc2e774e7809d233c8ca955a28756f52ee5dee88ccb0611d2","impliedFormat":1},{"version":"cac793cc47c29e26e4ac3601dcb00b4435ebed26203485790e44f2ad8b6ad847","impliedFormat":1}],"root":[[73,77]],"options":{"allowJs":false,"allowSyntheticDefaultImports":true,"composite":true,"declaration":true,"declarationMap":true,"emitDeclarationOnly":false,"emitDecoratorMetadata":false,"esModuleInterop":true,"experimentalDecorators":false,"importHelpers":true,"module":199,"noEmitOnError":true,"noFallthroughCasesInSwitch":true,"noImplicitOverride":true,"noImplicitReturns":true,"noUnusedLocals":true,"outDir":"./","removeComments":false,"rootDir":"../src","skipDefaultLibCheck":false,"skipLibCheck":true,"sourceMap":false,"strict":true,"target":9,"tsBuildInfoFile":"./tsconfig.lib.tsbuildinfo","verbatimModuleSyntax":false},"referencedMap":[[77,1],[74,2],[75,3],[73,4],[76,5],[123,6],[124,6],[125,7],[83,8],[126,9],[127,10],[128,11],[78,12],[81,13],[79,12],[80,12],[129,14],[130,15],[131,16],[132,17],[133,18],[134,19],[135,19],[137,12],[136,20],[138,21],[139,22],[140,23],[122,24],[82,12],[141,25],[142,26],[143,27],[175,28],[144,29],[145,30],[146,31],[147,32],[148,33],[149,34],[150,35],[151,36],[152,37],[153,38],[154,38],[155,39],[156,12],[157,40],[159,41],[158,42],[160,43],[161,44],[162,45],[163,46],[164,47],[165,48],[166,49],[167,50],[168,51],[169,52],[170,53],[171,54],[172,55],[173,56],[174,57],[84,12],[59,58],[58,12],[56,12],[57,12],[11,12],[10,12],[2,12],[12,12],[13,12],[14,12],[15,12],[16,12],[17,12],[18,12],[19,12],[3,12],[20,12],[21,12],[4,12],[22,12],[26,12],[23,12],[24,12],[25,12],[27,12],[28,12],[29,12],[5,12],[30,12],[31,12],[32,12],[33,12],[6,12],[37,12],[34,12],[35,12],[36,12],[38,12],[7,12],[39,12],[44,12],[45,12],[40,12],[41,12],[42,12],[43,12],[8,12],[49,12],[46,12],[47,12],[48,12],[50,12],[9,12],[51,12],[52,12],[53,12],[55,12],[54,12],[1,12],[100,59],[110,60],[99,59],[120,61],[91,62],[90,63],[119,64],[113,65],[118,66],[93,67],[107,68],[92,69],[116,70],[88,71],[87,64],[117,72],[89,73],[94,74],[95,12],[98,74],[85,12],[121,75],[111,76],[102,77],[103,78],[105,79],[101,80],[104,81],[114,64],[96,82],[97,83],[106,84],[86,85],[109,76],[108,74],[112,12],[115,86],[72,87],[62,88],[64,89],[70,90],[66,12],[67,12],[65,91],[68,87],[60,12],[61,12],[71,92],[63,93],[69,94]],"latestChangedDtsFile":"./index.d.ts","version":"5.9.3"}
@@ -0,0 +1,46 @@
1
+ import type { ZodTypeAny } from 'zod';
2
+ export type AnyZodSchema = ZodTypeAny;
3
+ export type RepoName = string;
4
+ export type RepoBlueId = string;
5
+ export type PackageName = string;
6
+ export type TypeBlueId = string;
7
+ export type TypeAlias = string;
8
+ export type BlueRepositoryStatus = 'stable' | 'dev';
9
+ export interface BlueRepositoryVersionEntry {
10
+ repositoryVersionIndex: number;
11
+ typeBlueId: TypeBlueId;
12
+ attributesAdded: string[];
13
+ }
14
+ export interface BlueTypeRuntimeMeta {
15
+ status: BlueRepositoryStatus;
16
+ name: string;
17
+ versions: readonly BlueRepositoryVersionEntry[];
18
+ }
19
+ export interface BlueRepositoryPackage<ContentT = unknown, SchemaT = AnyZodSchema> {
20
+ name: PackageName;
21
+ aliases: Record<TypeAlias, TypeBlueId>;
22
+ typesMeta: Record<TypeBlueId, BlueTypeRuntimeMeta>;
23
+ contents: Record<TypeBlueId, ContentT>;
24
+ schemas: Record<TypeBlueId, SchemaT>;
25
+ }
26
+ export interface BlueRepository<ContentT = unknown, SchemaT = AnyZodSchema> {
27
+ name: RepoName;
28
+ repositoryVersions: readonly RepoBlueId[];
29
+ packages: Record<PackageName, BlueRepositoryPackage<ContentT, SchemaT>>;
30
+ }
31
+ export type BlueTypeVersion = BlueRepositoryVersionEntry;
32
+ export interface BlueTypeMetadata<ContentT = unknown> {
33
+ status: BlueRepositoryStatus;
34
+ content: ContentT;
35
+ versions: BlueTypeVersion[];
36
+ }
37
+ export interface BluePackage<ContentT = unknown> {
38
+ name: PackageName;
39
+ types: BlueTypeMetadata<ContentT>[];
40
+ }
41
+ export interface BlueRepositoryDocument<ContentT = unknown> {
42
+ name: RepoName;
43
+ packages: BluePackage<ContentT>[];
44
+ repositoryVersions: RepoBlueId[];
45
+ }
46
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,KAAK,CAAC;AAEtC,MAAM,MAAM,YAAY,GAAG,UAAU,CAAC;AAEtC,MAAM,MAAM,QAAQ,GAAG,MAAM,CAAC;AAC9B,MAAM,MAAM,UAAU,GAAG,MAAM,CAAC;AAChC,MAAM,MAAM,WAAW,GAAG,MAAM,CAAC;AACjC,MAAM,MAAM,UAAU,GAAG,MAAM,CAAC;AAChC,MAAM,MAAM,SAAS,GAAG,MAAM,CAAC;AAE/B,MAAM,MAAM,oBAAoB,GAAG,QAAQ,GAAG,KAAK,CAAC;AAEpD,MAAM,WAAW,0BAA0B;IACzC,sBAAsB,EAAE,MAAM,CAAC;IAC/B,UAAU,EAAE,UAAU,CAAC;IACvB,eAAe,EAAE,MAAM,EAAE,CAAC;CAC3B;AAED,MAAM,WAAW,mBAAmB;IAClC,MAAM,EAAE,oBAAoB,CAAC;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,SAAS,0BAA0B,EAAE,CAAC;CACjD;AAED,MAAM,WAAW,qBAAqB,CACpC,QAAQ,GAAG,OAAO,EAClB,OAAO,GAAG,YAAY;IAEtB,IAAI,EAAE,WAAW,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;IACvC,SAAS,EAAE,MAAM,CAAC,UAAU,EAAE,mBAAmB,CAAC,CAAC;IACnD,QAAQ,EAAE,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;IACvC,OAAO,EAAE,MAAM,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;CACtC;AAED,MAAM,WAAW,cAAc,CAAC,QAAQ,GAAG,OAAO,EAAE,OAAO,GAAG,YAAY;IACxE,IAAI,EAAE,QAAQ,CAAC;IACf,kBAAkB,EAAE,SAAS,UAAU,EAAE,CAAC;IAC1C,QAAQ,EAAE,MAAM,CAAC,WAAW,EAAE,qBAAqB,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,CAAC;CACzE;AAED,MAAM,MAAM,eAAe,GAAG,0BAA0B,CAAC;AAEzD,MAAM,WAAW,gBAAgB,CAAC,QAAQ,GAAG,OAAO;IAClD,MAAM,EAAE,oBAAoB,CAAC;IAC7B,OAAO,EAAE,QAAQ,CAAC;IAClB,QAAQ,EAAE,eAAe,EAAE,CAAC;CAC7B;AAED,MAAM,WAAW,WAAW,CAAC,QAAQ,GAAG,OAAO;IAC7C,IAAI,EAAE,WAAW,CAAC;IAClB,KAAK,EAAE,gBAAgB,CAAC,QAAQ,CAAC,EAAE,CAAC;CACrC;AAED,MAAM,WAAW,sBAAsB,CAAC,QAAQ,GAAG,OAAO;IACxD,IAAI,EAAE,QAAQ,CAAC;IACf,QAAQ,EAAE,WAAW,CAAC,QAAQ,CAAC,EAAE,CAAC;IAClC,kBAAkB,EAAE,UAAU,EAAE,CAAC;CAClC"}
package/dist/types.js ADDED
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,3 @@
1
+ export { collectTypeRefsFromContent, validateNoCycles, validateStableDoesNotDependOnDev, } from './refs.js';
2
+ export { parsePointer, unescapePointerToken, validatePointer, validateAttributesAddedPointer, RESERVED_ATTRIBUTES_POINTER_SEGMENTS, InvalidRepositoryPointerError, } from './pointers.js';
3
+ //# sourceMappingURL=validation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.d.ts","sourceRoot":"","sources":["../src/validation.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,0BAA0B,EAC1B,gBAAgB,EAChB,gCAAgC,GACjC,MAAM,WAAW,CAAC;AACnB,OAAO,EACL,YAAY,EACZ,oBAAoB,EACpB,eAAe,EACf,8BAA8B,EAC9B,oCAAoC,EACpC,6BAA6B,GAC9B,MAAM,eAAe,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { collectTypeRefsFromContent, validateNoCycles, validateStableDoesNotDependOnDev, } from './refs.js';
2
+ export { parsePointer, unescapePointerToken, validatePointer, validateAttributesAddedPointer, RESERVED_ATTRIBUTES_POINTER_SEGMENTS, InvalidRepositoryPointerError, } from './pointers.js';
package/package.json ADDED
@@ -0,0 +1,41 @@
1
+ {
2
+ "name": "@blue-labs/repository-contract",
3
+ "version": "2.0.0-rc.14",
4
+ "type": "module",
5
+ "sideEffects": false,
6
+ "main": "./dist/index.js",
7
+ "types": "./dist/index.d.ts",
8
+ "exports": {
9
+ ".": {
10
+ "types": "./dist/index.d.ts",
11
+ "default": "./dist/index.js"
12
+ },
13
+ "./pointers": {
14
+ "types": "./dist/pointers.d.ts",
15
+ "default": "./dist/pointers.js"
16
+ },
17
+ "./refs": {
18
+ "types": "./dist/refs.d.ts",
19
+ "default": "./dist/refs.js"
20
+ },
21
+ "./validation": {
22
+ "types": "./dist/validation.d.ts",
23
+ "default": "./dist/validation.js"
24
+ },
25
+ "./types": {
26
+ "types": "./dist/types.d.ts",
27
+ "default": "./dist/types.js"
28
+ }
29
+ },
30
+ "license": "MIT",
31
+ "publishConfig": {
32
+ "access": "public"
33
+ },
34
+ "repository": {
35
+ "type": "git",
36
+ "url": "git+https://github.com/bluecontract/blue-js.git"
37
+ },
38
+ "peerDependencies": {
39
+ "zod": "^3.0.0"
40
+ }
41
+ }
package/project.json ADDED
@@ -0,0 +1,23 @@
1
+ {
2
+ "name": "repository-contract",
3
+ "$schema": "../../node_modules/nx/schemas/project-schema.json",
4
+ "sourceRoot": "libs/repository-contract/src",
5
+ "projectType": "library",
6
+ "tags": [],
7
+ "targets": {
8
+ "build": {
9
+ "executor": "nx:run-commands",
10
+ "outputs": ["{projectRoot}/dist"],
11
+ "options": {
12
+ "cwd": "{projectRoot}",
13
+ "command": "node -e \"require('fs').rmSync('dist', { recursive: true, force: true })\" && tsc -p tsconfig.lib.json"
14
+ }
15
+ },
16
+ "tsc": {
17
+ "executor": "@webpro/nx-tsc:tsc",
18
+ "options": {
19
+ "tsConfig": ["tsconfig.lib.json", "tsconfig.spec.json"]
20
+ }
21
+ }
22
+ }
23
+ }
package/src/index.ts ADDED
@@ -0,0 +1,4 @@
1
+ export * from './types.js';
2
+ export * from './pointers.js';
3
+ export * from './refs.js';
4
+ export * from './validation.js';
@@ -0,0 +1,62 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import {
3
+ InvalidRepositoryPointerError,
4
+ parsePointer,
5
+ RESERVED_ATTRIBUTES_POINTER_SEGMENTS,
6
+ unescapePointerToken,
7
+ validateAttributesAddedPointer,
8
+ validatePointer,
9
+ } from './pointers.js';
10
+
11
+ describe('pointers', () => {
12
+ it('parses and unescapes a valid pointer', () => {
13
+ expect(parsePointer('/a/b~1c/~0d')).toEqual(['a', 'b/c', '~d']);
14
+ expect(parsePointer('')).toEqual([]);
15
+ expect(unescapePointerToken('foo~1bar~0baz')).toBe('foo/bar~baz');
16
+ });
17
+
18
+ it('rejects missing leading slash', () => {
19
+ expect(() => parsePointer('a/b')).toThrow(InvalidRepositoryPointerError);
20
+ });
21
+
22
+ it('rejects empty segments', () => {
23
+ expect(() => parsePointer('/a//b')).toThrow(InvalidRepositoryPointerError);
24
+ });
25
+
26
+ it('rejects invalid escapes', () => {
27
+ expect(() => unescapePointerToken('foo~2bar')).toThrow(
28
+ InvalidRepositoryPointerError,
29
+ );
30
+ });
31
+
32
+ it('validatePointer rethrows invalid pointers', () => {
33
+ expect(() => validatePointer('/a//b')).toThrow(
34
+ InvalidRepositoryPointerError,
35
+ );
36
+ });
37
+
38
+ it('validateAttributesAddedPointer rejects reserved segments', () => {
39
+ expect(RESERVED_ATTRIBUTES_POINTER_SEGMENTS.has('schema')).toBe(true);
40
+ expect(() => validateAttributesAddedPointer('/schema/min')).toThrow(
41
+ InvalidRepositoryPointerError,
42
+ );
43
+ });
44
+
45
+ it('validateAttributesAddedPointer allows itemType/valueType paths', () => {
46
+ expect(() =>
47
+ validateAttributesAddedPointer('/prop/itemType/x'),
48
+ ).not.toThrow(InvalidRepositoryPointerError);
49
+ expect(() =>
50
+ validateAttributesAddedPointer('/prop/valueType/x'),
51
+ ).not.toThrow(InvalidRepositoryPointerError);
52
+ });
53
+
54
+ it('validateAttributesAddedPointer rejects pointers ending in itemType/valueType', () => {
55
+ expect(() => validateAttributesAddedPointer('/prop/itemType')).toThrow(
56
+ InvalidRepositoryPointerError,
57
+ );
58
+ expect(() => validateAttributesAddedPointer('/prop/valueType')).toThrow(
59
+ InvalidRepositoryPointerError,
60
+ );
61
+ });
62
+ });
@@ -0,0 +1,102 @@
1
+ export class InvalidRepositoryPointerError extends Error {
2
+ constructor(
3
+ public readonly pointer: string,
4
+ message?: string,
5
+ ) {
6
+ super(message ?? `Invalid repository pointer: ${pointer}`);
7
+ this.name = 'InvalidRepositoryPointerError';
8
+ }
9
+ }
10
+
11
+ export const RESERVED_ATTRIBUTES_POINTER_SEGMENTS: ReadonlySet<string> =
12
+ new Set([
13
+ 'name',
14
+ 'description',
15
+ 'type',
16
+ 'keyType',
17
+ 'value',
18
+ 'items',
19
+ 'blueId',
20
+ 'blue',
21
+ 'schema',
22
+ 'mergePolicy',
23
+ 'contracts',
24
+ ]);
25
+
26
+ export function unescapePointerToken(token: string): string {
27
+ let result = '';
28
+ for (let i = 0; i < token.length; i++) {
29
+ const ch = token[i];
30
+ if (ch !== '~') {
31
+ result += ch;
32
+ continue;
33
+ }
34
+ const next = token[i + 1];
35
+ if (next === '0') {
36
+ result += '~';
37
+ i++;
38
+ } else if (next === '1') {
39
+ result += '/';
40
+ i++;
41
+ } else {
42
+ throw new InvalidRepositoryPointerError(
43
+ token,
44
+ `Invalid escape sequence in pointer token: ~${next ?? ''}`,
45
+ );
46
+ }
47
+ }
48
+ return result;
49
+ }
50
+
51
+ export function parsePointer(pointer: string): string[] {
52
+ if (pointer === '') {
53
+ return [];
54
+ }
55
+ if (!pointer.startsWith('/')) {
56
+ throw new InvalidRepositoryPointerError(
57
+ pointer,
58
+ 'Pointer must start with "/" or be empty',
59
+ );
60
+ }
61
+
62
+ const segments = pointer.split('/').slice(1);
63
+ if (segments.some((segment) => segment.length === 0)) {
64
+ throw new InvalidRepositoryPointerError(
65
+ pointer,
66
+ 'Pointer must not contain empty segments',
67
+ );
68
+ }
69
+
70
+ return segments.map(unescapePointerToken);
71
+ }
72
+
73
+ export function validatePointer(pointer: string): void {
74
+ parsePointer(pointer);
75
+ }
76
+
77
+ export function validateAttributesAddedPointer(pointer: string): void {
78
+ if (pointer === '' || !pointer.startsWith('/')) {
79
+ throw new InvalidRepositoryPointerError(
80
+ pointer,
81
+ 'Pointer must start with "/"',
82
+ );
83
+ }
84
+
85
+ const segments = parsePointer(pointer);
86
+ for (const segment of segments) {
87
+ if (RESERVED_ATTRIBUTES_POINTER_SEGMENTS.has(segment)) {
88
+ throw new InvalidRepositoryPointerError(
89
+ pointer,
90
+ `attributesAdded pointers must not reference reserved field '${segment}'`,
91
+ );
92
+ }
93
+ }
94
+
95
+ const terminal = segments.at(-1);
96
+ if (terminal === 'itemType' || terminal === 'valueType') {
97
+ throw new InvalidRepositoryPointerError(
98
+ pointer,
99
+ `attributesAdded pointers must not end with '${terminal}'`,
100
+ );
101
+ }
102
+ }
@@ -0,0 +1,76 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import {
3
+ collectTypeRefsFromContent,
4
+ validateNoCycles,
5
+ validateStableDoesNotDependOnDev,
6
+ } from './refs.js';
7
+ import type { BlueRepository } from './types.js';
8
+
9
+ describe('refs utilities', () => {
10
+ it('collects references from inline shapes', () => {
11
+ const refs = collectTypeRefsFromContent({
12
+ type: 'core/Text',
13
+ nested: {
14
+ keyType: { blueId: 'blue/key' },
15
+ child: { valueType: 'core/Integer' },
16
+ },
17
+ itemType: 'core/List',
18
+ });
19
+ expect(refs).toEqual(
20
+ new Set(['core/Text', 'blue/key', 'core/Integer', 'core/List']),
21
+ );
22
+ });
23
+
24
+ it('detects cycles across properties', () => {
25
+ const repo: BlueRepository = {
26
+ name: 'test',
27
+ repositoryVersions: [],
28
+ packages: {
29
+ pkg: {
30
+ name: 'pkg',
31
+ aliases: { 'pkg/A': 'A', 'pkg/B': 'B' },
32
+ typesMeta: {
33
+ A: { name: 'A', status: 'stable', versions: [] },
34
+ B: { name: 'B', status: 'stable', versions: [] },
35
+ },
36
+ contents: {
37
+ A: { fields: { ref: { type: 'pkg/B' } } },
38
+ B: { fields: { ref: { type: 'pkg/A' } } },
39
+ },
40
+ schemas: {},
41
+ },
42
+ },
43
+ };
44
+
45
+ expect(() => validateNoCycles(repo)).toThrowError(/cycle/i);
46
+ });
47
+
48
+ it('rejects stable depending on dev across packages', () => {
49
+ const repo: BlueRepository = {
50
+ name: 'test',
51
+ repositoryVersions: [],
52
+ packages: {
53
+ a: {
54
+ name: 'a',
55
+ aliases: { 'a/A': 'A' },
56
+ typesMeta: { A: { name: 'A', status: 'stable', versions: [] } },
57
+ contents: {
58
+ A: { type: 'b/B' },
59
+ },
60
+ schemas: {},
61
+ },
62
+ b: {
63
+ name: 'b',
64
+ aliases: { 'b/B': 'B' },
65
+ typesMeta: { B: { name: 'B', status: 'dev', versions: [] } },
66
+ contents: { B: {} },
67
+ schemas: {},
68
+ },
69
+ },
70
+ };
71
+
72
+ expect(() => validateStableDoesNotDependOnDev(repo)).toThrowError(
73
+ /dev type/i,
74
+ );
75
+ });
76
+ });
package/src/refs.ts ADDED
@@ -0,0 +1,196 @@
1
+ import type {
2
+ BlueRepository,
3
+ BlueRepositoryPackage,
4
+ BlueTypeRuntimeMeta,
5
+ TypeAlias,
6
+ TypeBlueId,
7
+ } from './types.js';
8
+
9
+ type Ref = TypeBlueId | TypeAlias;
10
+
11
+ function extractRef(value: unknown): Ref | null {
12
+ if (typeof value === 'string') {
13
+ return value;
14
+ }
15
+ if (
16
+ value &&
17
+ typeof value === 'object' &&
18
+ typeof (value as { blueId?: unknown }).blueId === 'string'
19
+ ) {
20
+ return (value as { blueId: string }).blueId;
21
+ }
22
+ return null;
23
+ }
24
+
25
+ export function collectTypeRefsFromContent(content: unknown): Set<Ref> {
26
+ const refs = new Set<Ref>();
27
+ const seen = new WeakSet<object>();
28
+
29
+ const visit = (node: unknown) => {
30
+ if (!node || typeof node !== 'object') {
31
+ return;
32
+ }
33
+ const obj = node as Record<string, unknown>;
34
+ if (seen.has(obj)) {
35
+ return;
36
+ }
37
+ seen.add(obj);
38
+
39
+ for (const key of ['type', 'itemType', 'keyType', 'valueType']) {
40
+ if (key in obj) {
41
+ const ref = extractRef(obj[key]);
42
+ if (ref) {
43
+ refs.add(ref);
44
+ }
45
+ }
46
+ }
47
+
48
+ for (const value of Object.values(obj)) {
49
+ if (value && typeof value === 'object') {
50
+ visit(value);
51
+ }
52
+ }
53
+ };
54
+
55
+ if (Array.isArray(content)) {
56
+ content.forEach(visit);
57
+ } else {
58
+ visit(content);
59
+ }
60
+
61
+ return refs;
62
+ }
63
+
64
+ function getTypesMeta(
65
+ pkg: BlueRepositoryPackage,
66
+ ): Record<TypeBlueId, BlueTypeRuntimeMeta> {
67
+ const fromTypesMeta = (
68
+ pkg as {
69
+ typesMeta?: Record<TypeBlueId, BlueTypeRuntimeMeta>;
70
+ }
71
+ ).typesMeta;
72
+ if (fromTypesMeta) {
73
+ return fromTypesMeta;
74
+ }
75
+ const fromLegacyTypeMetas = (
76
+ pkg as {
77
+ typeMetas?: Record<TypeBlueId, BlueTypeRuntimeMeta>;
78
+ }
79
+ ).typeMetas;
80
+ if (fromLegacyTypeMetas) {
81
+ return fromLegacyTypeMetas;
82
+ }
83
+ const direct = pkg.typesMeta as
84
+ | Record<TypeBlueId, BlueTypeRuntimeMeta>
85
+ | undefined;
86
+ return direct ?? {};
87
+ }
88
+
89
+ function buildResolver(packages: Record<string, BlueRepositoryPackage>): {
90
+ allTypeMetas: Map<TypeBlueId, BlueTypeRuntimeMeta>;
91
+ aliasToBlueId: Map<TypeAlias, TypeBlueId>;
92
+ } {
93
+ const aliasToBlueId = new Map<TypeAlias, TypeBlueId>();
94
+ const allTypeMetas = new Map<TypeBlueId, BlueTypeRuntimeMeta>();
95
+
96
+ Object.values(packages).forEach((pkg) => {
97
+ Object.entries(pkg.aliases || {}).forEach(([alias, blueId]) => {
98
+ aliasToBlueId.set(alias, blueId);
99
+ aliasToBlueId.set(alias.toLowerCase(), blueId);
100
+ });
101
+ Object.entries(getTypesMeta(pkg)).forEach(([blueId, meta]) => {
102
+ allTypeMetas.set(blueId, meta);
103
+ });
104
+ });
105
+
106
+ return { allTypeMetas, aliasToBlueId };
107
+ }
108
+
109
+ function resolveRef(
110
+ ref: Ref,
111
+ aliasToBlueId: Map<TypeAlias, TypeBlueId>,
112
+ allTypeMetas: Map<TypeBlueId, BlueTypeRuntimeMeta>,
113
+ ): TypeBlueId | null {
114
+ if (allTypeMetas.has(ref)) {
115
+ return ref;
116
+ }
117
+ const resolved =
118
+ aliasToBlueId.get(ref) ?? aliasToBlueId.get(ref.toLowerCase());
119
+ if (resolved) {
120
+ return resolved;
121
+ }
122
+ return null;
123
+ }
124
+
125
+ export function validateNoCycles(repository: BlueRepository): void {
126
+ const { aliasToBlueId, allTypeMetas } = buildResolver(repository.packages);
127
+ const adjacency = new Map<TypeBlueId, Set<TypeBlueId>>();
128
+
129
+ for (const pkg of Object.values(repository.packages)) {
130
+ Object.entries(pkg.contents || {}).forEach(([blueId, content]) => {
131
+ const refs = collectTypeRefsFromContent(content);
132
+ refs.forEach((ref) => {
133
+ const resolved = resolveRef(ref, aliasToBlueId, allTypeMetas);
134
+ if (resolved) {
135
+ const set = adjacency.get(blueId) ?? new Set<TypeBlueId>();
136
+ set.add(resolved);
137
+ adjacency.set(blueId, set);
138
+ }
139
+ });
140
+ });
141
+ }
142
+
143
+ const visiting = new Set<TypeBlueId>();
144
+ const visited = new Set<TypeBlueId>();
145
+
146
+ const dfs = (node: TypeBlueId, path: TypeBlueId[]) => {
147
+ if (visiting.has(node)) {
148
+ const cyclePath = [...path, node].join(' -> ');
149
+ throw new Error(
150
+ `Repository contains a type reference cycle: ${cyclePath}`,
151
+ );
152
+ }
153
+ if (visited.has(node)) {
154
+ return;
155
+ }
156
+ visiting.add(node);
157
+ const next = adjacency.get(node);
158
+ if (next) {
159
+ next.forEach((neighbor) => dfs(neighbor, [...path, neighbor]));
160
+ }
161
+ visiting.delete(node);
162
+ visited.add(node);
163
+ };
164
+
165
+ adjacency.forEach((_neighbors, node) => {
166
+ if (!visited.has(node)) {
167
+ dfs(node, [node]);
168
+ }
169
+ });
170
+ }
171
+
172
+ export function validateStableDoesNotDependOnDev(
173
+ repository: BlueRepository,
174
+ ): void {
175
+ const { aliasToBlueId, allTypeMetas } = buildResolver(repository.packages);
176
+
177
+ const resolve = (ref: Ref) => resolveRef(ref, aliasToBlueId, allTypeMetas);
178
+
179
+ for (const [pkgName, pkg] of Object.entries(repository.packages)) {
180
+ const typesMeta = getTypesMeta(pkg);
181
+ for (const [blueId, meta] of Object.entries(typesMeta)) {
182
+ if (meta.status !== 'stable') continue;
183
+ const refs = collectTypeRefsFromContent(pkg.contents[blueId]);
184
+ for (const ref of refs) {
185
+ const resolved = resolve(ref);
186
+ if (!resolved) continue;
187
+ const targetMeta = allTypeMetas.get(resolved);
188
+ if (targetMeta?.status === 'dev') {
189
+ throw new Error(
190
+ `Stable type ${pkgName}/${meta.name} depends on dev type ${resolved}`,
191
+ );
192
+ }
193
+ }
194
+ }
195
+ }
196
+ }
package/src/types.ts ADDED
@@ -0,0 +1,59 @@
1
+ import type { ZodTypeAny } from 'zod';
2
+
3
+ export type AnyZodSchema = ZodTypeAny;
4
+
5
+ export type RepoName = string;
6
+ export type RepoBlueId = string;
7
+ export type PackageName = string;
8
+ export type TypeBlueId = string;
9
+ export type TypeAlias = string;
10
+
11
+ export type BlueRepositoryStatus = 'stable' | 'dev';
12
+
13
+ export interface BlueRepositoryVersionEntry {
14
+ repositoryVersionIndex: number;
15
+ typeBlueId: TypeBlueId;
16
+ attributesAdded: string[];
17
+ }
18
+
19
+ export interface BlueTypeRuntimeMeta {
20
+ status: BlueRepositoryStatus;
21
+ name: string;
22
+ versions: readonly BlueRepositoryVersionEntry[];
23
+ }
24
+
25
+ export interface BlueRepositoryPackage<
26
+ ContentT = unknown,
27
+ SchemaT = AnyZodSchema,
28
+ > {
29
+ name: PackageName;
30
+ aliases: Record<TypeAlias, TypeBlueId>;
31
+ typesMeta: Record<TypeBlueId, BlueTypeRuntimeMeta>;
32
+ contents: Record<TypeBlueId, ContentT>;
33
+ schemas: Record<TypeBlueId, SchemaT>;
34
+ }
35
+
36
+ export interface BlueRepository<ContentT = unknown, SchemaT = AnyZodSchema> {
37
+ name: RepoName;
38
+ repositoryVersions: readonly RepoBlueId[];
39
+ packages: Record<PackageName, BlueRepositoryPackage<ContentT, SchemaT>>;
40
+ }
41
+
42
+ export type BlueTypeVersion = BlueRepositoryVersionEntry;
43
+
44
+ export interface BlueTypeMetadata<ContentT = unknown> {
45
+ status: BlueRepositoryStatus;
46
+ content: ContentT;
47
+ versions: BlueTypeVersion[];
48
+ }
49
+
50
+ export interface BluePackage<ContentT = unknown> {
51
+ name: PackageName;
52
+ types: BlueTypeMetadata<ContentT>[];
53
+ }
54
+
55
+ export interface BlueRepositoryDocument<ContentT = unknown> {
56
+ name: RepoName;
57
+ packages: BluePackage<ContentT>[];
58
+ repositoryVersions: RepoBlueId[];
59
+ }
@@ -0,0 +1,13 @@
1
+ export {
2
+ collectTypeRefsFromContent,
3
+ validateNoCycles,
4
+ validateStableDoesNotDependOnDev,
5
+ } from './refs.js';
6
+ export {
7
+ parsePointer,
8
+ unescapePointerToken,
9
+ validatePointer,
10
+ validateAttributesAddedPointer,
11
+ RESERVED_ATTRIBUTES_POINTER_SEGMENTS,
12
+ InvalidRepositoryPointerError,
13
+ } from './pointers.js';
package/tsconfig.json ADDED
@@ -0,0 +1,13 @@
1
+ {
2
+ "extends": "../../tsconfig.base.json",
3
+ "files": [],
4
+ "include": [],
5
+ "references": [
6
+ {
7
+ "path": "./tsconfig.lib.json"
8
+ },
9
+ {
10
+ "path": "./tsconfig.spec.json"
11
+ }
12
+ ]
13
+ }
@@ -0,0 +1,15 @@
1
+ {
2
+ "extends": "../../tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "rootDir": "src",
5
+ "outDir": "dist",
6
+ "tsBuildInfoFile": "dist/tsconfig.lib.tsbuildinfo",
7
+ "types": ["node"],
8
+ "emitDeclarationOnly": false
9
+ },
10
+ "include": ["src/**/*.ts"],
11
+ "exclude": [
12
+ "src/**/*.test.ts",
13
+ "src/**/*.spec.ts"
14
+ ]
15
+ }
@@ -0,0 +1,10 @@
1
+ {
2
+ "extends": "./tsconfig.lib.json",
3
+ "compilerOptions": {
4
+ "types": ["vitest/globals", "node"],
5
+ "declaration": true,
6
+ "composite": true
7
+ },
8
+ "include": ["src/**/*.ts"],
9
+ "exclude": ["src/**/*.test.ts", "src/**/*.spec.ts"]
10
+ }
@@ -0,0 +1,14 @@
1
+ import { defineConfig } from 'vitest/config';
2
+ import { fileURLToPath } from 'node:url';
3
+ import { dirname, resolve } from 'node:path';
4
+
5
+ const __filename = fileURLToPath(import.meta.url);
6
+ const __dirname = dirname(__filename);
7
+
8
+ export default defineConfig({
9
+ root: resolve(__dirname),
10
+ test: {
11
+ environment: 'node',
12
+ include: ['src/**/*.test.ts'],
13
+ },
14
+ });