Копіюйте папку рекурсивно в node.js


154

Є простіший спосіб , щоб скопіювати папку і всі її вміст , не вручну робити послідовність fs.readir, fs.readfile, fs.writefileрекурсивно?

Мені цікаво, чи не вистачає мені функції, яка в ідеалі могла б працювати саме так

fs.copy("/path/to/source/folder","/path/to/destination/folder");

3
Чи є спосіб це зробити без модулів? Може бути, рекурсивна функція / код фрагменту?
Сукіма

@Sukima - Дивіться мою відповідь тут .
jmort253

Відповіді:


121

Ви можете використовувати модуль ncp . Я думаю, що це те, що вам потрібно


2
Ідеально! npm install ncpі працює менше ніж за 30 років. Дякую.
Aebsubis

1
Ключ для мене краще, тому що він підтримує більше варіантів. Наприклад, з NCP ви не можете вирішити символьні посилання.
Слава Фомін II

3
Як дивовижний бонус, можна використовувати ncp в крос-платформних сценаріях запуску npm.
Сімантичний

Я отримав кілька простих випадків, коли ncp не переходить у зворотний дзвінок, де fs-extra правильно робить.
bumpmann

40
Зверніть увагу, що ncp, здається, не збережено . fs-extra , мабуть, найкращий варіант замість цього.
chris

74

Це мій підхід до вирішення цієї проблеми без зайвих модулів. Тільки з допомогою вбудованого fsі pathмодулів.

Примітка. При цьому використовується функція читання / запису fs, щоб вона не копіювала жодних метаданих (час створення тощо). Станом на вузол 8.5 copyFileSyncдоступні функції, які викликають функції копіювання ОС, а отже, також копіюють метадані. Я їх ще не перевіряв, але це повинно працювати, щоб просто замінити їх. (Див. Https://nodejs.org/api/fs.html#fs_fs_copyfilesync_src_dest_flags )

var fs = require('fs');
var path = require('path');

function copyFileSync( source, target ) {

    var targetFile = target;

    //if target is a directory a new file with the same name will be created
    if ( fs.existsSync( target ) ) {
        if ( fs.lstatSync( target ).isDirectory() ) {
            targetFile = path.join( target, path.basename( source ) );
        }
    }

    fs.writeFileSync(targetFile, fs.readFileSync(source));
}

function copyFolderRecursiveSync( source, target ) {
    var files = [];

    //check if folder needs to be created or integrated
    var targetFolder = path.join( target, path.basename( source ) );
    if ( !fs.existsSync( targetFolder ) ) {
        fs.mkdirSync( targetFolder );
    }

    //copy
    if ( fs.lstatSync( source ).isDirectory() ) {
        files = fs.readdirSync( source );
        files.forEach( function ( file ) {
            var curSource = path.join( source, file );
            if ( fs.lstatSync( curSource ).isDirectory() ) {
                copyFolderRecursiveSync( curSource, targetFolder );
            } else {
                copyFileSync( curSource, targetFolder );
            }
        } );
    }
}

він не копіює папки, якщо у своїх іменах є пробіл
31415926

Для мене це копіює папки з пробілами у своїх іменах. Можливо, це було викликано помилкою, виправленою @victor. Оскільки я використовую цю функцію досить регулярно (у поточному стані, як я забув оновити той самий переможець виправлення), я цілком впевнений, що вона взагалі працює.
Simon Zyx

1
Також потрібні: javascript var fs = require('fs'); var path = require('path');
Тайлер

2
Це фактично не копіює файли. Він читає їх, потім пише їх. Це не копіювання. Копіювання включає дату створення, а також інші потоки метаданих, які підтримують і Windows, і MacOS і не копіюються цим кодом. Станом на вузол 8.5, вам слід зателефонувати fs.copyабо, fs.copySyncяк вони фактично викликають функції копіювання рівня ОС в MacOS та Windows, і так фактично копіювати файли.
gman

1
вибачте, fs.copyFileі якщо ви перекопаєте джерело вузла, яке ви побачите на Mac і Windows, вони зателефонують на функцію ОС, щоб скопіювати файл
gman

52

Є деякі модулі, які підтримують копіювання папок своїм вмістом. Найпопулярнішим був би гайковий ключ

// Deep-copy an existing directory
wrench.copyDirSyncRecursive('directory_to_copy', 'location_where_copy_should_end_up');

Альтернативою може бути node-fs-extra

fs.copy('/tmp/mydir', '/tmp/mynewdir', function (err) {
  if (err) {
    console.error(err);
  } else {
    console.log("success!");
  }
}); //copies directory, even if it has subdirectories or files

3
гайковий ключ не вдається, якщо каталог для копіювання містить символічне посилання
DoubleMalt

2
він також не працює в Windows, якщо каталог вже існує, ncp працював прямо з пакета.
Бланширований

6
node-fs-extra працював на мене. Він успадковує оригінальний fs, і мені сподобався спосіб керування процесом. Менше коду для оновлення в додатку.
dvdmn

15
Зверніть увагу, що wrenchця система була застаріла і її слід замінити на node-fs-extra( github.com/jprichardson/node-fs-extra )
Ambidex

1
Гайковий ключ насправді не копіює файли. Він читає їх, потім записує їх, а потім копіює їх дату. Це не копіювання. Копіювання включає в себе інші потоки метаданих, які підтримують і Windows, і MacOS і не копіюються гайковим ключем.
gman

38

Ось функція, яка рекурсивно копіює каталог та його вміст в інший каталог:

const fs = require("fs")
const path = require("path")

/**
 * Look ma, it's cp -R.
 * @param {string} src The path to the thing to copy.
 * @param {string} dest The path to the new copy.
 */
var copyRecursiveSync = function(src, dest) {
  var exists = fs.existsSync(src);
  var stats = exists && fs.statSync(src);
  var isDirectory = exists && stats.isDirectory();
  if (isDirectory) {
    fs.mkdirSync(dest);
    fs.readdirSync(src).forEach(function(childItemName) {
      copyRecursiveSync(path.join(src, childItemName),
                        path.join(dest, childItemName));
    });
  } else {
    fs.copyFileSync(src, dest);
  }
};

3
Навіть якщо ви вставили справжню функцію копіювання, ви не повинні переходити до символічних посилань (використовувати fs.lstatSyncзамість них fs.statSync)
Simon Zyx

3
що може спричинити цю плутанину - це те, що fs.unlink видаляє файли, але fs.link не копіює, а посилається.
Simon Zyx

3
@SimonSeyock: правильно .. ІТ linkingне копіює .. Проблема полягає в тому, що ви змінюєте вміст пов'язаного файлу, оригінальний файл також буде змінено.
Abdennour TOUMI


22

Для ОС Linux / unix можна використовувати синтаксис оболонки

const shell = require('child_process').execSync ; 

const src= `/path/src`;
const dist= `/path/dist`;

shell(`mkdir -p ${dist}`);
shell(`cp -r ${src}/* ${dist}`);

Це воно!


1
Ласкаво просимо 👋
Abdennour TOUMI

1
Це найпростіше рішення. Не потрібно заново вигадувати інструменти UNIX!
Майкл Францл

11
оскільки nodejs працює на OSX / linux / windows, це лише відповідь для 2-х не всіх 3.
mjwrazor

2
@AbdennourTOUMI що робити, якщо ви працюєте на сервері Windows.
mjwrazor

3
Ось чому я почав відповідь «Для Linux / unix OS, ви можете використовувати синтаксис оболонки ..» 👍🏼
Abdennour TOUMI

19

Модуль fs-extra працює як шарм.

Встановіть fs-extra

$ npm install fs-extra

Далі йде програма для копіювання вихідного каталогу в каталог призначення.

// include fs-extra package
var fs = require("fs-extra");

var source = 'folderA'
var destination = 'folderB'

// copy source folder to destination
fs.copy(source, destination, function (err) {
    if (err){
        console.log('An error occured while copying the folder.')
        return console.error(err)
    }
    console.log('Copy completed!')
});

Список літератури

fs-extra: https://www.npmjs.com/package/fs-extra

Приклад: Підручник NodeJS - Node.js Скопіюйте папку


цей процес замінює каталог чи зливається з ним?
С.М. Шахінул Іслам

14

Ось як я це зробив особисто:

function copyFolderSync(from, to) {
    fs.mkdirSync(to);
    fs.readdirSync(from).forEach(element => {
        if (fs.lstatSync(path.join(from, element)).isFile()) {
            fs.copyFileSync(path.join(from, element), path.join(to, element));
        } else {
            copyFolderSync(path.join(from, element), path.join(to, element));
        }
    });
}

працює для папок і файлів


3
Це рішення коротке і просте. Це було б майже точно так, як я це зробив, тому +1 від мене. Ви повинні вдосконалити свою відповідь за допомогою коментарів у своєму коді та описати, чому це рішення є кращим перед іншими та які недоліки у нього можуть бути. - Оновіть також необхідні модулі. ("шлях", "fs")
Андрій

перевірте, чи папка існує вгорі ... врятує життя ;-) якщо (! fs.existsSync (до)) fs.mkdirSync (до);
Тобіас

9

Я створив невеликий робочий приклад, який копіює вихідну папку в іншу папку призначення лише за кілька кроків (на основі відповіді @ shift66 за допомогою ncp):

крок 1 - Встановіть модуль ncp:

npm install ncp --save

крок 2 - створити copy.js (змінити srcPath і destPath vars на все, що потрібно):

var path = require('path');
var ncp = require('ncp').ncp;

ncp.limit = 16;

var srcPath = path.dirname(require.main.filename); //current folder
var destPath = '/path/to/destination/folder'; //Any destination folder

console.log('Copying files...');
ncp(srcPath, destPath, function (err) {
  if (err) {
    return console.error(err);
  }
  console.log('Copying files complete.');
});

крок 3 - запустіть

node copy.js

7

Це досить легко за допомогою вузла 10.

const FSP = require('fs').promises;

async function copyDir(src,dest) {
    const entries = await FSP.readdir(src,{withFileTypes:true});
    await FSP.mkdir(dest);
    for(let entry of entries) {
        const srcPath = Path.join(src,entry.name);
        const destPath = Path.join(dest,entry.name);
        if(entry.isDirectory()) {
            await copyDir(srcPath,destPath);
        } else {
            await FSP.copyFile(srcPath,destPath);
        }
    }
}

Це передбачає, destщо не існує.


3
Ми можемо отримати цю роботу в вузлі 8.x, використовуючи require('util').promisifyз fs.mkdirі fs.copyFileзамість того require('fs').promises, що все ще є експериментальною на v11.1.
Sơn Trần-Nguyễn

@sntran Чи є у 8.x withFileTypesваріант? Тому що це рятує вас від statдзвінка
вересня

На жаль, у 8.x немає withFileTypesможливості.
Sơn Trần-Nguyễn

@ SơnTrần-Nguyễn 8.x досягає кінця життя на 31 грудня 2019 року - може бути час , щоб оновити :-)
mpen

6

Я знаю вже стільки відповідей тут, але ніхто не відповів на це просто. Що стосується офіційної документації fs-exra , ви можете зробити це дуже просто

const fs = require('fs-extra')

// copy file
fs.copySync('/tmp/myfile', '/tmp/mynewfile')

// copy directory, even if it has subdirectories or files
fs.copySync('/tmp/mydir', '/tmp/mynewdir')

не забудьте встановити рекурсивну опцію. fs.copySync ('/ tmp / mydir', '/ tmp / mynewdir', {рекурсивний: правда})
Dheeraj Kumar

Я не можу знайти варіант { recursive: true }від github doc, який ви згадали. Не знаю, чи це працює.
Фредді Деніел

Я думаю, ми говоримо про fs-extra, але ваше посилання github вказує на node-fs-extra. Чи може бути інша бібліотека?
Dheeraj Kumar

@DheerajKumar, він показує node-fs-extra в github, але fs-extra в npm . Я не знаю, що обидва однакові, будь ласка, зверніться до пакету з нпм
Фредді Деніел

Чи замінює fs-extra fs?
Метт

4

Оскільки я просто будую простий сценарій вузла, я не хотів, щоб користувачі скрипту потребували імпортування купу зовнішніх модулів і залежностей, тому я надів мірку для роздумів і здійснив пошук запущених команд з bash оболонки.

Цей фрагмент коду node.js рекурсивно копіює папку під назвою node-webkit.app у папку під назвою build:

   child = exec("cp -r node-webkit.app build", function(error, stdout, stderr) {
        sys.print("stdout: " + stdout);
        sys.print("stderr: " + stderr);
        if(error !== null) {
            console.log("exec error: " + error);
        } else {

        }
    });

Завдяки Ланс Поллард у зоні за те, що я почав працювати.

Вищенаведений фрагмент обмежений платформами на базі Unix, таких як Mac OS та Linux, але подібна методика може працювати для Windows.


4

@ mallikarjun-m дякую!

fs-extra зробив це, і він навіть може повернути Обіцяння, якщо ви не надаєте зворотній дзвінок! :)

const path = require('path')
const fs = require('fs-extra')

let source = path.resolve( __dirname, 'folderA')
let destination = path.resolve( __dirname, 'folderB')

fs.copy(source, destination)
  .then(() => console.log('Copy completed!'))
  .catch( err => {
    console.log('An error occured while copying the folder.')
    return console.error(err)
  })

2

Той, який підтримує символічне посилання + не кидає, якщо існує каталог.

function copyFolderSync(from, to) {
  try {
    fs.mkdirSync(to);
  } catch(e) {}

  fs.readdirSync(from).forEach((element) => {
    const stat = fs.lstatSync(path.join(from, element));
    if (stat.isFile()) {
      fs.copyFileSync(path.join(from, element), path.join(to, element));
    } else if (stat.isSymbolicLink()) {
      fs.symlinkSync(fs.readlinkSync(path.join(from, element)), path.join(to, element));
    } else if (stat.isDirectory()) {
      copyFolderSync(path.join(from, element), path.join(to, element));
    }
  });
}

1

Цей код буде добре працювати, рекурсивно копіюючи будь-яку папку в будь-яке місце. Тільки для Windows.

var child=require("child_process");
function copySync(from,to){
    from=from.replace(/\//gim,"\\");
    to=to.replace(/\//gim,"\\");
    child.exec("xcopy /y /q \""+from+"\\*\" \""+to+"\\\"");
}

Відмінно працює в моїй текстовій грі для створення нових гравців.


1

Я спробував fs-extra та copy-dir скопіювати папку-рекурсивно. але я цього хочу

  1. працює нормально (копія-режр викидає невиправдану помилку)
  2. надає два аргументи у фільтрі: filepath та filetype (fs-extra не повідомляє тип файлу)
  3. має перевірку dir-to-subdir та dir-to-file

Тому я написав своє:

//node module for node 8.6+
var path=require("path");
var fs=require("fs");

function copyDirSync(src,dest,options){
  var srcPath=path.resolve(src);
  var destPath=path.resolve(dest);
  if(path.relative(srcPath,destPath).charAt(0)!=".")
    throw new Error("dest path must be out of src path");
  var settings=Object.assign(Object.create(copyDirSync.options),options);
  copyDirSync0(srcPath,destPath,settings);
  function copyDirSync0(srcPath,destPath,settings){
    var files=fs.readdirSync(srcPath);
    if (!fs.existsSync(destPath)) {
      fs.mkdirSync(destPath);
    }else if(!fs.lstatSync(destPath).isDirectory()){
      if(settings.overwrite)
        throw new Error(`Cannot overwrite non-directory '${destPath}' with directory '${srcPath}'.`);
      return;
    }
    files.forEach(function(filename){
      var childSrcPath=path.join(srcPath,filename);
      var childDestPath=path.join(destPath,filename);
      var type=fs.lstatSync(childSrcPath).isDirectory()?"directory":"file";
      if(!settings.filter(childSrcPath,type))
        return;
      if (type=="directory") {
        copyDirSync0(childSrcPath,childDestPath,settings);
      } else {
        fs.copyFileSync(childSrcPath, childDestPath, settings.overwrite?0:fs.constants.COPYFILE_EXCL);
        if(!settings.preserveFileDate)
          fs.futimesSync(childDestPath,Date.now(),Date.now());
      }
    });
  }
}
copyDirSync.options={
  overwrite: true,
  preserveFileDate: true,
  filter: function(filepath,type){return true;}
};

і аналогічну функцію mkdirs, яка є альтернативою mkdirp

function mkdirsSync(dest) {
  var destPath=path.resolve(dest);
  mkdirsSync0(destPath);
  function mkdirsSync0(destPath){
    var parentPath=path.dirname(destPath);
    if(parentPath==destPath)
      throw new Error(`cannot mkdir ${destPath}, invalid root`);
    if (!fs.existsSync(destPath)) {
      mkdirsSync0(parentPath);
      fs.mkdirSync(destPath);
    }else if(!fs.lstatSync(destPath).isDirectory()){
      throw new Error(`cannot mkdir ${destPath}, a file already exists there`);
    }
  }
}

0

Я написав цю функцію як для копіювання (copyFileSync), так і для переміщення (renameSync) файлів рекурсивно між каталогами:

//copy files
copyDirectoryRecursiveSync(sourceDir, targetDir);
//move files
copyDirectoryRecursiveSync(sourceDir, targetDir, true);


function copyDirectoryRecursiveSync(source, target, move) {
if (!fs.lstatSync(source).isDirectory()) return;

var operation = move ? fs.renameSync : fs.copyFileSync;
fs.readdirSync(source).forEach(function (itemName) {
    var sourcePath = path.join(source, itemName);
    var targetPath = path.join(target, itemName);

    if (fs.lstatSync(sourcePath).isDirectory()) {
        fs.mkdirSync(targetPath);
        copyDirectoryRecursiveSync(sourcePath, targetDir);
    }
    else {
        operation(sourcePath, targetPath);
    }
});}

0

Якщо ви перебуваєте в Linux, а продуктивність не є проблемою, ви можете використовувати execфункцію з child_processмодуля для виконання команди bash:

const { exec } = require('child_process');
exec('cp -r source dest', (error, stdout, stderr) => {...});

У деяких випадках я вважаю це рішення більш чистим, ніж завантаження цілого модуля або навіть використання fsмодуля.


0

ncp блокує дескриптор файлу та спрацьовує зворотний виклик, коли він ще не розблокований. Рекомендую натомість використовувати модуль рекурсивної копії . Він підтримує події, і ви можете бути впевнені в копії.


0

Будьте уважні, вибираючи пакет. Деякі пакети, такі як copy-dir, не підтримують копіювати великий файл довжиною більше 0x1fffffe8. Це призведе до такої помилки, як:

buffer.js:630 Uncaught Error: Cannot create a string longer than 0x1fffffe8 characters 

Я пережив щось подібне в одному зі своїх проектів. Зрештою, мені довелося змінити пакет, який я використовував, і відкоригувати багато коду. Я б сказав, що це не дуже приємне враження.

Якщо потрібно декілька джерел і кілька копій призначення, ви можете скористатися кращою копією та написати щось подібне:

// copy from multiple source into a directory
bCopy(['/path/to/your/folder1', '/path/to/some/file.txt'], '/path/to/destination/folder');

або навіть :

// copy from multiple source into multiple destination
bCopy(['/path/to/your/folder1', '/path/to/some/file.txt'], ['/path/to/destination/folder', '/path/to/another/folder']);

-1

ТАК, ncpєcool хоча ...

Можливо, ви хочете / повинні вподобати, що його функція є super cool. Оскільки ви в ньому, додайте його доtools файл, щоб повторно його використовувати.

Нижче представлена ​​робоча версія, яка є Asyncта використовується Promises.


index.js

const {copyFolder} = require('./tools/');

return copyFolder(
    yourSourcePath,
    yourDestinationPath
)
.then(() => {
    console.log('-> Backup completed.')
}) .catch((err) => {
    console.log("-> [ERR] Could not copy the folder: ", err);
})

tools.js

const ncp = require("ncp");

/**
 * Promise Version of ncp.ncp()
 * 
 * This function promisifies ncp.ncp().
 * We take the asynchronous function ncp.ncp() with 
 * callback semantics and derive from it a new function with
 * promise semantics.
 */
ncp.ncpAsync = function (sourcePath, destinationPath) {
  return new Promise(function (resolve, reject) {
      try {
          ncp.ncp(sourcePath, destinationPath, function(err){
              if (err) reject(err); else resolve();
          });
      } catch (err) {
          reject(err);
      }
  });
};

/**
 * Utility function to copy folders asynchronously using
 * the Promise returned by ncp.ncp(). 
 */
const copyFolder = (sourcePath, destinationPath) => {
    return ncp.ncpAsync(sourcePath, destinationPath, function (err) {
        if (err) {
            return console.error(err);
        }
    });
}
module.exports.copyFolder = copyFolder;

-1

Найпростішим підходом до цієї проблеми є використання лише модулів 'fs' та 'Path' та певної логіки .....

Усі файли в кореневій папці копіюються з Новим ім'ям, якщо ви хочете просто встановити номер версії, тобто ....................... "var v = 'Ваша директорія Ім'я ""

у вмісті префікса V імені файлу, доданому разом з ім'ям файлу.

var fs = require('fs-extra');
var path = require('path');

var c = 0;
var i =0 ;
var v = "1.0.2";
var copyCounter = 0;
var directoryCounter = 0; 
var directoryMakerCounter = 0;
var recursionCounter = -1;
var Flag = false;
var directoryPath = [] ;
var directoryName = [] ;
var directoryFileName = [];
var fileName;
var directoryNameStorer;
var dc = 0;
var route ;



if (!fs.existsSync(v)){
   fs.mkdirSync(v);
}

var basePath = path.join(__dirname, v);


function walk(dir){

  fs.readdir(dir, function(err, items) {

    items.forEach(function(file){

        file = path.resolve(dir, file);

        fs.stat(file, function(err, stat){
            if(stat && stat.isDirectory()){

                directoryNameStorer = path.basename(file);
                route = file;
                route = route.replace("gd",v);

                directoryFileName[directoryCounter] = route;
                directoryPath[directoryCounter] = file;
                directoryName[directoryCounter] = directoryNameStorer;

                directoryCounter++;
                dc++;

                if (!fs.existsSync(basePath+"/"+directoryName[directoryMakerCounter])){
                    fs.mkdirSync(directoryFileName[directoryMakerCounter]);
                    directoryMakerCounter++;
                }

            }else{

                    fileName = path.basename(file);
                    if(recursionCounter >= 0){
                        fs.copyFileSync(file, directoryFileName[recursionCounter]+"/"+v+"_"+fileName, err => {
                            if(err) return console.error(err);
                        });
                        copyCounter++;
                    }else{
                        fs.copyFileSync(file, v+"/"+v+"_"+fileName, err => {
                            if(err) return console.error(err);
                        });
                        copyCounter++;    
                    }

                }
                if(copyCounter + dc == items.length && directoryCounter > 0 && recursionCounter < directoryMakerCounter-1){
                    console.log("COPY COUNTER :             "+copyCounter);
                    console.log("DC COUNTER :               "+dc);                        
                    recursionCounter++;
                    dc = 0;
                    copyCounter = 0;
                    console.log("ITEM DOT LENGTH :          "+items.length);
                    console.log("RECURSION COUNTER :        "+recursionCounter);
                    console.log("DIRECOTRY MAKER COUNTER :  "+directoryMakerCounter);
                    console.log(": START RECURSION :        "+directoryPath[recursionCounter]);
                    walk(directoryPath[recursionCounter]); //recursive call to copy sub-folder

                }

        })
    })
 });

}
 walk('./gd', function(err, data){ //Just Pass The Root Directory Which You Want to Copy
 if(err) throw err;
 console.log("done");
})

-1

Ось як я зробив:

let fs = require('fs');
let path = require('path');

тоді:

let filePath = //your FilePath

let fileList = []
        var walkSync = function(filePath, filelist) 
        {
          let files = fs.readdirSync(filePath);
          filelist = filelist || [];
          files.forEach(function(file) 
          {
            if (fs.statSync(path.join(filePath, file)).isDirectory()) 
            {
              filelist = walkSync(path.join(filePath, file), filelist);
            }
            else 
            {
              filelist.push(path.join(filePath, file));
            }
          });

          // Ignore hidden files
          filelist = filelist.filter(item => !(/(^|\/)\.[^\/\.]/g).test(item));

          return filelist;
        };

Потім зателефонуйте до методу:

This.walkSync(filePath, fileList)
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.