Files
bower/lib/core/resolvers/Resolver.js

219 lines
5.9 KiB
JavaScript

var fs = require('fs');
var path = require('path');
var Q = require('q');
var tmp = require('tmp');
var mkdirp = require('mkdirp');
var bowerJson = require('bower-json');
var defaultConfig = require('../../config');
var createError = require('../../util/createError');
var removeIgnores = require('../../util/removeIgnores');
tmp.setGracefulCleanup();
var Resolver = function (source, options) {
options = options || {};
this._source = source;
this._target = options.target || '*';
this._name = options.name || path.basename(this._source);
this._guessedName = !options.name;
this._config = options.config || defaultConfig;
};
// -----------------
Resolver.prototype.getSource = function () {
return this._source;
};
Resolver.prototype.getName = function () {
return this._name;
};
Resolver.prototype.getTarget = function () {
return this._target;
};
Resolver.prototype.getTempDir = function () {
return this._tempDir;
};
Resolver.prototype.hasNew = function (canonicalPkg) {
var that = this;
var promise;
var metaFile;
// TODO: Change arguments to canonicalPkg, pkgMeta
// where pkgMeta is optional
// Change _hasNew to the same
// If already working, error out
if (this._working) {
return Q.reject(createError('Already working', 'EWORKING'));
}
this._working = true;
// Avoid reading the package meta if _hasNew was not rewritten
if (this._hasNew === Resolver.prototype._hasNew) {
promise = this._hasNew();
// Otherwise call _hasNew with both the package meta and the canonical package
} else {
metaFile = path.join(canonicalPkg, '.bower.json');
promise = Q.nfcall(fs.readFile, metaFile)
.then(function (contents) {
var pkgMeta = JSON.parse(contents.toString());
return that._hasNew(pkgMeta, canonicalPkg);
}, function () {
return true; // Simply resolve to true if there was an error reading the meta
});
}
return promise.fin(function () {
that._working = false;
});
};
Resolver.prototype.resolve = function () {
var that = this;
// If already working, error out
if (this._working) {
return Q.reject(createError('Already working', 'EWORKING'));
}
this._working = true;
// Create temporary dir
return this._createTempDir()
// Resolve self
.then(this._resolve.bind(this))
// Read json, generating the package meta
.then(function () {
return that._readJson(that._tempDir);
})
.then(function (meta) {
return Q.all([
// Apply package meta
that._applyPkgMeta(meta),
// Save package meta
that._savePkgMeta(meta)
]);
})
.then(function () {
// Resolve with the folder
return that._tempDir;
}, function (err) {
// If something went wrong, unset the temporary dir
that._tempDir = null;
throw err;
})
.fin(function () {
that._working = false;
});
};
Resolver.prototype.getPkgMeta = function () {
return this._pkgMeta;
};
// -----------------
Resolver.clearRuntimeCache = function () {};
// -----------------
// Abstract function that should be implemented by concrete resolvers
Resolver.prototype._resolve = function () {
throw new Error('_resolve not implemented');
};
// -----------------
Resolver.prototype._hasNew = function (pkgMeta) {
return Q.resolve(true);
};
Resolver.prototype._createTempDir = function () {
var baseDir = path.join(this._config.tmp, 'bower');
return Q.nfcall(mkdirp, baseDir)
.then(function () {
return Q.nfcall(tmp.dir, {
template: path.join(baseDir, this._name + '-XXXXXX'),
mode: 0777 & ~process.umask(),
unsafeCleanup: true
});
}.bind(this))
.then(function (dir) {
this._tempDir = dir;
return dir;
}.bind(this));
};
Resolver.prototype._readJson = function (dir) {
var deferred = Q.defer();
Q.nfcall(bowerJson.find, dir)
.then(function (filename) {
// If it is a component.json, warn about the deprecation
if (path.basename(filename) === 'component.json') {
deferred.notify({
type: 'warn',
data: 'Package "' + this._name + '" is using the deprecated component.json file'
});
}
// Read it
return Q.nfcall(bowerJson.read, filename)
.fail(function (err) {
throw createError('Something went wrong while reading "' + filename + '"', err.code, {
details: err.message
});
});
}.bind(this), function () {
// No json file was found, assume one
return Q.nfcall(bowerJson.parse, { name: this._name });
}.bind(this))
.then(deferred.resolve, deferred.reject, deferred.notify);
return deferred.promise;
};
Resolver.prototype._applyPkgMeta = function (meta) {
// Check if name defined in the json is different
// If so and if the name was "guessed", assume the json name
if (meta.name !== this._name && this._guessedName) {
this._name = meta.name;
}
// Handle ignore property, deleting all files from the temporary directory
// If no ignores were specified, simply resolve
if (!meta.ignore || !meta.ignore.length) {
return Q.resolve(meta);
}
// Otherwise remove them from the temp dir
return removeIgnores(this._tempDir, meta.ignore)
.then(function () {
return meta;
});
};
Resolver.prototype._savePkgMeta = function (meta) {
var contents;
// Store original source
meta._source = this._source;
// Stringify contents
contents = JSON.stringify(meta, null, 2);
return Q.nfcall(fs.writeFile, path.join(this._tempDir, '.bower.json'), contents)
.then(function () {
return this._pkgMeta = meta;
}.bind(this));
};
module.exports = Resolver;