В этом варианте используется максимальное количество новых, модных слов, доступных в узле 8, включая обещания, использование / обещание, деструктуризацию, асинхронное ожидание, отображение + уменьшение и многое другое, что заставляет ваших коллег почесывать голову при попытке выяснить, что происходит.
Узел 8 +
Нет внешних зависимостей.
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), []);
}
Использование
getFiles(__dirname)
.then(files => console.log(files))
.catch(e => console.error(e));
Узел 10.10 +
Обновлено для узла 10+ с еще большим количеством ударов:
const { resolve } = require('path');
const { readdir } = require('fs').promises;
async function getFiles(dir) {
const dirents = await readdir(dir, { withFileTypes: true });
const files = dirents.map((dirent) => {
const res = resolve(dir, dirent.name);
return dirent.isDirectory() ? getFiles(res) : res;
});
return Array.prototype.concat(...files);
}
Обратите внимание, что начиная с узла 11.15.0, вы можете использовать files.flat()
вместо Array.prototype.concat(...files)
для выравнивания массива файлов.
Узел 11 +
Если вы хотите полностью поднять голову всем, вы можете использовать следующую версию, используя асинхронные итераторы . Помимо того, что он действительно крут, он также позволяет потребителям получать результаты по одному, что делает его более подходящим для действительно больших каталогов.
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;
}
}
}
Использование изменилось, потому что тип возвращаемого значения теперь асинхронный итератор вместо обещания
(async () => {
for await (const f of getFiles('.')) {
console.log(f);
}
})()
На случай, если кому-то интересно, я написал больше об асинхронных итераторах здесь: https://qwtel.com/posts/software/async-generators-in-the-wild/