knex/lib/seed/Seeder.js

191 lines
5.2 KiB
JavaScript
Raw Normal View History

2016-03-02 17:07:05 +01:00
// Seeder
// -------
const fs = require('fs');
const path = require('path');
const mkdirp = require('mkdirp');
const Bluebird = require('bluebird');
const {
filter,
includes,
map,
bind,
template,
each,
extend,
} = require('lodash');
// The new seeds we're performing, typically called from the `knex.seed`
2016-03-02 17:07:05 +01:00
// interface on the main `knex` object. Passes the `knex` instance performing
// the seeds.
function Seeder(knex) {
this.knex = knex;
2016-03-02 17:07:05 +01:00
this.config = this.setConfig(knex.client.config.seeds);
}
// Runs seed files for the given environment.
Seeder.prototype.run = async function(config) {
2016-03-02 17:07:05 +01:00
this.config = this.setConfig(config);
return this._seedData()
.bind(this)
.then(([all]) => {
const files =
config && config.specific
? all.filter((file) => file === config.specific)
: all;
return this._runSeeds(files);
2016-03-02 17:07:05 +01:00
});
};
2016-03-02 17:07:05 +01:00
// Creates a new seed file, with a given name.
Seeder.prototype.make = function(name, config) {
this.config = this.setConfig(config);
if (!name)
Bluebird.rejected(
new Error('A name must be specified for the generated seed')
);
2016-03-02 17:07:05 +01:00
return this._ensureFolder(config)
.bind(this)
.then(this._generateStubTemplate)
.then(this._writeNewSeed(name));
};
// Lists all available seed files as a sorted array.
Seeder.prototype._listAll = async function(config) {
2016-03-02 17:07:05 +01:00
this.config = this.setConfig(config);
const loadExtensions = this.config.loadExtensions;
return Bluebird.promisify(fs.readdir, { context: fs })(
this._absoluteConfigDir()
)
2016-03-02 17:07:05 +01:00
.bind(this)
.then((seeds) =>
filter(seeds, function(value) {
const extension = path.extname(value);
return includes(loadExtensions, extension);
}).sort()
);
};
2016-03-02 17:07:05 +01:00
// Gets the seed file list from the specified seed directory.
2016-03-02 17:07:05 +01:00
Seeder.prototype._seedData = function() {
return Bluebird.join(this._listAll());
2016-03-02 17:07:05 +01:00
};
// Ensures a folder for the seeds exist, dependent on the
// seed config settings.
Seeder.prototype._ensureFolder = function() {
const dir = this._absoluteConfigDir();
return Bluebird.promisify(fs.stat, { context: fs })(dir).catch(() =>
Bluebird.promisify(mkdirp)(dir)
);
2016-03-02 17:07:05 +01:00
};
// Run seed files, in sequence.
Seeder.prototype._runSeeds = function(seeds) {
return Bluebird.all(map(seeds, bind(this._validateSeedStructure, this)))
2016-03-02 17:07:05 +01:00
.bind(this)
.then(function(seeds) {
return Bluebird.bind(this).then(function() {
return this._waterfallBatch(seeds);
});
2016-03-02 17:07:05 +01:00
});
};
// Validates seed files by requiring and checking for a `seed` function.
Seeder.prototype._validateSeedStructure = function(name) {
const seed = require(path.join(this._absoluteConfigDir(), name));
2016-03-02 17:07:05 +01:00
if (typeof seed.seed !== 'function') {
throw new Error(`Invalid seed file: ${name} must have a seed function`);
2016-03-02 17:07:05 +01:00
}
return name;
};
// Generates the stub template for the current seed file, returning a compiled template.
Seeder.prototype._generateStubTemplate = function() {
const stubPath =
this.config.stub ||
path.join(__dirname, 'stub', this.config.extension + '.stub');
return Bluebird.promisify(fs.readFile, { context: fs })(stubPath).then(
(stub) => template(stub.toString(), { variable: 'd' })
);
2016-03-02 17:07:05 +01:00
};
// Write a new seed to disk, using the config and generated filename,
// passing any `variables` given in the config to the template.
Seeder.prototype._writeNewSeed = function(name) {
const { config } = this;
const dir = this._absoluteConfigDir();
2016-03-02 17:07:05 +01:00
return function(tmpl) {
if (name[0] === '-') name = name.slice(1);
const filename = name + '.' + config.extension;
return Bluebird.promisify(fs.writeFile, { context: fs })(
2016-03-02 17:07:05 +01:00
path.join(dir, filename),
tmpl(config.variables || {})
).return(path.join(dir, filename));
};
};
// Runs a batch of seed files.
Seeder.prototype._waterfallBatch = function(seeds) {
const { knex } = this;
const seedDirectory = this._absoluteConfigDir();
let current = Bluebird.bind({ failed: false, failedOn: 0 });
const log = [];
each(seeds, (seed) => {
const name = path.join(seedDirectory, seed);
2016-03-02 17:07:05 +01:00
seed = require(name);
// Run each seed file.
current = current.then(() =>
// Nesting promise to prevent bubbling up of error on catch
Promise.resolve()
.then(() => seed.seed(knex))
.then(() => log.push(name))
.catch((originalError) => {
const error = new Error(
`Error while executing "${name}" seed: ${originalError.message}`
);
error.original = originalError;
error.stack =
error.stack
.split('\n')
.slice(0, 2)
.join('\n') +
'\n' +
originalError.stack;
throw error;
})
);
2016-03-02 17:07:05 +01:00
});
return current.then(() => [log]);
2016-03-02 17:07:05 +01:00
};
Seeder.prototype._absoluteConfigDir = function() {
return path.resolve(process.cwd(), this.config.directory);
};
Seeder.prototype.setConfig = function(config) {
return extend(
{
extension: 'js',
directory: './seeds',
loadExtensions: [
'.co',
'.coffee',
'.eg',
'.iced',
'.js',
'.litcoffee',
'.ls',
'.ts',
],
},
this.config || {},
config
);
2016-03-02 17:07:05 +01:00
};
module.exports = Seeder;