Files
Tiber365/node_modules/unstorage/drivers/fs.mjs
2025-07-24 18:46:24 +02:00

123 lines
3.2 KiB
JavaScript

import { existsSync, promises as fsp } from "node:fs";
import { resolve, relative, join } from "node:path";
import { watch } from "chokidar";
import anymatch from "anymatch";
import { createError, createRequiredError, defineDriver } from "./utils/index.mjs";
import {
readFile,
writeFile,
readdirRecursive,
rmRecursive,
unlink
} from "./utils/node-fs.mjs";
const PATH_TRAVERSE_RE = /\.\.:|\.\.$/;
const DRIVER_NAME = "fs";
export default defineDriver((userOptions = {}) => {
if (!userOptions.base) {
throw createRequiredError(DRIVER_NAME, "base");
}
const base = resolve(userOptions.base);
const ignore = anymatch(
userOptions.ignore || ["**/node_modules/**", "**/.git/**"]
);
const r = (key) => {
if (PATH_TRAVERSE_RE.test(key)) {
throw createError(
DRIVER_NAME,
`Invalid key: ${JSON.stringify(key)}. It should not contain .. segments`
);
}
const resolved = join(base, key.replace(/:/g, "/"));
return resolved;
};
let _watcher;
const _unwatch = async () => {
if (_watcher) {
await _watcher.close();
_watcher = void 0;
}
};
return {
name: DRIVER_NAME,
options: userOptions,
flags: {
maxDepth: true
},
hasItem(key) {
return existsSync(r(key));
},
getItem(key) {
return readFile(r(key), "utf8");
},
getItemRaw(key) {
return readFile(r(key));
},
async getMeta(key) {
const { atime, mtime, size, birthtime, ctime } = await fsp.stat(r(key)).catch(() => ({}));
return { atime, mtime, size, birthtime, ctime };
},
setItem(key, value) {
if (userOptions.readOnly) {
return;
}
return writeFile(r(key), value, "utf8");
},
setItemRaw(key, value) {
if (userOptions.readOnly) {
return;
}
return writeFile(r(key), value);
},
removeItem(key) {
if (userOptions.readOnly) {
return;
}
return unlink(r(key));
},
getKeys(_base, topts) {
return readdirRecursive(r("."), ignore, topts?.maxDepth);
},
async clear() {
if (userOptions.readOnly || userOptions.noClear) {
return;
}
await rmRecursive(r("."));
},
async dispose() {
if (_watcher) {
await _watcher.close();
}
},
async watch(callback) {
if (_watcher) {
return _unwatch;
}
await new Promise((resolve2, reject) => {
const watchOptions = {
ignoreInitial: true,
...userOptions.watchOptions
};
if (!watchOptions.ignored) {
watchOptions.ignored = [];
} else if (Array.isArray(watchOptions.ignored)) {
watchOptions.ignored = [...watchOptions.ignored];
} else {
watchOptions.ignored = [watchOptions.ignored];
}
watchOptions.ignored.push(ignore);
_watcher = watch(base, watchOptions).on("ready", () => {
resolve2();
}).on("error", reject).on("all", (eventName, path) => {
path = relative(base, path);
if (eventName === "change" || eventName === "add") {
callback("update", path);
} else if (eventName === "unlink") {
callback("remove", path);
}
});
});
return _unwatch;
}
};
});