@aidc-toolkit/gs1 1.0.31-beta → 1.0.33-beta

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.
Files changed (110) hide show
  1. package/README.md +290 -17
  2. package/dist/character-set.d.ts +17 -2
  3. package/dist/character-set.d.ts.map +1 -1
  4. package/dist/character-set.js +44 -11
  5. package/dist/character-set.js.map +1 -1
  6. package/dist/gcp-length-cache.d.ts +81 -0
  7. package/dist/gcp-length-cache.d.ts.map +1 -0
  8. package/dist/gcp-length-cache.js +232 -0
  9. package/dist/gcp-length-cache.js.map +1 -0
  10. package/dist/gcp-length-data.d.ts +108 -0
  11. package/dist/gcp-length-data.d.ts.map +1 -0
  12. package/dist/gcp-length-data.js +53 -0
  13. package/dist/gcp-length-data.js.map +1 -0
  14. package/dist/gcp-length.d.ts +61 -0
  15. package/dist/gcp-length.d.ts.map +1 -0
  16. package/dist/gcp-length.js +300 -0
  17. package/dist/gcp-length.js.map +1 -0
  18. package/dist/gtin-creator.d.ts +0 -17
  19. package/dist/gtin-creator.d.ts.map +1 -1
  20. package/dist/gtin-creator.js +1 -93
  21. package/dist/gtin-creator.js.map +1 -1
  22. package/dist/gtin-validator.d.ts +1 -46
  23. package/dist/gtin-validator.d.ts.map +1 -1
  24. package/dist/gtin-validator.js +31 -125
  25. package/dist/gtin-validator.js.map +1 -1
  26. package/dist/identifier-validator.d.ts +1 -4
  27. package/dist/identifier-validator.d.ts.map +1 -1
  28. package/dist/identifier-validator.js +2 -5
  29. package/dist/identifier-validator.js.map +1 -1
  30. package/dist/index.d.ts +5 -1
  31. package/dist/index.d.ts.map +1 -1
  32. package/dist/index.js +4 -0
  33. package/dist/index.js.map +1 -1
  34. package/dist/locale/en/locale-resources.d.ts +39 -35
  35. package/dist/locale/en/locale-resources.d.ts.map +1 -1
  36. package/dist/locale/en/locale-resources.js +6 -2
  37. package/dist/locale/en/locale-resources.js.map +1 -1
  38. package/dist/locale/fr/locale-resources.d.ts +39 -35
  39. package/dist/locale/fr/locale-resources.d.ts.map +1 -1
  40. package/dist/locale/fr/locale-resources.js +6 -2
  41. package/dist/locale/fr/locale-resources.js.map +1 -1
  42. package/dist/locale/i18n.d.ts +4 -4
  43. package/dist/locale/i18n.d.ts.map +1 -1
  44. package/dist/locale/i18n.js +6 -7
  45. package/dist/locale/i18n.js.map +1 -1
  46. package/dist/non-numeric-identifier-validator.js +1 -1
  47. package/dist/non-numeric-identifier-validator.js.map +1 -1
  48. package/dist/numeric-identifier-validator.d.ts.map +1 -1
  49. package/dist/numeric-identifier-validator.js +2 -2
  50. package/dist/numeric-identifier-validator.js.map +1 -1
  51. package/dist/prefix-manager.d.ts +35 -0
  52. package/dist/prefix-manager.d.ts.map +1 -1
  53. package/dist/prefix-manager.js +56 -0
  54. package/dist/prefix-manager.js.map +1 -1
  55. package/dist/prefix-validator.d.ts +5 -4
  56. package/dist/prefix-validator.d.ts.map +1 -1
  57. package/dist/prefix-validator.js +18 -22
  58. package/dist/prefix-validator.js.map +1 -1
  59. package/dist/serializable-numeric-identifier-validator.d.ts +26 -0
  60. package/dist/serializable-numeric-identifier-validator.d.ts.map +1 -1
  61. package/dist/serializable-numeric-identifier-validator.js +19 -0
  62. package/dist/serializable-numeric-identifier-validator.js.map +1 -1
  63. package/dist/variable-measure.d.ts +68 -0
  64. package/dist/variable-measure.d.ts.map +1 -0
  65. package/dist/variable-measure.js +210 -0
  66. package/dist/variable-measure.js.map +1 -0
  67. package/dist/verified-by-gs1.d.ts +22 -0
  68. package/dist/verified-by-gs1.d.ts.map +1 -0
  69. package/dist/verified-by-gs1.js +46 -0
  70. package/dist/verified-by-gs1.js.map +1 -0
  71. package/package.json +12 -11
  72. package/src/character-set.ts +55 -11
  73. package/src/gcp-length-cache.ts +271 -0
  74. package/src/gcp-length-data.ts +136 -0
  75. package/src/gcp-length.ts +380 -0
  76. package/src/gtin-creator.ts +1 -117
  77. package/src/gtin-validator.ts +42 -173
  78. package/src/identifier-validator.ts +2 -5
  79. package/src/index.ts +7 -1
  80. package/src/locale/en/locale-resources.ts +7 -3
  81. package/src/locale/fr/locale-resources.ts +7 -3
  82. package/src/locale/i18n.ts +7 -8
  83. package/src/locale/i18next.d.ts +2 -0
  84. package/src/non-numeric-identifier-validator.ts +1 -1
  85. package/src/numeric-identifier-validator.ts +3 -3
  86. package/src/prefix-manager.ts +65 -0
  87. package/src/prefix-validator.ts +19 -23
  88. package/src/serializable-numeric-identifier-validator.ts +36 -0
  89. package/src/variable-measure.ts +268 -0
  90. package/src/verified-by-gs1.ts +54 -0
  91. package/test/character-set.test.ts +46 -0
  92. package/test/creator.test.ts +5 -5
  93. package/test/data/gcpprefixformatlist-1.json +662625 -0
  94. package/test/data/gcpprefixformatlist-2.json +735431 -0
  95. package/test/gcp-length.test.ts +344 -0
  96. package/test/gtin-creator.ts +4 -4
  97. package/test/gtin-validator.test.ts +205 -113
  98. package/test/gtin-validator.ts +30 -0
  99. package/test/identifier-creator.ts +6 -6
  100. package/test/non-numeric-identifier-creator.ts +0 -8
  101. package/test/non-serializable-numeric-identifier-creator.ts +4 -54
  102. package/test/numeric-identifier-creator.ts +4 -4
  103. package/test/prefix-manager.test.ts +5 -5
  104. package/test/serializable-numeric-identifier-creator.ts +32 -19
  105. package/test/validator.test.ts +6 -6
  106. package/test/variable-measure-rcn.test.ts +63 -68
  107. package/test/verified-by-gs1.test.ts +55 -0
  108. package/tsconfig-src.json +7 -1
  109. package/tsconfig-src.tsbuildinfo +1 -0
  110. package/tsconfig-tsup.json +7 -0
@@ -0,0 +1,271 @@
1
+ import { type AppData, type AppDataStorage, Cache, omit, RemoteAppDataStorage } from "@aidc-toolkit/core";
2
+ import {
3
+ type GCPLengthAppData,
4
+ type GCPLengthData,
5
+ type GCPLengthHeader,
6
+ type GCPLengthSourceJSON,
7
+ isGCPLengthAppData,
8
+ isGCPLengthHeader
9
+ } from "./gcp-length-data.js";
10
+ import { i18nextGS1 } from "./locale/i18n.js";
11
+
12
+ /**
13
+ * GS1 Company Prefix length cache.
14
+ */
15
+ export abstract class GCPLengthCache extends Cache<GCPLengthData, GCPLengthData | GCPLengthSourceJSON> {
16
+ /**
17
+ * Storage key for full application data object.
18
+ */
19
+ static APP_DATA_STORAGE_KEY = "gcp-length";
20
+
21
+ /**
22
+ * Storage key for next check date/time.
23
+ */
24
+ static NEXT_CHECK_DATE_TIME_STORAGE_KEY = "gcp-length-next-check-date-time";
25
+
26
+ /**
27
+ * Storage key for header information (date/time and disclaimer).
28
+ */
29
+ static HEADER_STORAGE_KEY = "gcp-length-header";
30
+
31
+ /**
32
+ * Storage key for data only.
33
+ */
34
+ static DATA_STORAGE_KEY = "gcp-length-data";
35
+
36
+ /**
37
+ * Application data storage.
38
+ */
39
+ readonly #appDataStorage: AppDataStorage<boolean>;
40
+
41
+ /**
42
+ * GS1 Company Prefix application data.
43
+ */
44
+ #cacheGCPLengthAppData?: GCPLengthAppData | undefined;
45
+
46
+ /**
47
+ * Constructor.
48
+ *
49
+ * @param appDataStorage
50
+ * Application data storage.
51
+ */
52
+ constructor(appDataStorage: AppDataStorage<boolean>) {
53
+ super();
54
+
55
+ this.#appDataStorage = appDataStorage;
56
+ }
57
+
58
+ /**
59
+ * Get the application data storage.
60
+ */
61
+ get appDataStorage(): AppDataStorage<boolean> {
62
+ return this.#appDataStorage;
63
+ }
64
+
65
+ /**
66
+ * Load cached GS1 Company Prefix length application data from application data storage.
67
+ *
68
+ * @returns
69
+ * GS1 Company Prefix length application data.
70
+ */
71
+ async #loadCache(): Promise<GCPLengthAppData> {
72
+ if (this.#cacheGCPLengthAppData === undefined) {
73
+ if (!this.appDataStorage.supportsBinary) {
74
+ // Everything is stored in a single object.
75
+ const appData = await this.appDataStorage.read(GCPLengthCache.APP_DATA_STORAGE_KEY);
76
+
77
+ if (isGCPLengthAppData(appData)) {
78
+ this.#cacheGCPLengthAppData = appData;
79
+ }
80
+ } else {
81
+ const nextCheckDateTime = await this.appDataStorage.read(GCPLengthCache.NEXT_CHECK_DATE_TIME_STORAGE_KEY);
82
+ const header = await this.appDataStorage.read(GCPLengthCache.HEADER_STORAGE_KEY);
83
+ const data = await this.appDataStorage.read(GCPLengthCache.DATA_STORAGE_KEY, true);
84
+
85
+ this.#cacheGCPLengthAppData = {
86
+ nextCheckDateTime: nextCheckDateTime instanceof Date ? nextCheckDateTime : undefined,
87
+ gcpLengthData: isGCPLengthHeader(header) && data instanceof Uint8Array ?
88
+ {
89
+ ...header,
90
+ data
91
+ } :
92
+ undefined
93
+ };
94
+ }
95
+ }
96
+
97
+ // If there's nothing in the cache yet, create an object with the next check date/time set to now.
98
+ if (this.#cacheGCPLengthAppData === undefined) {
99
+ this.#cacheGCPLengthAppData = {
100
+ nextCheckDateTime: new Date(),
101
+ gcpLengthData: undefined
102
+ };
103
+ }
104
+
105
+ return this.#cacheGCPLengthAppData;
106
+ }
107
+
108
+ /**
109
+ * @inheritDoc
110
+ */
111
+ override get nextCheckDateTime(): Promise<Date | undefined> {
112
+ return this.#loadCache().then(gcpLengthAppData => gcpLengthAppData.nextCheckDateTime);
113
+ }
114
+
115
+ /**
116
+ * @inheritDoc
117
+ */
118
+ override get cacheDateTime(): Promise<Date | undefined> {
119
+ return this.#loadCache().then(gcpLengthAppData => gcpLengthAppData.gcpLengthData?.dateTime);
120
+ }
121
+
122
+ /**
123
+ * @inheritDoc
124
+ */
125
+ override get cacheData(): Promise<GCPLengthData> {
126
+ return this.#loadCache().then((gcpLengthAppData) => {
127
+ if (gcpLengthAppData.gcpLengthData === undefined) {
128
+ // Application error; no localization necessary.
129
+ throw new Error("GS1 Company Prefix length application data not loaded");
130
+ }
131
+
132
+ return gcpLengthAppData.gcpLengthData;
133
+ });
134
+ }
135
+
136
+ /**
137
+ * @inheritDoc
138
+ */
139
+ override async update(nextCheckDateTime: Date, _cacheDateTime?: Date, cacheData?: GCPLengthData): Promise<void> {
140
+ this.#cacheGCPLengthAppData = {
141
+ nextCheckDateTime,
142
+ // Retain existing data if cache data is not provided.
143
+ gcpLengthData: cacheData ?? this.#cacheGCPLengthAppData?.gcpLengthData
144
+ };
145
+
146
+ if (!this.appDataStorage.supportsBinary) {
147
+ // Everything is stored in a single object.
148
+ await this.appDataStorage.write(GCPLengthCache.APP_DATA_STORAGE_KEY, this.#cacheGCPLengthAppData);
149
+ } else {
150
+ await this.appDataStorage.write(GCPLengthCache.NEXT_CHECK_DATE_TIME_STORAGE_KEY, nextCheckDateTime);
151
+
152
+ if (cacheData !== undefined) {
153
+ await this.appDataStorage.write(GCPLengthCache.HEADER_STORAGE_KEY, omit(cacheData, "data"));
154
+ await this.appDataStorage.write(GCPLengthCache.DATA_STORAGE_KEY, cacheData.data);
155
+ }
156
+ }
157
+ }
158
+ }
159
+
160
+ /**
161
+ * GS1 Company Prefix length cache with remote source.
162
+ */
163
+ export class RemoteGCPLengthCache extends GCPLengthCache {
164
+ /**
165
+ * Default base URL pointing to AIDC Toolkit website data directory.
166
+ */
167
+ static DEFAULT_BASE_URL = "https://aidc-toolkit.com/data";
168
+
169
+ /**
170
+ * Remote application data storage.
171
+ */
172
+ readonly #remoteAppDataStorage: RemoteAppDataStorage;
173
+
174
+ /**
175
+ * Remote GS1 Company Prefix header data.
176
+ */
177
+ #remoteGCPLengthHeader?: GCPLengthHeader | undefined;
178
+
179
+ /**
180
+ * Constructor.
181
+ *
182
+ * @param appDataStorage
183
+ * Application data storage.
184
+ *
185
+ * @param baseURL
186
+ * Base URL. The URL must not end with a slash and must host the `gcp-length-header.json` and `gcp-length-data.bin`
187
+ * files.
188
+ */
189
+ constructor(appDataStorage: AppDataStorage<boolean>, baseURL: string = RemoteGCPLengthCache.DEFAULT_BASE_URL) {
190
+ super(appDataStorage);
191
+
192
+ this.#remoteAppDataStorage = new RemoteAppDataStorage(baseURL);
193
+ }
194
+
195
+ /**
196
+ * Get remote application data. If an exception is thrown while retrieving or processing the data, retrying is
197
+ * delayed for ten minutes to prevent repeated network calls.
198
+ *
199
+ * @param pathKey
200
+ * Key relative to base URL.
201
+ *
202
+ * @param asBinary
203
+ * True if binary data is requested, false if string data is requested.
204
+ *
205
+ * @param callback
206
+ * Callback function to process the application data.
207
+ *
208
+ * @returns
209
+ * Application data.
210
+ */
211
+ async #getRemoteAppData<T>(pathKey: string, asBinary: boolean, callback: (appData: AppData) => T): Promise<T> {
212
+ return this.#remoteAppDataStorage.read(pathKey, asBinary).then((appData) => {
213
+ if (appData === undefined) {
214
+ throw new RangeError(i18nextGS1.t("Prefix.gs1CompanyPrefixLengthDataFileNotFound", {
215
+ key: pathKey
216
+ }));
217
+ }
218
+
219
+ return callback(appData);
220
+ }).catch(async (e: unknown) => {
221
+ // Try again in ten minutes.
222
+ const nowPlus10Minutes = new Date();
223
+ nowPlus10Minutes.setMinutes(nowPlus10Minutes.getMinutes() + 10);
224
+
225
+ await this.update(nowPlus10Minutes);
226
+
227
+ throw e;
228
+ });
229
+ }
230
+
231
+ /**
232
+ * @inheritDoc
233
+ */
234
+ override get sourceDateTime(): Promise<Date> {
235
+ // Header is fetched on each call.
236
+ return this.#getRemoteAppData(RemoteGCPLengthCache.HEADER_STORAGE_KEY, false, (appData) => {
237
+ if (!isGCPLengthHeader(appData)) {
238
+ // Application error; no localization necessary.
239
+ throw new Error("Invalid GS1 Company Prefix length header");
240
+ }
241
+
242
+ this.#remoteGCPLengthHeader = appData;
243
+
244
+ return this.#remoteGCPLengthHeader.dateTime;
245
+ });
246
+ }
247
+
248
+ /**
249
+ * @inheritDoc
250
+ */
251
+ override get sourceData(): Promise<GCPLengthData> {
252
+ return this.#getRemoteAppData(RemoteGCPLengthCache.DATA_STORAGE_KEY, true, (appData) => {
253
+ const gcpLengthHeader = this.#remoteGCPLengthHeader;
254
+
255
+ if (gcpLengthHeader === undefined) {
256
+ // Application error; no localization necessary.
257
+ throw new Error("GS1 Company Prefix length header not loaded");
258
+ }
259
+
260
+ if (!(appData instanceof Uint8Array)) {
261
+ // Application error; no localization necessary.
262
+ throw new Error("Invalid GS1 Company Prefix length data");
263
+ }
264
+
265
+ return {
266
+ ...gcpLengthHeader,
267
+ data: appData
268
+ };
269
+ });
270
+ }
271
+ }
@@ -0,0 +1,136 @@
1
+ import type { AppData } from "@aidc-toolkit/core";
2
+
3
+ /**
4
+ * GS1 Company Prefix length header.
5
+ */
6
+ export interface GCPLengthHeader {
7
+ /**
8
+ * Date/time the data was last updated.
9
+ */
10
+ readonly dateTime: Date;
11
+
12
+ /**
13
+ * Disclaimer.
14
+ */
15
+ readonly disclaimer: string;
16
+ }
17
+
18
+ /**
19
+ * Determine if application data object is GS1 Company Prefix length header.
20
+ *
21
+ * @param appData
22
+ * Application data object.
23
+ *
24
+ * @returns
25
+ * True if application data object is GS1 Company Prefix length header.
26
+ */
27
+ export function isGCPLengthHeader(appData: AppData | undefined): appData is GCPLengthHeader {
28
+ // Property type check is necessary to guard against data corruption or changes in format.
29
+ return typeof appData === "object" && "dateTime" in appData && appData.dateTime instanceof Date && "disclaimer" in appData && typeof appData.disclaimer === "string";
30
+ }
31
+
32
+ /**
33
+ * GS1 Company Prefix length data.
34
+ */
35
+ export interface GCPLengthData extends GCPLengthHeader {
36
+ /**
37
+ * Tree data in binary form.
38
+ */
39
+ readonly data: Uint8Array;
40
+ }
41
+
42
+ /**
43
+ * Determine if application data object is GS1 Company Prefix length data.
44
+ *
45
+ * @param appData
46
+ * Application data object.
47
+ *
48
+ * @returns
49
+ * True if application data object is GS1 Company Prefix length data.
50
+ */
51
+ export function isGCPLengthData(appData: AppData | undefined): appData is GCPLengthData {
52
+ // Property type check is necessary to guard against data corruption or changes in format.
53
+ return isGCPLengthHeader(appData) && "data" in appData && appData.data instanceof Uint8Array;
54
+ }
55
+
56
+ /**
57
+ * GS1 Company Prefix length application data.
58
+ */
59
+ export interface GCPLengthAppData {
60
+ /**
61
+ * Next check date/time.
62
+ */
63
+ nextCheckDateTime: Date | undefined;
64
+
65
+ /**
66
+ * GS1 Company Prefix length data.
67
+ */
68
+ gcpLengthData: GCPLengthData | undefined;
69
+ }
70
+
71
+ /**
72
+ * Determine if application data object is GS1 Company Prefix length application data.
73
+ *
74
+ * @param appData
75
+ * Application data object.
76
+ *
77
+ * @returns
78
+ * True if application data object is GS1 Company Prefix length application data.
79
+ */
80
+ export function isGCPLengthAppData(appData: AppData | undefined): appData is GCPLengthAppData {
81
+ // Property type check is necessary to guard against data corruption or changes in format.
82
+ return typeof appData === "object" && "nextCheckDateTime" in appData && appData.nextCheckDateTime instanceof Date && (
83
+ !("gcpLengthData" in appData) || appData.gcpLengthData === undefined || (
84
+ typeof appData.gcpLengthData === "object" && appData.gcpLengthData !== null && isGCPLengthData(appData.gcpLengthData)
85
+ )
86
+ );
87
+ }
88
+
89
+ /**
90
+ * GS1 Company Prefix length JSON source file format.
91
+ */
92
+ export interface GCPLengthSourceJSON {
93
+ /**
94
+ * Disclaimer.
95
+ */
96
+ _disclaimer: string[];
97
+
98
+ /**
99
+ * Format list.
100
+ */
101
+ GCPPrefixFormatList: {
102
+ /**
103
+ * ISO data/time the table was last updated.
104
+ */
105
+ date: string;
106
+
107
+ /**
108
+ * Entries.
109
+ */
110
+ entry: Array<{
111
+ /**
112
+ * Identification key prefix start.
113
+ */
114
+ prefix: string;
115
+
116
+ /**
117
+ * Length of GS1 Company Prefix.
118
+ */
119
+ gcpLength: number;
120
+ }>;
121
+ };
122
+ }
123
+
124
+ /**
125
+ * Determine if application data object is GS1 Company Prefix length source JSON.
126
+ *
127
+ * @param appData
128
+ * Application data object.
129
+ *
130
+ * @returns
131
+ * True if application data object is GS1 Company Prefix length source JSON.
132
+ */
133
+ export function isGCPLengthSourceJSON(appData: AppData | undefined): appData is GCPLengthSourceJSON {
134
+ // Checking is minimal as this is generally used in a controlled environment (e.g., updating remote storage).
135
+ return typeof appData === "object" && "_disclaimer" in appData && "GCPPrefixFormatList" in appData;
136
+ }