UNPKG

igir

Version:

🕹 A zero-setup ROM collection manager that sorts, filters, extracts or archives, patches, and reports on collections of any size on any OS.

238 lines (237 loc) • 9.44 kB
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; var __metadata = (this && this.__metadata) || function (k, v) { if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v); }; import { Memoize } from 'typescript-memoize'; import ArrayPoly from '../polyfill/arrayPoly.js'; /** * A lookup table of {@link File}s based on their checksums. */ export default class IndexedFiles { crc32; md5; sha1; sha256; constructor(crc32, md5, sha1, sha256) { this.crc32 = crc32; this.md5 = md5; this.sha1 = sha1; this.sha256 = sha256; } /** * Generate a {@link IndexedFiles} based on a set of {@link File}s. */ static fromFiles(files) { const crc32RawMap = new Map(); const crc32WithoutHeaderMap = new Map(); const crc32PaddedMap = new Map(); const md5RawMap = new Map(); const md5WithoutHeaderMap = new Map(); const md5PaddedMap = new Map(); const sha1RawMap = new Map(); const sha1WithoutHeaderMap = new Map(); const sha1PaddedMap = new Map(); const sha256RawMap = new Map(); const sha256WithoutHeaderMap = new Map(); const sha256PaddedMap = new Map(); // Build the maps files.forEach((file) => { const crc32WithSize = `${file.getCrc32()}|${file.getSize()}`; if (crc32RawMap.has(crc32WithSize)) { crc32RawMap.get(crc32WithSize)?.unshift(file); } else { crc32RawMap.set(crc32WithSize, [file]); } const md5 = file.getMd5(); if (md5) { if (md5RawMap.has(md5)) { md5RawMap.get(md5)?.unshift(file); } else { md5RawMap.set(md5, [file]); } } const sha1 = file.getSha1(); if (sha1) { if (sha1RawMap.has(sha1)) { sha1RawMap.get(sha1)?.unshift(file); } else { sha1RawMap.set(sha1, [file]); } } const sha256 = file.getSha256(); if (sha256) { if (sha256RawMap.has(sha256)) { sha256RawMap.get(sha256)?.unshift(file); } else { sha256RawMap.set(sha256, [file]); } } if (file.getFileHeader()) { const crc32WithoutHeader = `${file.getCrc32WithoutHeader()}|${file.getSizeWithoutHeader()}`; if (crc32WithoutHeaderMap.has(crc32WithoutHeader)) { crc32WithoutHeaderMap.get(crc32WithoutHeader)?.push(file); } else { crc32WithoutHeaderMap.set(crc32WithoutHeader, [file]); } const md5WithoutHeader = file.getMd5WithoutHeader(); if (md5WithoutHeader) { if (md5WithoutHeaderMap.has(md5WithoutHeader)) { md5WithoutHeaderMap.get(md5WithoutHeader)?.push(file); } else { md5WithoutHeaderMap.set(md5WithoutHeader, [file]); } } const sha1WithoutHeader = file.getSha1WithoutHeader(); if (sha1WithoutHeader) { if (sha1WithoutHeaderMap.has(sha1WithoutHeader)) { sha1WithoutHeaderMap.get(sha1WithoutHeader)?.push(file); } else { sha1WithoutHeaderMap.set(sha1WithoutHeader, [file]); } } const sha256WithoutHeader = file.getSha256WithoutHeader(); if (sha256WithoutHeader) { if (sha256WithoutHeaderMap.has(sha256WithoutHeader)) { sha256WithoutHeaderMap.get(sha256WithoutHeader)?.push(file); } else { sha256WithoutHeaderMap.set(sha256WithoutHeader, [file]); } } } for (const romPadding of file.getPaddings()) { const paddedCrc32 = romPadding.getCrc32(); if (paddedCrc32) { if (crc32PaddedMap.has(paddedCrc32)) { crc32PaddedMap.get(paddedCrc32)?.push(file); } else { crc32PaddedMap.set(paddedCrc32, [file]); } } const paddedMd5 = romPadding.getMd5(); if (paddedMd5) { if (md5PaddedMap.has(paddedMd5)) { md5PaddedMap.get(paddedMd5)?.push(file); } else { md5PaddedMap.set(paddedMd5, [file]); } } const paddedSha1 = romPadding.getSha1(); if (paddedSha1) { if (sha1PaddedMap.has(paddedSha1)) { sha1PaddedMap.get(paddedSha1)?.push(file); } else { sha1PaddedMap.set(paddedSha1, [file]); } } const paddedSha256 = romPadding.getSha256(); if (paddedSha256) { if (sha256PaddedMap.has(paddedSha256)) { sha256PaddedMap.get(paddedSha256)?.push(file); } else { sha256PaddedMap.set(paddedSha256, [file]); } } } }); const crc32Map = this.combineMaps(crc32RawMap, crc32WithoutHeaderMap, crc32PaddedMap); const md5Map = this.combineMaps(md5RawMap, md5WithoutHeaderMap, md5PaddedMap); const sha1Map = this.combineMaps(sha1RawMap, sha1WithoutHeaderMap, sha1PaddedMap); const sha256Map = this.combineMaps(sha256RawMap, sha256WithoutHeaderMap, sha256PaddedMap); return new IndexedFiles(crc32Map, md5Map, sha1Map, sha256Map); } static combineMaps(withHeaders, withoutHeaders, padded) { const result = new Map(withHeaders); [...withoutHeaders.entries()] // Prefer "raw" files as they exist on disk, without any header manipulation .filter(([checksum]) => !result.has(checksum)) .forEach(([checksum, files]) => { result.set(checksum, files); }); [...padded.entries()] // Prefer "raw" files as they exist on disk, without any padding manipulation .filter(([checksum]) => !result.has(checksum)) .forEach(([checksum, files]) => { result.set(checksum, files); }); return result; } getFiles() { return [ ...this.crc32.values(), ...this.md5.values(), ...this.sha1.values(), ...this.sha256.values(), ] .flat() .filter(ArrayPoly.filterUniqueMapped((file) => file.toString())); } getFilesByFilePath() { return this.getFiles().reduce((map, file) => { const key = file.getFilePath(); if (map.has(key)) { map.get(key)?.push(file); } else { map.set(key, [file]); } return map; }, new Map()); } getSize() { return this.getFiles().length; } /** * Find file(s) in the index based some search criteria. */ findFiles(file) { const sha256 = file.sha256?.replace(/[^0-9a-f]/gi, ''); if (sha256 && this.sha256.has(sha256)) { return this.sha256.get(sha256); } const sha1 = file.sha1?.replace(/[^0-9a-f]/gi, ''); if (sha1 && this.sha1.has(sha1)) { return this.sha1.get(sha1); } const md5 = file.md5?.replace(/[^0-9a-f]/gi, ''); if (md5 && this.md5.has(md5)) { return this.md5.get(md5); } const crc32 = file.crc32?.replace(/[^0-9a-f]/gi, ''); if (crc32) { const crc32WithSize = `${crc32}|${file.getSize()}`; if (this.crc32.has(crc32WithSize)) { return this.crc32.get(crc32WithSize); } } return undefined; } } __decorate([ Memoize(), __metadata("design:type", Function), __metadata("design:paramtypes", []), __metadata("design:returntype", Array) ], IndexedFiles.prototype, "getFiles", null); __decorate([ Memoize(), __metadata("design:type", Function), __metadata("design:paramtypes", []), __metadata("design:returntype", Map) ], IndexedFiles.prototype, "getFilesByFilePath", null);