Image Importer Improvements

ref #4608, #4609, #4690

- fix errors with cleaning up files
- improve handling of base directories, and introduce a simple valid format for zips (must contain importable files or folders, and may contain up to one base directory)
- vastly improve test coverage
This commit is contained in:
Hannah Wolfe 2014-12-29 18:33:47 +00:00
parent 75e081dbf0
commit add4c6b078
12 changed files with 343 additions and 84 deletions

View File

@ -9,7 +9,8 @@ var Promise = require('bluebird'),
handleErrors,
checkDuplicateAttributes,
sanitize,
cleanError;
cleanError,
doImport;
cleanError = function cleanError(error) {
var temp,
@ -184,7 +185,7 @@ validate = function validate(data) {
});
};
module.exports = function (data) {
doImport = function (data) {
var sanitizeResults = sanitize(data);
data = sanitizeResults.data;
@ -197,3 +198,5 @@ module.exports = function (data) {
return handleErrors(result);
});
};
module.exports.doImport = doImport;

View File

@ -10,24 +10,25 @@ ImageHandler = {
type: 'images',
extensions: config.uploads.extensions,
types: config.uploads.contentTypes,
directories: ['images', 'content'],
loadFile: function (files, startDir) {
loadFile: function (files, baseDir) {
var store = storage.getStorage(),
startDirRegex = startDir ? new RegExp('^' + startDir + '/') : new RegExp(''),
baseDirRegex = baseDir ? new RegExp('^' + baseDir + '/') : new RegExp(''),
imageFolderRegexes = _.map(config.paths.imagesRelPath.split('/'), function (dir) {
return new RegExp('^' + dir + '/');
});
// normalize the directory structure
files = _.map(files, function (file) {
var noStartDir = file.name.replace(startDirRegex, ''),
noGhostDirs = noStartDir;
var noBaseDir = file.name.replace(baseDirRegex, ''),
noGhostDirs = noBaseDir;
_.each(imageFolderRegexes, function (regex) {
noGhostDirs = noGhostDirs.replace(regex, '');
});
file.originalPath = noStartDir;
file.originalPath = noBaseDir;
file.name = noGhostDirs;
file.targetDir = path.join(config.paths.imagesPath, path.dirname(noGhostDirs));
return file;

View File

@ -8,6 +8,7 @@ JSONHandler = {
type: 'data',
extensions: ['.json'],
types: ['application/octet-stream', 'application/json'],
directories: [],
loadFile: function (files, startDir) {
/*jshint unused:false */

View File

@ -8,7 +8,7 @@ DataImporter = {
return importData;
},
doImport: function (importData) {
return importer(importData);
return importer.doImport(importData);
}
};

View File

@ -14,16 +14,24 @@ var _ = require('lodash'),
ImageImporter = require('./importers/image'),
DataImporter = require('./importers/data'),
// Glob levels
ROOT_ONLY = 0,
ROOT_OR_SINGLE_DIR = 1,
ALL_DIRS = 2,
defaults;
defaults = {
extensions: ['.zip'],
types: ['application/zip', 'application/x-zip-compressed']
types: ['application/zip', 'application/x-zip-compressed'],
directories: []
};
function ImportManager() {
this.importers = [ImageImporter, DataImporter];
this.handlers = [ImageHandler, JSONHandler];
// Keep track of files to cleanup at the end
this.filesToDelete = [];
}
/**
@ -36,40 +44,73 @@ function ImportManager() {
_.extend(ImportManager.prototype, {
/**
* Get an array of all the file extensions for which we have handlers
* @returns []
* @returns {string[]}
*/
getExtensions: function () {
return _.flatten(_.union(_.pluck(this.handlers, 'extensions'), defaults.extensions));
},
/**
* Get an array of all the mime types for which we have handlers
* @returns []
* @returns {string[]}
*/
getTypes: function () {
return _.flatten(_.union(_.pluck(this.handlers, 'types'), defaults.types));
},
/**
* Convert the extensions supported by a given handler into a glob string
* @returns String
* Get an array of directories for which we have handlers
* @returns {string[]}
*/
getGlobPattern: function (handler) {
return '**/*+(' + _.reduce(handler.extensions, function (memo, ext) {
getDirectories: function () {
return _.flatten(_.union(_.pluck(this.handlers, 'directories'), defaults.directories));
},
/**
* Convert items into a glob string
* @param {String[]} items
* @returns {String}
*/
getGlobPattern: function (items) {
return '+(' + _.reduce(items, function (memo, ext) {
return memo !== '' ? memo + '|' + ext : ext;
}, '') + ')';
},
/**
* Remove a file after we're done (abstracted into a function for easier testing)
* @param {File} file
* @param {String[]} extensions
* @param {Number} level
* @returns {String}
*/
getExtensionGlob: function (extensions, level) {
var prefix = level === ALL_DIRS ? '**/*' :
(level === ROOT_OR_SINGLE_DIR ? '{*/*,*}' : '*');
return prefix + this.getGlobPattern(extensions);
},
/**
*
* @param {String[]} directories
* @param {Number} level
* @returns {String}
*/
getDirectoryGlob: function (directories, level) {
var prefix = level === ALL_DIRS ? '**/' :
(level === ROOT_OR_SINGLE_DIR ? '{*/,}' : '');
return prefix + this.getGlobPattern(directories);
},
/**
* Remove files after we're done (abstracted into a function for easier testing)
* @returns {Function}
*/
cleanUp: function (file) {
var fileToDelete = file;
cleanUp: function () {
var filesToDelete = this.filesToDelete;
return function (result) {
try {
fs.remove(fileToDelete);
} catch (err) {
errors.logError(err, 'Import could not clean up file', 'You blog will continue to work as expected');
_.each(filesToDelete, function (fileToDelete) {
fs.remove(fileToDelete, function (err) {
if (err) {
errors.logError(err, 'Import could not clean up file ', 'Your blog will continue to work as expected');
}
});
});
return result;
};
},
@ -80,6 +121,39 @@ _.extend(ImportManager.prototype, {
isZip: function (ext) {
return _.contains(defaults.extensions, ext);
},
/**
* Checks the content of a zip folder to see if it is valid.
* Importable content includes any files or directories which the handlers can process
* Importable content must be found either in the root, or inside one base directory
*
* @param {String} directory
* @returns {Promise}
*/
isValidZip: function (directory) {
// Globs match content in the root or inside a single directory
var extMatchesBase = glob.sync(
this.getExtensionGlob(this.getExtensions(), ROOT_OR_SINGLE_DIR), {cwd: directory}
),
extMatchesAll = glob.sync(
this.getExtensionGlob(this.getExtensions(), ALL_DIRS), {cwd: directory}
),
dirMatches = glob.sync(
this.getDirectoryGlob(this.getDirectories(), ROOT_OR_SINGLE_DIR), {cwd: directory}
);
// If this folder contains importable files or a content or images directory
if (extMatchesBase.length > 0 || (dirMatches.length > 0 && extMatchesAll.length > 0)) {
return Promise.resolve(true);
}
if (extMatchesAll.length < 1) {
return Promise.reject(new errors.UnsupportedMediaTypeError(
'Zip did not include any content to import.'
));
}
return Promise.reject(new errors.UnsupportedMediaTypeError('Invalid zip file structure.'));
},
/**
* Use the extract module to extract the given zip file to a temp directory & return the temp directory path
* @param {String} filePath
@ -87,6 +161,7 @@ _.extend(ImportManager.prototype, {
*/
extractZip: function (filePath) {
var tmpDir = path.join(os.tmpdir(), uuid.v4());
this.filesToDelete.push(tmpDir);
return Promise.promisify(extract)(filePath, {dir: tmpDir}).then(function () {
return tmpDir;
});
@ -99,11 +174,37 @@ _.extend(ImportManager.prototype, {
* @returns [] Files
*/
getFilesFromZip: function (handler, directory) {
var globPattern = this.getGlobPattern(handler);
var globPattern = this.getExtensionGlob(handler.extensions, ALL_DIRS);
return _.map(glob.sync(globPattern, {cwd: directory}), function (file) {
return {name: file, path: path.join(directory, file)};
});
},
/**
* Get the name of the single base directory if there is one, else return an empty string
* @param {String} directory
* @returns {Promise (String)}
*/
getBaseDirectory: function (directory) {
// Globs match root level only
var extMatches = glob.sync(this.getExtensionGlob(this.getExtensions(), ROOT_ONLY), {cwd: directory}),
dirMatches = glob.sync(this.getDirectoryGlob(this.getDirectories(), ROOT_ONLY), {cwd: directory}),
extMatchesAll;
// There is no base directory
if (extMatches.length > 0 || dirMatches.length > 0) {
return Promise.resolve();
}
// There is a base directory, grab it from any ext match
extMatchesAll = glob.sync(
this.getExtensionGlob(this.getExtensions(), ALL_DIRS), {cwd: directory}
);
if (extMatchesAll.length < 1 || extMatchesAll[0].split('/') < 1) {
return Promise.resolve(new errors.ValidationError('Invalid zip file: base directory read failed'));
}
return Promise.resolve(extMatchesAll[0].split('/')[0]);
},
/**
* Process Zip
* Takes a reference to a zip file, extracts it, sends any relevant files from inside to the right handler, and
@ -114,19 +215,22 @@ _.extend(ImportManager.prototype, {
* @returns {Promise(ImportData)}
*/
processZip: function (file) {
var self = this;
return this.extractZip(file.path).then(function (directory) {
var self = this,
directory;
return this.extractZip(file.path).then(function (zipDirectory) {
directory = zipDirectory;
return self.isValidZip(directory);
}).then(function () {
return self.getBaseDirectory(directory);
}).then(function (baseDir) {
var ops = [],
importData = {},
startDir = glob.sync(file.name.replace('.zip', ''), {cwd: directory});
startDir = startDir[0] || false;
importData = {};
_.each(self.handlers, function (handler) {
if (importData.hasOwnProperty(handler.type)) {
// This limitation is here to reduce the complexity of the importer for now
return Promise.reject(new errors.UnsupportedMediaTypeError(
'Zip file contains too many types of import data. Please split it up and import separately.'
'Zip file contains multiple data formats. Please split up and import separately.'
));
}
@ -134,7 +238,7 @@ _.extend(ImportManager.prototype, {
if (files.length > 0) {
ops.push(function () {
return handler.loadFile(files, startDir).then(function (data) {
return handler.loadFile(files, baseDir).then(function (data) {
importData[handler.type] = data;
});
});
@ -149,7 +253,7 @@ _.extend(ImportManager.prototype, {
return sequence(ops).then(function () {
return importData;
}).finally(self.cleanUp(directory));
});
});
},
/**
@ -184,6 +288,8 @@ _.extend(ImportManager.prototype, {
var self = this,
ext = path.extname(file.name).toLowerCase();
this.filesToDelete.push(file.path);
return Promise.resolve(this.isZip(ext)).then(function (isZip) {
if (isZip) {
// If it's a zip, process the zip file
@ -192,7 +298,7 @@ _.extend(ImportManager.prototype, {
// Else process the file
return self.processFile(file, ext);
}
}).finally(self.cleanUp(file.path));
});
},
/**
* Import Step 2:
@ -245,7 +351,7 @@ _.extend(ImportManager.prototype, {
* Import From File
* The main method of the ImportManager, call this to kick everything off!
* @param {File} file
* @returns {*}
* @returns {Promise}
*/
importFromFile: function (file) {
var self = this;
@ -259,8 +365,10 @@ _.extend(ImportManager.prototype, {
// @TODO: It would be cool to have some sort of dry run flag here
return self.doImport(importData);
}).then(function (importData) {
// Step 4: Finally, report on the import
return self.generateReport(importData);
// Step 4: Report on the import
return self.generateReport(importData)
// Step 5: Cleanup any files
.finally(self.cleanUp());
});
}
});

View File

@ -47,7 +47,7 @@ describe('Import', function () {
}),
fakeData = {test: true};
importer(fakeData).then(function () {
importer.doImport(fakeData).then(function () {
importStub.calledWith(fakeData).should.equal(true);
importStub.restore();
@ -68,7 +68,7 @@ describe('Import', function () {
testUtils.fixtures.loadExportFixture('export-003').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function (importResult) {
should.exist(importResult);
should.exist(importResult.data);
@ -83,7 +83,7 @@ describe('Import', function () {
testUtils.fixtures.loadExportFixture('export-003-duplicate-posts').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function (importResult) {
should.exist(importResult.data.data.posts);
@ -100,7 +100,7 @@ describe('Import', function () {
testUtils.fixtures.loadExportFixture('export-003-duplicate-tags').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function (importResult) {
should.exist(importResult.data.data.tags);
should.exist(importResult.data.data.posts_tags);
@ -135,7 +135,7 @@ describe('Import', function () {
testUtils.fixtures.loadExportFixture('export-000').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
// Grab the data from tables
return Promise.all([
@ -182,7 +182,7 @@ describe('Import', function () {
exportData.data.posts[0].updated_at = timestamp;
exportData.data.posts[0].published_at = timestamp;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
// Grab the data from tables
return Promise.all([
@ -252,7 +252,7 @@ describe('Import', function () {
// change title to 151 characters
exportData.data.posts[0].title = new Array(152).join('a');
exportData.data.posts[0].tags = 'Tag';
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
(1).should.eql(0, 'Data import should not resolve promise.');
}, function (error) {
@ -297,7 +297,7 @@ describe('Import', function () {
exportData = exported;
// change to blank settings key
exportData.data.settings[3].key = null;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
(1).should.eql(0, 'Data import should not resolve promise.');
}, function (error) {
@ -354,7 +354,7 @@ describe('Import', function () {
exportData.data.posts[0].updated_at = timestamp;
exportData.data.posts[0].published_at = timestamp;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
// Grab the data from tables
return Promise.all([
@ -424,7 +424,7 @@ describe('Import', function () {
// change title to 151 characters
exportData.data.posts[0].title = new Array(152).join('a');
exportData.data.posts[0].tags = 'Tag';
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
(1).should.eql(0, 'Data import should not resolve promise.');
}, function (error) {
@ -468,7 +468,7 @@ describe('Import', function () {
exportData = exported;
// change to blank settings key
exportData.data.settings[3].key = null;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
(1).should.eql(0, 'Data import should not resolve promise.');
}, function (error) {
@ -517,7 +517,7 @@ describe('Import', function () {
testUtils.fixtures.loadExportFixture('export-003').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
// Grab the data from tables
return Promise.all([
@ -562,7 +562,7 @@ describe('Import', function () {
testUtils.fixtures.loadExportFixture('export-003-badValidation').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done(new Error('Allowed import of duplicate data'));
}).catch(function (response) {
@ -585,7 +585,7 @@ describe('Import', function () {
var exportData;
testUtils.fixtures.loadExportFixture('export-003-dbErrors').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done(new Error('Allowed import of duplicate data'));
}).catch(function (response) {
@ -601,7 +601,7 @@ describe('Import', function () {
testUtils.fixtures.loadExportFixture('export-003-mu-unknownAuthor').then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done(new Error('Allowed import of unknown author'));
}).catch(function (response) {
@ -622,7 +622,7 @@ describe('Import', function () {
exportData.data.tags.length.should.be.above(1);
exportData.data.posts_tags.length.should.be.above(1);
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done(new Error('Allowed import of invalid tags data'));
}).catch(function (response) {
@ -643,7 +643,7 @@ describe('Import', function () {
exportData.data.posts.length.should.be.above(1);
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done(new Error('Allowed import of invalid tags data'));
}).catch(function (response) {
@ -660,7 +660,7 @@ describe('Import', function () {
exportData.data.posts.length.should.be.above(0);
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
// Grab the data from tables
return knex('posts').select();
@ -693,7 +693,7 @@ describe('Import (new test structure)', function () {
return testUtils.fixtures.loadExportFixture('export-003-mu');
}).then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done();
}).catch(done);
@ -920,7 +920,7 @@ describe('Import (new test structure)', function () {
return testUtils.fixtures.loadExportFixture('export-003-mu-noOwner');
}).then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done();
}).catch(done);
@ -1148,7 +1148,7 @@ describe('Import (new test structure)', function () {
return testUtils.fixtures.loadExportFixture('export-003-mu');
}).then(function (exported) {
exportData = exported;
return importer(exportData);
return importer.doImport(exportData);
}).then(function () {
done();
}).catch(done);

View File

@ -6,6 +6,7 @@ var should = require('should'),
_ = require('lodash'),
testUtils = require('../utils'),
config = require('../../server/config'),
path = require('path'),
// Stuff we are testing
ImportManager = require('../../server/data/importer'),
@ -14,8 +15,8 @@ var should = require('should'),
DataImporter = require('../../server/data/importer/importers/data'),
ImageImporter = require('../../server/data/importer/importers/image'),
sandbox = sinon.sandbox.create(),
storage = require('../../server/storage');
storage = require('../../server/storage'),
sandbox = sinon.sandbox.create();
// To stop jshint complaining
should.equal(true, true);
@ -50,8 +51,25 @@ describe('Importer', function () {
ImportManager.getTypes().should.containEql('application/x-zip-compressed');
});
it('gets the correct directories', function () {
ImportManager.getDirectories().should.be.instanceof(Array).and.have.lengthOf(2);
ImportManager.getDirectories().should.containEql('images');
ImportManager.getDirectories().should.containEql('content');
});
it('globs extensions correctly', function () {
ImportManager.getGlobPattern(JSONHandler).should.equal('**/*+(.json)');
ImportManager.getGlobPattern(ImportManager.getExtensions()).should.equal('+(.jpg|.jpeg|.gif|.png|.svg|.svgz|.json|.zip)');
ImportManager.getGlobPattern(ImportManager.getDirectories()).should.equal('+(images|content)');
ImportManager.getGlobPattern(JSONHandler.extensions).should.equal('+(.json)');
ImportManager.getGlobPattern(ImageHandler.extensions).should.equal('+(.jpg|.jpeg|.gif|.png|.svg|.svgz)');
ImportManager.getExtensionGlob(ImportManager.getExtensions()).should.equal('*+(.jpg|.jpeg|.gif|.png|.svg|.svgz|.json|.zip)');
ImportManager.getDirectoryGlob(ImportManager.getDirectories()).should.equal('+(images|content)');
ImportManager.getExtensionGlob(ImportManager.getExtensions(), 0).should.equal('*+(.jpg|.jpeg|.gif|.png|.svg|.svgz|.json|.zip)');
ImportManager.getDirectoryGlob(ImportManager.getDirectories(), 0).should.equal('+(images|content)');
ImportManager.getExtensionGlob(ImportManager.getExtensions(), 1).should.equal('{*/*,*}+(.jpg|.jpeg|.gif|.png|.svg|.svgz|.json|.zip)');
ImportManager.getDirectoryGlob(ImportManager.getDirectories(), 1).should.equal('{*/,}+(images|content)');
ImportManager.getExtensionGlob(ImportManager.getExtensions(), 2).should.equal('**/*+(.jpg|.jpeg|.gif|.png|.svg|.svgz|.json|.zip)');
ImportManager.getDirectoryGlob(ImportManager.getDirectories(), 2).should.equal('**/+(images|content)');
});
// Step 1 of importing is loadFile
@ -59,13 +77,11 @@ describe('Importer', function () {
it('knows when to process a file', function (done) {
var testFile = {name: 'myFile.json', path: '/my/path/myFile.json'},
zipSpy = sandbox.stub(ImportManager, 'processZip').returns(Promise.resolve()),
fileSpy = sandbox.stub(ImportManager, 'processFile').returns(Promise.resolve()),
cleanSpy = sandbox.stub(ImportManager, 'cleanUp').returns(Promise.resolve());
fileSpy = sandbox.stub(ImportManager, 'processFile').returns(Promise.resolve());
ImportManager.loadFile(testFile).then(function () {
zipSpy.calledOnce.should.be.false;
fileSpy.calledOnce.should.be.true;
cleanSpy.calledOnce.should.be.true;
done();
});
});
@ -74,13 +90,11 @@ describe('Importer', function () {
it('knows when to process a zip', function (done) {
var testZip = {name: 'myFile.zip', path: '/my/path/myFile.zip'},
zipSpy = sandbox.stub(ImportManager, 'processZip').returns(Promise.resolve()),
fileSpy = sandbox.stub(ImportManager, 'processFile').returns(Promise.resolve()),
cleanSpy = sandbox.stub(ImportManager, 'cleanUp').returns(Promise.resolve());
fileSpy = sandbox.stub(ImportManager, 'processFile').returns(Promise.resolve());
ImportManager.loadFile(testZip).then(function () {
zipSpy.calledOnce.should.be.true;
fileSpy.calledOnce.should.be.false;
cleanSpy.calledOnce.should.be.true;
done();
});
});
@ -90,20 +104,20 @@ describe('Importer', function () {
testZip = {name: 'myFile.zip', path: '/my/path/myFile.zip'},
// need to stub out the extract and glob function for zip
extractSpy = sandbox.stub(ImportManager, 'extractZip').returns(Promise.resolve('/tmp/dir/')),
validSpy = sandbox.stub(ImportManager, 'isValidZip').returns(Promise.resolve()),
getFileSpy = sandbox.stub(ImportManager, 'getFilesFromZip'),
jsonSpy = sandbox.stub(JSONHandler, 'loadFile').returns(Promise.resolve({posts: []})),
imageSpy = sandbox.stub(ImageHandler, 'loadFile'),
cleanSpy = sandbox.stub(ImportManager, 'cleanUp').returns(Promise.resolve());
imageSpy = sandbox.stub(ImageHandler, 'loadFile');
getFileSpy.withArgs(JSONHandler).returns(['/tmp/dir/myFile.json']);
getFileSpy.withArgs(ImageHandler).returns([]);
ImportManager.processZip(testZip).then(function (zipResult) {
extractSpy.calledOnce.should.be.true;
validSpy.calledOnce.should.be.true;
getFileSpy.calledTwice.should.be.true;
jsonSpy.calledOnce.should.be.true;
imageSpy.called.should.be.false;
cleanSpy.calledOnce.should.be.true;
ImportManager.processFile(testFile, '.json').then(function (fileResult) {
jsonSpy.calledTwice.should.be.true;
@ -116,6 +130,72 @@ describe('Importer', function () {
});
});
});
describe('Validate Zip', function () {
it('accepts a zip with a base directory', function (done) {
var testDir = path.resolve('core/test/utils/fixtures/import/zips/zip-with-base-dir');
ImportManager.isValidZip(testDir).then(function (isValid) {
isValid.should.be.ok;
done();
});
});
it('accepts a zip without a base directory', function (done) {
var testDir = path.resolve('core/test/utils/fixtures/import/zips/zip-without-base-dir');
ImportManager.isValidZip(testDir).then(function (isValid) {
isValid.should.be.ok;
done();
});
});
it('accepts a zip with an image directory', function (done) {
var testDir = path.resolve('core/test/utils/fixtures/import/zips/zip-image-dir');
ImportManager.isValidZip(testDir).then(function (isValid) {
isValid.should.be.ok;
done();
});
});
it('fails a zip with two base directories', function (done) {
var testDir = path.resolve('core/test/utils/fixtures/import/zips/zip-with-double-base-dir');
ImportManager.isValidZip(testDir).then(function () {
done(new Error('Double base directory did not throw error'));
}).catch(function (err) {
err.message.should.equal('Invalid zip file structure.');
err.type.should.equal('UnsupportedMediaTypeError');
done();
});
});
it('fails a zip with no content', function (done) {
var testDir = path.resolve('core/test/utils/fixtures/import/zips/zip-invalid');
ImportManager.isValidZip(testDir).then(function () {
done(new Error('Double base directory did not throw error'));
}).catch(function (err) {
err.message.should.equal('Zip did not include any content to import.');
err.type.should.equal('UnsupportedMediaTypeError');
done();
});
});
});
describe('Get Base Dir', function () {
it('returns string for base directory', function (done) {
var testDir = path.resolve('core/test/utils/fixtures/import/zips/zip-with-base-dir');
ImportManager.getBaseDirectory(testDir).then(function (baseDir) {
baseDir.should.equal('basedir');
done();
});
});
it('returns empty for no base directory', function (done) {
var testDir = path.resolve('core/test/utils/fixtures/import/zips/zip-without-base-dir');
ImportManager.getBaseDirectory(testDir).then(function (baseDir) {
should.not.exist(baseDir);
done();
});
});
});
});
// Step 2 of importing is preProcess
@ -125,15 +205,19 @@ describe('Importer', function () {
var input = {data: {}, images: []},
// pass a copy so that input doesn't get modified
inputCopy = _.cloneDeep(input),
dataSpy = sandbox.spy(DataImporter, 'preProcess');
dataSpy = sandbox.spy(DataImporter, 'preProcess'),
imageSpy = sandbox.spy(ImageImporter, 'preProcess');
ImportManager.preProcess(inputCopy).then(function (output) {
dataSpy.calledOnce.should.be.true;
dataSpy.calledWith(inputCopy).should.be.true;
imageSpy.calledOnce.should.be.true;
imageSpy.calledWith(inputCopy).should.be.true;
// eql checks for equality
// equal checks the references are for the same object
output.should.not.equal(input);
output.should.have.property('preProcessedByData', true);
output.should.have.property('preProcessedByImage', true);
done();
});
});
@ -185,6 +269,27 @@ describe('Importer', function () {
});
});
});
describe('importFromFile', function () {
it('does the import steps in order', function (done) {
var loadFileSpy = sandbox.stub(ImportManager, 'loadFile').returns(Promise.resolve()),
preProcessSpy = sandbox.stub(ImportManager, 'preProcess').returns(Promise.resolve()),
doImportSpy = sandbox.stub(ImportManager, 'doImport').returns(Promise.resolve()),
generateReportSpy = sandbox.stub(ImportManager, 'generateReport').returns(Promise.resolve()),
cleanupSpy = sandbox.stub(ImportManager, 'cleanUp').returns({});
ImportManager.importFromFile({}).then(function () {
loadFileSpy.calledOnce.should.be.true;
preProcessSpy.calledOnce.should.be.true;
doImportSpy.calledOnce.should.be.true;
generateReportSpy.calledOnce.should.be.true;
cleanupSpy.calledOnce.should.be.true;
sinon.assert.callOrder(loadFileSpy, preProcessSpy, doImportSpy, generateReportSpy, cleanupSpy);
done();
});
});
});
});
describe('JSONHandler', function () {
@ -227,7 +332,6 @@ describe('Importer', function () {
describe('ImageHandler', function () {
var origConfig = _.cloneDeep(config),
storage = require('../../server/storage'),
store = storage.getStorage();
afterEach(function () {
@ -375,11 +479,33 @@ describe('Importer', function () {
});
describe('DataImporter', function () {
var importer = require('../../server/data/import');
it('has the correct interface', function () {
DataImporter.type.should.eql('data');
DataImporter.preProcess.should.be.instanceof(Function);
DataImporter.doImport.should.be.instanceof(Function);
});
it('does preprocess posts, users and tags correctly', function () {
var inputData = require('../utils/fixtures/import/import-data-1.json'),
outputData = DataImporter.preProcess(_.cloneDeep(inputData));
// Data preprocess is a noop
inputData.data.data.posts[0].should.eql(outputData.data.data.posts[0]);
inputData.data.data.tags[0].should.eql(outputData.data.data.tags[0]);
inputData.data.data.users[0].should.eql(outputData.data.data.users[0]);
});
it('does import the data correctly', function () {
var inputData = require('../utils/fixtures/import/import-data-1.json'),
importerSpy = sandbox.stub(importer, 'doImport').returns(Promise.resolve());
DataImporter.doImport(inputData.data).then(function () {
importerSpy.calledOnce.should.be.true;
importerSpy.calledWith(inputData.data).should.be.true;
});
});
});
describe('ImageImporter', function () {
@ -389,22 +515,33 @@ describe('Importer', function () {
ImageImporter.doImport.should.be.instanceof(Function);
});
it('does preprocess posts correctly', function () {
it('does preprocess posts, users and tags correctly', function () {
var inputData = require('../utils/fixtures/import/import-data-1.json'),
outputData = ImageImporter.preProcess(_.cloneDeep(inputData));
inputData.data.data.posts[0].markdown.should.not.containEql('/content/images/my-image.png');
inputData.data.data.posts[0].html.should.not.containEql('/content/images/my-image.png');
outputData.data.data.posts[0].markdown.should.containEql('/content/images/my-image.png');
outputData.data.data.posts[0].html.should.containEql('/content/images/my-image.png');
inputData = inputData.data.data;
outputData = outputData.data.data;
inputData.data.data.posts[0].markdown.should.not.containEql('/content/images/photos/cat.jpg');
inputData.data.data.posts[0].html.should.not.containEql('/content/images/photos/cat.jpg');
outputData.data.data.posts[0].markdown.should.containEql('/content/images/photos/cat.jpg');
outputData.data.data.posts[0].html.should.containEql('/content/images/photos/cat.jpg');
inputData.posts[0].markdown.should.not.containEql('/content/images/my-image.png');
inputData.posts[0].html.should.not.containEql('/content/images/my-image.png');
outputData.posts[0].markdown.should.containEql('/content/images/my-image.png');
outputData.posts[0].html.should.containEql('/content/images/my-image.png');
inputData.data.data.posts[0].image.should.eql('/images/my-image.png');
outputData.data.data.posts[0].image.should.eql('/content/images/my-image.png');
inputData.posts[0].markdown.should.not.containEql('/content/images/photos/cat.jpg');
inputData.posts[0].html.should.not.containEql('/content/images/photos/cat.jpg');
outputData.posts[0].markdown.should.containEql('/content/images/photos/cat.jpg');
outputData.posts[0].html.should.containEql('/content/images/photos/cat.jpg');
inputData.posts[0].image.should.eql('/images/my-image.png');
outputData.posts[0].image.should.eql('/content/images/my-image.png');
inputData.tags[0].image.should.eql('/images/my-image.png');
outputData.tags[0].image.should.eql('/content/images/my-image.png');
inputData.users[0].image.should.eql('/images/my-image.png');
inputData.users[0].cover.should.eql('/images/photos/cat.jpg');
outputData.users[0].image.should.eql('/content/images/my-image.png');
outputData.users[0].cover.should.eql('/content/images/photos/cat.jpg');
});
it('does import the images correctly', function () {

View File

@ -52,6 +52,7 @@
"name": "Getting Started",
"slug": "getting-started",
"description": null,
"image": "/images/my-image.png",
"parent_id": null,
"meta_title": null,
"meta_description": null,
@ -61,6 +62,14 @@
"updated_by": 1
}
],
"users": [
{
"name": "test user",
"email": "test@ghost.org",
"image": "/images/my-image.png",
"cover": "/images/photos/cat.jpg"
}
],
"posts_tags": [
{
"id": 1,