Viewing File: /home/ubuntu/efiexchange-node-base/node_modules/swarm-js/src/files.js

// This module implements some file download utils. Its most important export
// is `safeDownloadTargzFile`, which, given a file, its md5, a tar.gz url, its
// md5 and a path, returns a Promise that will only resolve once the exact file
// you expect is available on that path.

const Q = require("bluebird");
const assert = require("assert");
const crypto = require("crypto");
const fs = require("fs-extra");
const got = require("got");
const mkdirp = require("mkdirp-promise");
const path = require("path");
const tar = require("tar");

// String -> String ~> Promise String
//   Downloads a file from an url to a path.
//   Returns a promise containing the path.
const download = url => filePath => {
  const promise = Q.resolve(mkdirp(path.dirname(filePath))).then(() =>
    new Q((resolve, reject) => {
      const writeStream = fs.createWriteStream(filePath);
      const downloadStream = got.stream(url);
      downloadStream.on("end", () => resolve(filePath));
      downloadStream.on("data", chunk => promise.onDataCallback(chunk));
      downloadStream.on("error", reject);
      downloadStream.pipe(writeStream);
    }));
  promise.onDataCallback = () => {};
  promise.onData = callback => {
    promise.onDataCallback = callback || (() => {});
    return promise;
  }
  return promise;
};

// String -> String ~> Promise String
//   Hashes a file using the given algorithm (ex: "md5").
//   Returns a promise containing the hashed string.
const hash = algorithm => path =>
  new Q((resolve, reject) => {
    const readStream = fs.ReadStream(path);
    const hash = crypto.createHash(algorithm);
    readStream.on("data", d => hash.update(d));
    readStream.on("end", () => resolve(hash.digest("hex")));
    readStream.on("error", reject);
  });

// String -> String ~> Promise ()
//   Asserts a file matches this md5 hash.
//   Returns a promise containing its path.
const checksum = fileHash => path =>
  hash("md5")(path)
    .then(actualHash => actualHash === fileHash)
    .then(assert)
    .then(() => path);

// String ~> String ~> String ~> Promise String
//   Downloads a file to a directory, check.
//   Checks if the md5 hash matches.
//   Returns a promise containing the path.
const downloadAndCheck = url => path => fileHash =>
  download(url)(path)
    .then(checksum(fileHash));

// String -> String ~> Promise String
//   TODO: work for zip and other types
const extract = fromPath => toPath =>
  tar.x(fromPath, toPath)
    .then(() => toPath);

// String ~> Promise String
//   Reads a file as an UTF8 string.
//   Returns a promise containing that string.
const readUTF8 = path =>
  fs.readFile (path, {encoding: "utf8"});

// String ~> Promise Bool
const isDirectory = path =>
  fs.exists(path)
    .then(assert)
    .then(() => fs.lstat(path))
    .then(stats => stats.isDirectory())
    .catch(() => false);

// String -> Promise String
const directoryTree = dirPath => {
  let paths = [];
  const search = dirPath =>
    isDirectory(dirPath).then(isDir => {
      if (isDir) {
        const searchOnDir = dir => search (path.join(dirPath, dir));
        return Q.all(Q.map(fs.readdir(dirPath), searchOnDir));
      } else {
        paths.push(dirPath);
      };
    });
  return Q.all(search (dirPath)).then(() => paths);
}

// Regex -> String ~> Promise (Array String)
const search = regex => dirPath => 
  directoryTree(dirPath)
    .then(tree => tree.filter(path => regex.test(path)));

// String -> String -> String -> String ~> Promise String
//   Downloads a file inside a tar.gz and places it at `filePath`.
//   Checks the md5 hash of the tar before extracting it.
//   Checks the md5 hash of the file after extracting it.
//   If all is OK, returns a promise containing the file path.
const safeDownloadArchived = url => archiveHash => fileHash => filePath => {
  const fileDir = path.dirname(filePath);
  const fileName = path.basename(filePath);
  const archivePath = path.join(fileDir, ".swarm_downloads/files.tar.gz");
  const archiveDir = path.dirname(archivePath);
  const promise = Q.resolve(mkdirp(archiveDir))
    .then(() => checksum (fileHash) (filePath))
    .then(() => filePath)
    .catch(() => fs.exists(archiveDir)
      .then(exists => !exists ? fs.mkdir(archiveDir) : null)
      .then(() => download (url)(archivePath).onData(promise.onDataCallback))
      .then(() => hash("md5")(archivePath))
      .then(() => archiveHash ? checksum(archiveHash)(archivePath) : null)
      .then(() => extract (archivePath) (archiveDir))
      .then(() => search (new RegExp(fileName+"$")) (archiveDir))
      .then(fp => fs.rename (fp[0], filePath))
      .then(() => fs.unlink (archivePath))
      .then(() => fileHash ? checksum(fileHash)(filePath) : null)
      .then(() => filePath));
  promise.onDataCallback = () => {};
  promise.onData = callback => {
    promise.onDataCallback = callback || (() => {});
    return promise;
  };
  return promise;
};

// String -> String ~> Promise String
//   Like `safeDownloadArchivedFile`, but without the checksums.
const downloadArchived = url => path =>
  safeDownloadArchived (url) (null) (null) (path);

// () => Promise Bool
//   Tests the implementation by downloading a predetermined tar.gz
//   from a mocked HTTP-server into a mocked filesystem. Does some
//   redundancy tests such as checking the file constents and double
//   checking its MD5 hash.
//   Returns a promise containing a boolean, true if tests passed.
const test = () => {
  const filePath = "/swarm/foo.txt";
  const fileHash = "d3b07384d113edec49eaa6238ad5ff00";
  const archiveUrl = "http://localhost:12534";
  const archiveHash = "7fa45f946bb2a696bdd9972e0fbceac2";
  const archiveData = new Buffer([
    0x1f,0x8b,0x08,0x00,0xf1,0x34,0xaf,0x58,0x00,0x03,0xed,0xcf,0x3d,0x0e,0x83,0x30,
    0x0c,0x86,0x61,0x66,0x4e,0xe1,0x13,0x54,0xce,0x0f,0xc9,0x79,0x58,0xb2,0x46,0x82,
    0x14,0x71,0x7c,0xd2,0x06,0x31,0x52,0x75,0x40,0x08,0xe9,0x7d,0x96,0x4f,0x96,0x3d,
    0x7c,0x4e,0x39,0xbf,0xca,0x5a,0xba,0x2b,0xa9,0x6a,0xf0,0x5e,0x3e,0x19,0xc3,0xf0,
    0x4d,0xb5,0x6d,0xde,0x79,0x31,0x4e,0x07,0x17,0x9c,0xb5,0x31,0x8a,0x1a,0xab,0xc6,
    0x77,0xa2,0x97,0xb6,0xda,0xbd,0xe7,0x32,0x4e,0xb5,0xca,0xf2,0xe3,0xae,0x9e,0xa5,
    0x74,0xb2,0x6f,0x8f,0xc8,0x91,0x0f,0x91,0x72,0xee,0xef,0xee,0x00,0x00,0x00,0x00,
    0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xf8,0xdf,0x06,0xb3,0x2a,0xda,
    0xed,0x00,0x28,0x00,0x00]);

  const crypto = require("crypto");
  const fsMock = require("mock-fs")({"/swarm":{}});
  const httpMock = require("http")
    .createServer((_,res) => res.end(archiveData))
    .listen(12534);

  return safeDownloadArchived (archiveUrl) (archiveHash) (fileHash) (filePath)
    .then(checksum (fileHash))
    .then(readUTF8)
    .then(text => text === "foo\n")
    .then(assert)
    .then(() => safeDownloadArchived (archiveUrl) (archiveHash) (fileHash) (filePath))
    .then(() => true)
    .catch(false)
    .finally(() => httpMock.close());
};

module.exports = {
  download,
  hash,
  checksum,
  downloadAndCheck,
  extract,
  readUTF8,
  safeDownloadArchived,
  directoryTree,
  downloadArchived,
  search,
  test
};
Back to Directory File Manager