In diesem Fall wird die maximale Anzahl neuer, in Knoten 8 verfügbarer, witziger Funktionen verwendet, einschließlich Versprechen, Nutzen / Versprechen, Destrukturieren, asynchrones Warten, Zuordnen + Reduzieren und mehr, sodass Ihre Mitarbeiter sich am Kopf kratzen, wenn sie herausfinden möchten, was passiert es geht voran.
Knoten 8+
Keine externen Abhängigkeiten.
const { promisify } = require('util');
const { resolve } = require('path');
const fs = require('fs');
const readdir = promisify(fs.readdir);
const stat = promisify(fs.stat);
async function getFiles(dir) {
const subdirs = await readdir(dir);
const files = await Promise.all(subdirs.map(async (subdir) => {
const res = resolve(dir, subdir);
return (await stat(res)).isDirectory() ? getFiles(res) : res;
}));
return files.reduce((a, f) => a.concat(f), []);
}
Verwendung
getFiles(__dirname)
.then(files => console.log(files))
.catch(e => console.error(e));
Knoten 10.10+
Aktualisiert für Knoten 10+ mit noch mehr Whizbang:
const { resolve } = require('path');
const { readdir } = require('fs').promises;
async function getFiles(dir) {
const dirents = await readdir(dir, { withFileTypes: true });
const files = await Promise.all(dirents.map((dirent) => {
const res = resolve(dir, dirent.name);
return dirent.isDirectory() ? getFiles(res) : res;
}));
return Array.prototype.concat(...files);
}
Beachten Sie, dass Sie ab Knoten 11.15.0 files.flat()
anstelle von verwenden könnenArray.prototype.concat(...files)
zu .
Knoten 11+
Wenn Sie alle völlig in die Luft jagen möchten, können Sie die folgende Version mit asynchronen Iteratoren verwenden . Es ist nicht nur wirklich cool, sondern ermöglicht es den Verbrauchern auch, die Ergebnisse einzeln abzurufen, wodurch es sich besser für wirklich große Verzeichnisse eignet.
const { resolve } = require('path');
const { readdir } = require('fs').promises;
async function* getFiles(dir) {
const dirents = await readdir(dir, { withFileTypes: true });
for (const dirent of dirents) {
const res = resolve(dir, dirent.name);
if (dirent.isDirectory()) {
yield* getFiles(res);
} else {
yield res;
}
}
}
Die Verwendung hat sich geändert, da der Rückgabetyp jetzt ein asynchroner Iterator anstelle eines Versprechens ist
(async () => {
for await (const f of getFiles('.')) {
console.log(f);
}
})()
Falls jemand interessiert ist, habe ich hier mehr über asynchrone Iteratoren geschrieben: https://qwtel.com/posts/software/async-generators-in-the-wild/