| import * as crypto from 'node:crypto'; | |
| const hashes = Object.create(null); | |
| //TODO shorter? | |
| const hash_length = 12; | |
| /** | |
| * replaces +/= in base64 output so they don't interfere | |
| * | |
| * @param {string} input | |
| * @returns {string} base64 hash safe to use in any context | |
| */ | |
| export function safeBase64Hash(input) { | |
| if (hashes[input]) { | |
| return hashes[input]; | |
| } | |
| //TODO if performance really matters, use a faster one like xx-hash etc. | |
| // should be evenly distributed because short input length and similarities in paths could cause collisions otherwise | |
| // OR DON'T USE A HASH AT ALL, what about a simple counter? | |
| const md5 = crypto.createHash('md5'); | |
| md5.update(input); | |
| const hash = toSafe(md5.digest('base64')).slice(0, hash_length); | |
| hashes[input] = hash; | |
| return hash; | |
| } | |
| /** @type {Record<string, string>} */ | |
| const replacements = { | |
| '+': '-', | |
| '/': '_', | |
| '=': '' | |
| }; | |
| const replaceRE = new RegExp(`[${Object.keys(replacements).join('')}]`, 'g'); | |
| /** | |
| * @param {string} base64 | |
| * @returns {string} | |
| */ | |
| function toSafe(base64) { | |
| return base64.replace(replaceRE, (x) => replacements[x]); | |
| } | |