@loaders.gl/zip 4.1.0-alpha.1 → 4.1.0-alpha.11

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 (68) hide show
  1. package/dist/dist.dev.js +875 -64
  2. package/dist/filesystems/zip-filesystem.d.ts.map +1 -1
  3. package/dist/filesystems/zip-filesystem.js.map +1 -1
  4. package/dist/hash-file-utility.d.ts +19 -0
  5. package/dist/hash-file-utility.d.ts.map +1 -1
  6. package/dist/hash-file-utility.js +30 -0
  7. package/dist/hash-file-utility.js.map +1 -1
  8. package/dist/index.cjs +542 -56
  9. package/dist/index.d.ts +3 -2
  10. package/dist/index.d.ts.map +1 -1
  11. package/dist/index.js +3 -2
  12. package/dist/index.js.map +1 -1
  13. package/dist/lib/tar/header.d.ts.map +1 -1
  14. package/dist/lib/tar/header.js.map +1 -1
  15. package/dist/lib/tar/tar.d.ts.map +1 -1
  16. package/dist/lib/tar/tar.js.map +1 -1
  17. package/dist/lib/tar/types.d.ts.map +1 -1
  18. package/dist/lib/tar/types.js.map +1 -1
  19. package/dist/lib/tar/utils.d.ts.map +1 -1
  20. package/dist/lib/tar/utils.js.map +1 -1
  21. package/dist/parse-zip/cd-file-header.d.ts +1 -1
  22. package/dist/parse-zip/cd-file-header.d.ts.map +1 -1
  23. package/dist/parse-zip/cd-file-header.js +4 -4
  24. package/dist/parse-zip/cd-file-header.js.map +1 -1
  25. package/dist/parse-zip/end-of-central-directory.d.ts +35 -0
  26. package/dist/parse-zip/end-of-central-directory.d.ts.map +1 -1
  27. package/dist/parse-zip/end-of-central-directory.js +161 -9
  28. package/dist/parse-zip/end-of-central-directory.js.map +1 -1
  29. package/dist/parse-zip/local-file-header.d.ts +16 -0
  30. package/dist/parse-zip/local-file-header.d.ts.map +1 -1
  31. package/dist/parse-zip/local-file-header.js +73 -1
  32. package/dist/parse-zip/local-file-header.js.map +1 -1
  33. package/dist/parse-zip/search-from-the-end.d.ts.map +1 -1
  34. package/dist/parse-zip/search-from-the-end.js.map +1 -1
  35. package/dist/parse-zip/zip-composition.d.ts +38 -0
  36. package/dist/parse-zip/zip-composition.d.ts.map +1 -0
  37. package/dist/parse-zip/zip-composition.js +115 -0
  38. package/dist/parse-zip/zip-composition.js.map +1 -0
  39. package/dist/parse-zip/zip64-info-generation.d.ts +5 -8
  40. package/dist/parse-zip/zip64-info-generation.d.ts.map +1 -1
  41. package/dist/parse-zip/zip64-info-generation.js +6 -3
  42. package/dist/parse-zip/zip64-info-generation.js.map +1 -1
  43. package/dist/tar-builder.d.ts.map +1 -1
  44. package/dist/tar-builder.js.map +1 -1
  45. package/dist/zip-loader.d.ts.map +1 -1
  46. package/dist/zip-loader.js +1 -1
  47. package/dist/zip-loader.js.map +1 -1
  48. package/dist/zip-writer.d.ts +2 -2
  49. package/dist/zip-writer.d.ts.map +1 -1
  50. package/dist/zip-writer.js +22 -7
  51. package/dist/zip-writer.js.map +1 -1
  52. package/package.json +7 -7
  53. package/src/filesystems/zip-filesystem.ts +2 -1
  54. package/src/hash-file-utility.ts +84 -3
  55. package/src/index.ts +6 -3
  56. package/src/lib/tar/header.ts +2 -1
  57. package/src/lib/tar/tar.ts +2 -1
  58. package/src/lib/tar/types.ts +2 -1
  59. package/src/lib/tar/utils.ts +2 -1
  60. package/src/parse-zip/cd-file-header.ts +8 -6
  61. package/src/parse-zip/end-of-central-directory.ts +338 -10
  62. package/src/parse-zip/local-file-header.ts +128 -2
  63. package/src/parse-zip/search-from-the-end.ts +2 -1
  64. package/src/parse-zip/zip-composition.ts +235 -0
  65. package/src/parse-zip/zip64-info-generation.ts +21 -5
  66. package/src/tar-builder.ts +2 -1
  67. package/src/zip-loader.ts +2 -1
  68. package/src/zip-writer.ts +24 -10
@@ -1,8 +1,10 @@
1
- // loaders.gl, MIT license
1
+ // loaders.gl
2
+ // SPDX-License-Identifier: MIT
2
3
  // Copyright (c) vis.gl contributors
3
4
 
4
- import {FileProvider, compareArrayBuffers} from '@loaders.gl/loader-utils';
5
+ import {FileProvider, compareArrayBuffers, concatenateArrayBuffers} from '@loaders.gl/loader-utils';
5
6
  import {ZipSignature, searchFromTheEnd} from './search-from-the-end';
7
+ import {setFieldToNumber} from './zip64-info-generation';
6
8
 
7
9
  /**
8
10
  * End of central directory info
@@ -13,6 +15,29 @@ export type ZipEoCDRecord = {
13
15
  cdStartOffset: bigint;
14
16
  /** Relative offset of local file header */
15
17
  cdRecordsNumber: bigint;
18
+ offsets: ZipEoCDRecordOffsets;
19
+ };
20
+
21
+ /**
22
+ * End of central directory offsets
23
+ * according to https://en.wikipedia.org/wiki/ZIP_(file_format)
24
+ */
25
+ export type ZipEoCDRecordOffsets = {
26
+ zipEoCDOffset: bigint;
27
+
28
+ zip64EoCDOffset?: bigint;
29
+ zip64EoCDLocatorOffset?: bigint;
30
+ };
31
+
32
+ /**
33
+ * Data to generate End of central directory record
34
+ * according to https://en.wikipedia.org/wiki/ZIP_(file_format)
35
+ */
36
+ export type ZipEoCDGenerationOptions = {
37
+ recordsNumber: number;
38
+ cdSize: number;
39
+ cdOffset: bigint;
40
+ eoCDStart: bigint;
16
41
  };
17
42
 
18
43
  const eoCDSignature: ZipSignature = new Uint8Array([0x50, 0x4b, 0x05, 0x06]);
@@ -21,10 +46,16 @@ const zip64EoCDSignature = new Uint8Array([0x50, 0x4b, 0x06, 0x06]);
21
46
 
22
47
  // offsets accroding to https://en.wikipedia.org/wiki/ZIP_(file_format)
23
48
  const CD_RECORDS_NUMBER_OFFSET = 8n;
49
+ const CD_RECORDS_NUMBER_ON_DISC_OFFSET = 10n;
50
+ const CD_CD_BYTE_SIZE_OFFSET = 12n;
24
51
  const CD_START_OFFSET_OFFSET = 16n;
52
+ const CD_COMMENT_OFFSET = 22n;
25
53
  const ZIP64_EOCD_START_OFFSET_OFFSET = 8n;
26
54
  const ZIP64_CD_RECORDS_NUMBER_OFFSET = 24n;
55
+ const ZIP64_CD_RECORDS_NUMBER_ON_DISC_OFFSET = 32n;
56
+ const ZIP64_CD_CD_BYTE_SIZE_OFFSET = 40n;
27
57
  const ZIP64_CD_START_OFFSET_OFFSET = 48n;
58
+ const ZIP64_COMMENT_OFFSET = 56n;
28
59
 
29
60
  /**
30
61
  * Parses end of central directory record of zip file
@@ -37,14 +68,12 @@ export const parseEoCDRecord = async (file: FileProvider): Promise<ZipEoCDRecord
37
68
  let cdRecordsNumber = BigInt(await file.getUint16(zipEoCDOffset + CD_RECORDS_NUMBER_OFFSET));
38
69
  let cdStartOffset = BigInt(await file.getUint32(zipEoCDOffset + CD_START_OFFSET_OFFSET));
39
70
 
40
- if (cdStartOffset === BigInt(0xffffffff) || cdRecordsNumber === BigInt(0xffffffff)) {
41
- const zip64EoCDLocatorOffset = zipEoCDOffset - 20n;
71
+ let zip64EoCDLocatorOffset = zipEoCDOffset - 20n;
72
+ let zip64EoCDOffset = 0n;
42
73
 
43
- const magicBytes = await file.slice(zip64EoCDLocatorOffset, zip64EoCDLocatorOffset + 4n);
44
- if (!compareArrayBuffers(magicBytes, zip64EoCDLocatorSignature)) {
45
- throw new Error('zip64 EoCD locator not found');
46
- }
47
- const zip64EoCDOffset = await file.getBigUint64(
74
+ const magicBytes = await file.slice(zip64EoCDLocatorOffset, zip64EoCDLocatorOffset + 4n);
75
+ if (compareArrayBuffers(magicBytes, zip64EoCDLocatorSignature)) {
76
+ zip64EoCDOffset = await file.getBigUint64(
48
77
  zip64EoCDLocatorOffset + ZIP64_EOCD_START_OFFSET_OFFSET
49
78
  );
50
79
 
@@ -55,10 +84,309 @@ export const parseEoCDRecord = async (file: FileProvider): Promise<ZipEoCDRecord
55
84
 
56
85
  cdRecordsNumber = await file.getBigUint64(zip64EoCDOffset + ZIP64_CD_RECORDS_NUMBER_OFFSET);
57
86
  cdStartOffset = await file.getBigUint64(zip64EoCDOffset + ZIP64_CD_START_OFFSET_OFFSET);
87
+ } else {
88
+ zip64EoCDLocatorOffset = 0n;
58
89
  }
59
90
 
60
91
  return {
61
92
  cdRecordsNumber,
62
- cdStartOffset
93
+ cdStartOffset,
94
+ offsets: {
95
+ zip64EoCDOffset,
96
+ zip64EoCDLocatorOffset,
97
+ zipEoCDOffset
98
+ }
63
99
  };
64
100
  };
101
+
102
+ /**
103
+ * updates EoCD record to add more files to the archieve
104
+ * @param eocdBody buffer containing header
105
+ * @param oldEoCDOffsets info read from EoCD record befor updating
106
+ * @param newCDStartOffset CD start offset to be updated
107
+ * @param eocdStartOffset EoCD start offset to be updated
108
+ * @returns new EoCD header
109
+ */
110
+ export function updateEoCD(
111
+ eocdBody: ArrayBuffer,
112
+ oldEoCDOffsets: ZipEoCDRecordOffsets,
113
+ newCDStartOffset: bigint,
114
+ eocdStartOffset: bigint,
115
+ newCDRecordsNumber: bigint
116
+ ): Uint8Array {
117
+ const eocd = new DataView(eocdBody);
118
+
119
+ const classicEoCDOffset = oldEoCDOffsets.zip64EoCDOffset
120
+ ? oldEoCDOffsets.zipEoCDOffset - oldEoCDOffsets.zip64EoCDOffset
121
+ : 0n;
122
+
123
+ // updating classic EoCD record with new CD records number in general and on disc
124
+ if (Number(newCDRecordsNumber) <= 0xffff) {
125
+ setFieldToNumber(eocd, 2, classicEoCDOffset + CD_RECORDS_NUMBER_OFFSET, newCDRecordsNumber);
126
+ setFieldToNumber(
127
+ eocd,
128
+ 2,
129
+ classicEoCDOffset + CD_RECORDS_NUMBER_ON_DISC_OFFSET,
130
+ newCDRecordsNumber
131
+ );
132
+ }
133
+
134
+ // updating zip64 EoCD record with new size of CD
135
+ if (eocdStartOffset - newCDStartOffset <= 0xffffffff) {
136
+ setFieldToNumber(
137
+ eocd,
138
+ 4,
139
+ classicEoCDOffset + CD_CD_BYTE_SIZE_OFFSET,
140
+ eocdStartOffset - newCDStartOffset
141
+ );
142
+ }
143
+
144
+ // updating classic EoCD record with new CD start offset
145
+ if (newCDStartOffset < 0xffffffff) {
146
+ setFieldToNumber(eocd, 4, classicEoCDOffset + CD_START_OFFSET_OFFSET, newCDStartOffset);
147
+ }
148
+
149
+ // updating zip64 EoCD locator and record with new EoCD record start offset and cd records number
150
+ if (oldEoCDOffsets.zip64EoCDLocatorOffset && oldEoCDOffsets.zip64EoCDOffset) {
151
+ // updating zip64 EoCD locator with new EoCD record start offset
152
+ const locatorOffset = oldEoCDOffsets.zip64EoCDLocatorOffset - oldEoCDOffsets.zip64EoCDOffset;
153
+ setFieldToNumber(eocd, 8, locatorOffset + ZIP64_EOCD_START_OFFSET_OFFSET, eocdStartOffset);
154
+
155
+ // updating zip64 EoCD record with new cd start offset
156
+ setFieldToNumber(eocd, 8, ZIP64_CD_START_OFFSET_OFFSET, newCDStartOffset);
157
+
158
+ // updating zip64 EoCD record with new cd records number
159
+ setFieldToNumber(eocd, 8, ZIP64_CD_RECORDS_NUMBER_OFFSET, newCDRecordsNumber);
160
+ setFieldToNumber(eocd, 8, ZIP64_CD_RECORDS_NUMBER_ON_DISC_OFFSET, newCDRecordsNumber);
161
+
162
+ // updating zip64 EoCD record with new size of CD
163
+ setFieldToNumber(eocd, 8, ZIP64_CD_CD_BYTE_SIZE_OFFSET, eocdStartOffset - newCDStartOffset);
164
+ }
165
+
166
+ return new Uint8Array(eocd.buffer);
167
+ }
168
+
169
+ /**
170
+ * generates EoCD record
171
+ * @param options data to generate EoCD record
172
+ * @returns ArrayBuffer with EoCD record
173
+ */
174
+ export function generateEoCD(options: ZipEoCDGenerationOptions): ArrayBuffer {
175
+ const header = new DataView(new ArrayBuffer(Number(CD_COMMENT_OFFSET)));
176
+
177
+ for (const field of EOCD_FIELDS) {
178
+ setFieldToNumber(
179
+ header,
180
+ field.size,
181
+ field.offset,
182
+ options[field.name ?? ''] ?? field.default ?? 0
183
+ );
184
+ }
185
+ const locator = generateZip64InfoLocator(options);
186
+
187
+ const zip64Record = generateZip64Info(options);
188
+
189
+ return concatenateArrayBuffers(zip64Record, locator, header.buffer);
190
+ }
191
+
192
+ /** standart EoCD fields */
193
+ const EOCD_FIELDS = [
194
+ // End of central directory signature = 0x06054b50
195
+ {
196
+ offset: 0,
197
+ size: 4,
198
+ default: new DataView(eoCDSignature.buffer).getUint32(0, true)
199
+ },
200
+
201
+ // Number of this disk (or 0xffff for ZIP64)
202
+ {
203
+ offset: 4,
204
+ size: 2,
205
+ default: 0
206
+ },
207
+
208
+ // Disk where central directory starts (or 0xffff for ZIP64)
209
+ {
210
+ offset: 6,
211
+ size: 2,
212
+ default: 0
213
+ },
214
+
215
+ // Number of central directory records on this disk (or 0xffff for ZIP64)
216
+ {
217
+ offset: 8,
218
+ size: 2,
219
+ name: 'recordsNumber'
220
+ },
221
+
222
+ // Total number of central directory records (or 0xffff for ZIP64)
223
+ {
224
+ offset: 10,
225
+ size: 2,
226
+ name: 'recordsNumber'
227
+ },
228
+
229
+ // Size of central directory (bytes) (or 0xffffffff for ZIP64)
230
+ {
231
+ offset: 12,
232
+ size: 4,
233
+ name: 'cdSize'
234
+ },
235
+
236
+ // Offset of start of central directory, relative to start of archive (or 0xffffffff for ZIP64)
237
+ {
238
+ offset: 16,
239
+ size: 4,
240
+ name: 'cdOffset'
241
+ },
242
+
243
+ // Comment length (n)
244
+ {
245
+ offset: 20,
246
+ size: 2,
247
+ default: 0
248
+ }
249
+ ];
250
+
251
+ /**
252
+ * generates eocd zip64 record
253
+ * @param options data to generate eocd zip64 record
254
+ * @returns buffer with eocd zip64 record
255
+ */
256
+ function generateZip64Info(options: ZipEoCDGenerationOptions): ArrayBuffer {
257
+ const record = new DataView(new ArrayBuffer(Number(ZIP64_COMMENT_OFFSET)));
258
+ for (const field of ZIP64_EOCD_FIELDS) {
259
+ setFieldToNumber(
260
+ record,
261
+ field.size,
262
+ field.offset,
263
+ options[field.name ?? ''] ?? field.default ?? 0
264
+ );
265
+ }
266
+
267
+ return record.buffer;
268
+ }
269
+
270
+ /**
271
+ * generates eocd zip64 record locator
272
+ * @param options data to generate eocd zip64 record
273
+ * @returns buffer with eocd zip64 record
274
+ */
275
+ function generateZip64InfoLocator(options: ZipEoCDGenerationOptions): ArrayBuffer {
276
+ const locator = new DataView(new ArrayBuffer(Number(20)));
277
+
278
+ for (const field of ZIP64_EOCD_LOCATOR_FIELDS) {
279
+ setFieldToNumber(
280
+ locator,
281
+ field.size,
282
+ field.offset,
283
+ options[field.name ?? ''] ?? field.default ?? 0
284
+ );
285
+ }
286
+
287
+ return locator.buffer;
288
+ }
289
+
290
+ /** zip64 EoCD record locater fields */
291
+ const ZIP64_EOCD_LOCATOR_FIELDS = [
292
+ // zip64 end of central dir locator signature
293
+ {
294
+ offset: 0,
295
+ size: 4,
296
+ default: new DataView(zip64EoCDLocatorSignature.buffer).getUint32(0, true)
297
+ },
298
+
299
+ // number of the disk with the start of the zip64 end of
300
+ {
301
+ offset: 4,
302
+ size: 4,
303
+ default: 0
304
+ },
305
+
306
+ // start of the zip64 end of central directory
307
+ {
308
+ offset: 8,
309
+ size: 8,
310
+ name: 'eoCDStart'
311
+ },
312
+
313
+ // total number of disks
314
+ {
315
+ offset: 16,
316
+ size: 4,
317
+ default: 1
318
+ }
319
+ ];
320
+
321
+ /** zip64 EoCD recodrd fields */
322
+ const ZIP64_EOCD_FIELDS = [
323
+ // End of central directory signature = 0x06064b50
324
+ {
325
+ offset: 0,
326
+ size: 4,
327
+ default: new DataView(zip64EoCDSignature.buffer).getUint32(0, true)
328
+ },
329
+
330
+ // Size of the EOCD64 minus 12
331
+ {
332
+ offset: 4,
333
+ size: 8,
334
+ default: 44
335
+ },
336
+
337
+ // Version made by
338
+ {
339
+ offset: 12,
340
+ size: 2,
341
+ default: 45
342
+ },
343
+
344
+ // Version needed to extract (minimum)
345
+ {
346
+ offset: 14,
347
+ size: 2,
348
+ default: 45
349
+ },
350
+
351
+ // Number of this disk
352
+ {
353
+ offset: 16,
354
+ size: 4,
355
+ default: 0
356
+ },
357
+
358
+ // Disk where central directory starts
359
+ {
360
+ offset: 20,
361
+ size: 4,
362
+ default: 0
363
+ },
364
+
365
+ // Number of central directory records on this disk
366
+ {
367
+ offset: 24,
368
+ size: 8,
369
+ name: 'recordsNumber'
370
+ },
371
+
372
+ // Total number of central directory records
373
+ {
374
+ offset: 32,
375
+ size: 8,
376
+ name: 'recordsNumber'
377
+ },
378
+
379
+ // Size of central directory (bytes)
380
+ {
381
+ offset: 40,
382
+ size: 8,
383
+ name: 'cdSize'
384
+ },
385
+
386
+ // Offset of start of central directory, relative to start of archive
387
+ {
388
+ offset: 48,
389
+ size: 8,
390
+ name: 'cdOffset'
391
+ }
392
+ ];
@@ -1,8 +1,10 @@
1
- // loaders.gl, MIT license
1
+ // loaders.gl
2
+ // SPDX-License-Identifier: MIT
2
3
  // Copyright (c) vis.gl contributors
3
4
 
4
- import {FileProvider, compareArrayBuffers} from '@loaders.gl/loader-utils';
5
+ import {FileProvider, compareArrayBuffers, concatenateArrayBuffers} from '@loaders.gl/loader-utils';
5
6
  import {ZipSignature} from './search-from-the-end';
7
+ import {createZip64Info, setFieldToNumber} from './zip64-info-generation';
6
8
 
7
9
  /**
8
10
  * zip local file header info
@@ -94,3 +96,127 @@ export const parseZipLocalFileHeader = async (
94
96
  compressionMethod
95
97
  };
96
98
  };
99
+
100
+ /** info that can be placed into cd header */
101
+ type GenerateLocalOptions = {
102
+ /** CRC-32 of uncompressed data */
103
+ crc32: number;
104
+ /** File name */
105
+ fileName: string;
106
+ /** File size */
107
+ length: number;
108
+ };
109
+
110
+ /**
111
+ * generates local header for the file
112
+ * @param options info that can be placed into local header
113
+ * @returns buffer with header
114
+ */
115
+ export function generateLocalHeader(options: GenerateLocalOptions): ArrayBuffer {
116
+ const optionsToUse = {
117
+ ...options,
118
+ extraLength: 0,
119
+ fnlength: options.fileName.length
120
+ };
121
+
122
+ let zip64header: ArrayBuffer = new ArrayBuffer(0);
123
+
124
+ const optionsToZip64: any = {};
125
+ if (optionsToUse.length >= 0xffffffff) {
126
+ optionsToZip64.size = optionsToUse.length;
127
+ optionsToUse.length = 0xffffffff;
128
+ }
129
+
130
+ if (Object.keys(optionsToZip64).length) {
131
+ zip64header = createZip64Info(optionsToZip64);
132
+ optionsToUse.extraLength = zip64header.byteLength;
133
+ }
134
+
135
+ // base length without file name and extra info is static
136
+ const header = new DataView(new ArrayBuffer(Number(FILE_NAME_OFFSET)));
137
+
138
+ for (const field of ZIP_HEADER_FIELDS) {
139
+ setFieldToNumber(
140
+ header,
141
+ field.size,
142
+ field.offset,
143
+ optionsToUse[field.name ?? ''] ?? field.default ?? 0
144
+ );
145
+ }
146
+
147
+ const encodedName = new TextEncoder().encode(optionsToUse.fileName);
148
+
149
+ const resHeader = concatenateArrayBuffers(header.buffer, encodedName, zip64header);
150
+
151
+ return resHeader;
152
+ }
153
+
154
+ const ZIP_HEADER_FIELDS = [
155
+ // Local file header signature = 0x04034b50
156
+ {
157
+ offset: 0,
158
+ size: 4,
159
+ default: new DataView(signature.buffer).getUint32(0, true)
160
+ },
161
+ // Version needed to extract (minimum)
162
+ {
163
+ offset: 4,
164
+ size: 2,
165
+ default: 45
166
+ },
167
+ // General purpose bit flag
168
+ {
169
+ offset: 6,
170
+ size: 2,
171
+ default: 0
172
+ },
173
+ // Compression method
174
+ {
175
+ offset: 8,
176
+ size: 2,
177
+ default: 0
178
+ },
179
+ // File last modification time
180
+ {
181
+ offset: 10,
182
+ size: 2,
183
+ default: 0
184
+ },
185
+ // File last modification date
186
+ {
187
+ offset: 12,
188
+ size: 2,
189
+ default: 0
190
+ },
191
+ // CRC-32 of uncompressed data
192
+ {
193
+ offset: 14,
194
+ size: 4,
195
+ name: 'crc32'
196
+ },
197
+ // Compressed size (or 0xffffffff for ZIP64)
198
+ {
199
+ offset: 18,
200
+ size: 4,
201
+ name: 'length'
202
+ },
203
+ // Uncompressed size (or 0xffffffff for ZIP64)
204
+ {
205
+ offset: 22,
206
+ size: 4,
207
+ name: 'length'
208
+ },
209
+ // File name length (n)
210
+ {
211
+ offset: 26,
212
+ size: 2,
213
+ name: 'fnlength'
214
+ },
215
+ // Extra field length (m)
216
+ {
217
+ offset: 28,
218
+ size: 2,
219
+ default: 0,
220
+ name: 'extraLength'
221
+ }
222
+ ];
@@ -1,4 +1,5 @@
1
- // loaders.gl, MIT license
1
+ // loaders.gl
2
+ // SPDX-License-Identifier: MIT
2
3
  // Copyright (c) vis.gl contributors
3
4
 
4
5
  import {FileProvider} from '@loaders.gl/loader-utils';