Files
bower/lib/core/PackageRepository.js
2013-05-13 16:07:46 +01:00

116 lines
4.2 KiB
JavaScript

var mout = require('mout');
var Q = require('q');
var RegistryClient = require('bower-registry-client');
var ResolveCache = require('./ResolveCache');
var resolverFactory = require('./resolverFactory');
var defaultConfig = require('../config');
var createError = require('../util/createError');
var PackageRepository = function (options) {
options = options || {};
this._options = options;
this._config = options.config || defaultConfig;
// Instantiate the registry and store it in the options object
// because it will be passed to the resolver factory
this._options.registry = new RegistryClient(mout.object.fillIn({
cache: this._config._registry
}, options.config));
this._cache = new ResolveCache(this._config._cache);
};
// -----------------
PackageRepository.prototype.fetch = function (decEndpoint) {
var resolver;
var deferred = Q.defer();
var that = this;
// Get the appropriate resolver
resolverFactory(decEndpoint, this._options)
// Retrieve from the resolve cache
.then(function (res) {
resolver = res;
// Set the resolver name in the decEndpoint
decEndpoint.resolverName = res.getName();
// If force flag is used, bypass cache
if (that._options.force) {
deferred.notify({ type: 'action', data: 'Resolving' });
return that._resolve(resolver);
}
// Note that we use the resolver methods to query the
// cache because transformations/normalisations can occur
return that._cache.retrieve(resolver.getSource(), resolver.getTarget())
// Decide if we can use the one from the resolve cache
.spread(function (canonicalPkg, pkgMeta) {
// If there's no package in the cache
if (!canonicalPkg) {
// And the offline flag is passed, error out
if (that._options.offline) {
throw createError('No cached version for ' + resolver.getTarget(), 'ENOCACHE');
}
// Otherwise, we have to resolve it
deferred.notify({ type: 'action', data: 'No cached version, resolving..' });
return that._resolve(resolver);
}
// If offline flag is used, use directly the cached one
if (that._options.offline) {
deferred.notify({ type: 'action', data: 'Got cached version' });
return [canonicalPkg, pkgMeta];
}
process.nextTick(function () {
deferred.notify({ type: 'action', data: 'Got cached version, validating..' });
});
// Otherwise check for new contents
return resolver.hasNew(canonicalPkg, pkgMeta)
.then(function (hasNew) {
// If there are no new contents, resolve to
// the cached one
if (!hasNew) {
return [canonicalPkg, pkgMeta];
}
// Otherwise resolve to the newest one
deferred.notify({ type: 'action', data: 'There\'s a new version, resolving..' });
return that._resolve(resolver);
});
});
})
.then(deferred.resolve, deferred.reject, deferred.notify);
return deferred.promise;
};
PackageRepository.prototype.empty = function (name) {
// TODO Think of a way to remove specific packages of a given name from the cache
// Since the ResolveCache.empty only works with source, one possible solution is to implement
// a forEach method that calls a function with the canonicalPackage and the pkgMeta
// so that we can match against the pkgMeta.name and call ResolveCache.empty with it
};
// ---------------------
PackageRepository.prototype._resolve = function (resolver) {
// Resolve the resolver
return resolver.resolve()
// Store in the cache
.then(function (canonicalPkg) {
return this._cache.store(canonicalPkg, resolver.getPkgMeta());
}.bind(this))
// Resolve promise with canonical package and package meta
.then(function (dir) {
return [dir, resolver.getPkgMeta()];
}.bind(this));
};
module.exports = PackageRepository;