greenlock.js/lib/core.js

533 lines
21 KiB
JavaScript
Raw Normal View History

2015-12-15 03:37:39 -08:00
'use strict';
2018-06-29 02:51:35 -06:00
var PromiseA;
try {
PromiseA = require('bluebird');
} catch(e) {
PromiseA = global.Promise;
}
2018-07-04 02:13:11 -06:00
var util = require('util');
function promisifyAll(obj) {
var aobj = {};
Object.keys(obj).forEach(function (key) {
2018-07-04 02:17:18 -06:00
if ('function' === typeof obj[key]) {
2018-07-04 02:21:07 -06:00
aobj[key] = obj[key];
2018-07-04 02:17:18 -06:00
aobj[key + 'Async'] = util.promisify(obj[key]);
}
2018-07-04 02:13:11 -06:00
});
return aobj;
}
2018-06-29 02:51:35 -06:00
function _log(debug) {
2016-08-09 15:02:10 -04:00
if (debug) {
var args = Array.prototype.slice.call(arguments);
args.shift();
2018-05-15 16:01:09 -06:00
args.unshift("[greenlock/lib/core.js]");
2016-08-09 15:02:10 -04:00
console.log.apply(console, args);
}
}
2018-05-15 16:01:09 -06:00
module.exports.create = function (gl) {
2016-08-08 19:14:53 -04:00
var utils = require('./utils');
2018-07-04 02:13:11 -06:00
var RSA = promisifyAll(require('rsa-compat').RSA);
2018-05-15 16:01:09 -06:00
var log = gl.log || _log; // allow custom log
var pendingRegistrations = {};
2016-08-04 18:49:35 -04:00
2016-08-06 02:05:04 -04:00
var core = {
//
// Helpers
//
getAcmeUrlsAsync: function (args) {
var now = Date.now();
// TODO check response header on request for cache time
2018-05-15 16:01:09 -06:00
if ((now - gl._ipc.acmeUrlsUpdatedAt) < 10 * 60 * 1000) {
return PromiseA.resolve(gl._ipc.acmeUrls);
2016-08-06 02:05:04 -04:00
}
2015-12-15 03:37:39 -08:00
// TODO acme-v2/nocompat
2018-05-15 16:01:09 -06:00
return gl.acme.getAcmeUrlsAsync(args.server).then(function (data) {
gl._ipc.acmeUrlsUpdatedAt = Date.now();
gl._ipc.acmeUrls = data;
2015-12-15 03:37:39 -08:00
2018-05-15 16:01:09 -06:00
return gl._ipc.acmeUrls;
2016-08-06 02:05:04 -04:00
});
}
2015-12-20 05:13:41 +00:00
2016-08-06 02:05:04 -04:00
//
// The Main Enchilada
//
//
// Accounts
//
, accounts: {
2016-08-08 18:11:25 -04:00
// Accounts
2016-08-06 02:05:04 -04:00
registerAsync: function (args) {
2016-08-08 15:17:09 -04:00
var err;
2018-05-15 16:01:09 -06:00
var copy = utils.merge(args, gl);
2016-08-08 19:14:53 -04:00
var disagreeTos;
2016-08-08 18:11:25 -04:00
args = utils.tplCopy(copy);
if (!args.account) { args.account = {}; }
if ('object' === typeof args.account && !args.account.id) { args.account.id = args.accountId || args.email || ''; }
2016-08-08 15:17:09 -04:00
2016-08-08 19:14:53 -04:00
disagreeTos = (!args.agreeTos && 'undefined' !== typeof args.agreeTos);
if (!args.email || disagreeTos || (parseInt(args.rsaKeySize, 10) < 2048)) {
2016-08-08 15:17:09 -04:00
err = new Error(
"In order to register an account both 'email' and 'agreeTos' must be present"
+ " and 'rsaKeySize' must be 2048 or greater."
);
err.code = 'E_ARGS';
return PromiseA.reject(err);
}
return utils.testEmail(args.email).then(function () {
if (args.account && args.account.privkey && (args.account.privkey.jwk || args.account.privkey.pem)) {
// TODO import jwk or pem and return it here
console.warn("TODO: implement accounts.checkKeypairAsync skipping");
}
var newKeypair = true;
var accountKeypair;
2018-05-15 16:01:09 -06:00
var promise = gl.store.accounts.checkKeypairAsync(args).then(function (keypair) {
2016-08-09 15:02:10 -04:00
if (keypair) {
// TODO keypairs
newKeypair = false;
accountKeypair = RSA.import(keypair);
return;
2016-08-09 15:02:10 -04:00
}
2016-08-08 19:43:31 -04:00
if (args.accountKeypair) {
// TODO keypairs
accountKeypair = RSA.import(args.accountKeypair);
return;
2016-08-08 19:43:31 -04:00
}
2018-07-13 04:20:57 -06:00
var keypairOpts = { bitlen: args.rsaKeySize, exp: 65537, public: true, pem: true };
// TODO keypairs
return (args.generateKeypair||RSA.generateKeypairAsync)(keypairOpts).then(function (keypair) {
2016-08-08 18:11:25 -04:00
keypair.privateKeyPem = RSA.exportPrivatePem(keypair);
2016-08-08 19:14:53 -04:00
keypair.publicKeyPem = RSA.exportPublicPem(keypair);
2016-08-08 18:11:25 -04:00
keypair.privateKeyJwk = RSA.exportPrivateJwk(keypair);
accountKeypair = keypair;
2016-08-08 18:11:25 -04:00
});
}).then(function () {
return accountKeypair;
});
2016-08-08 15:17:09 -04:00
2016-08-08 18:11:25 -04:00
return promise.then(function (keypair) {
2016-08-08 15:17:09 -04:00
// Note: the ACME urls are always fetched fresh on purpose
// TODO acme-v2/nocompat
2016-08-08 15:17:09 -04:00
return core.getAcmeUrlsAsync(args).then(function (urls) {
args._acmeUrls = urls;
// TODO acme-v2/nocompat
2018-05-15 16:01:09 -06:00
return gl.acme.registerNewAccountAsync({
2016-08-08 15:17:09 -04:00
email: args.email
, newRegUrl: args._acmeUrls.newReg
, newAuthzUrl: args._acmeUrls.newAuthz
2016-08-08 15:17:09 -04:00
, agreeToTerms: function (tosUrl, agreeCb) {
2018-05-15 16:01:09 -06:00
if (true === args.agreeTos || tosUrl === args.agreeTos || tosUrl === gl.agreeToTerms) {
2016-08-08 15:17:09 -04:00
agreeCb(null, tosUrl);
return;
}
// args.email = email; // already there
// args.domains = domains // already there
args.tosUrl = tosUrl;
2018-05-15 16:01:09 -06:00
gl.agreeToTerms(args, agreeCb);
2016-08-08 15:17:09 -04:00
}
, accountKeypair: keypair
2018-05-15 16:01:09 -06:00
, debug: gl.debug || args.debug
2016-08-08 19:14:53 -04:00
}).then(function (receipt) {
var reg = {
keypair: keypair
, receipt: receipt
, email: args.email
, newRegUrl: args._acmeUrls.newReg
, newAuthzUrl: args._acmeUrls.newAuthz
2016-08-08 19:14:53 -04:00
};
return gl.store.accounts.setKeypairAsync(args, keypair).then(function () {
2016-08-08 19:14:53 -04:00
// TODO move templating of arguments to right here?
if (!gl.store.accounts.setAsync) { return PromiseA.resolve({ keypair: keypair }); }
return gl.store.accounts.setAsync(args, reg).then(function (account) {
if (account && 'object' !== typeof account) {
throw new Error("store.accounts.setAsync should either return 'null' or an object with at least a string 'id'");
}
if (!account) { account = {}; }
account.keypair = keypair;
return account;
});
});
2016-08-08 15:17:09 -04:00
});
2016-08-06 02:05:04 -04:00
});
});
2016-08-04 18:49:35 -04:00
});
2016-08-06 02:05:04 -04:00
}
2016-08-08 15:17:09 -04:00
2016-08-08 18:11:25 -04:00
// Accounts
// (only used for keypair)
2016-08-06 02:05:04 -04:00
, getAsync: function (args) {
return core.accounts.checkAsync(args).then(function (account) {
if (!account) { return core.accounts.registerAsync(args); }
if (account.keypair) { return account; }
if (!args.account) { args.account = {}; }
if ('object' === typeof args.account && !args.account.id) { args.account.id = args.accountId || args.email || ''; }
var copy = utils.merge(args, gl);
args = utils.tplCopy(copy);
return gl.store.accounts.checkKeypairAsync(args).then(function (keypair) {
if (keypair) { return { keypair: keypair }; }
2016-08-08 15:17:09 -04:00
return core.accounts.registerAsync(args);
});
2016-08-06 02:05:04 -04:00
});
}
2016-08-08 15:17:09 -04:00
2016-08-08 18:11:25 -04:00
// Accounts
2016-08-06 02:05:04 -04:00
, checkAsync: function (args) {
2016-08-08 15:17:09 -04:00
var requiredArgs = ['accountId', 'email', 'domains', 'domain'];
2016-08-08 18:11:25 -04:00
if (!requiredArgs.some(function (key) { return -1 !== Object.keys(args).indexOf(key); })) {
2016-08-08 15:17:09 -04:00
return PromiseA.reject(new Error(
"In order to register or retrieve an account one of '" + requiredArgs.join("', '") + "' must be present"
));
}
2015-12-20 02:01:31 +00:00
2018-05-15 16:01:09 -06:00
var copy = utils.merge(args, gl);
2016-08-08 15:17:09 -04:00
args = utils.tplCopy(copy);
if (!args.account) { args.account = {}; }
if ('object' === typeof args.account && !args.account.id) { args.account.id = args.accountId || args.email || ''; }
2015-12-20 03:31:05 +00:00
// we can re-register the same account until we're blue in the face and it's all the same
// of course, we can also skip the lookup if we do store the account, but whatever
if (!gl.store.accounts.checkAsync) { return null; }
2018-05-15 16:01:09 -06:00
return gl.store.accounts.checkAsync(args).then(function (account) {
2015-12-20 05:13:41 +00:00
2016-08-08 15:17:09 -04:00
if (!account) {
return null;
}
2015-12-20 05:13:41 +00:00
2016-08-08 15:17:09 -04:00
args.account = account;
args.accountId = account.id;
2016-08-05 04:14:40 -04:00
2016-08-08 15:17:09 -04:00
return account;
2016-08-06 02:05:04 -04:00
});
2016-08-04 18:49:35 -04:00
}
2016-08-04 14:26:49 -04:00
}
2016-08-06 02:05:04 -04:00
, certificates: {
2016-08-08 18:11:25 -04:00
// Certificates
2016-08-06 02:05:04 -04:00
registerAsync: function (args) {
2016-08-07 02:02:02 -04:00
var err;
2018-05-15 16:01:09 -06:00
var challengeDefaults = gl['_challengeOpts_' + (args.challengeType || gl.challengeType)] || {};
2016-09-13 18:27:24 -06:00
var copy = utils.merge(args, challengeDefaults || {});
2018-05-15 16:01:09 -06:00
copy = utils.merge(copy, gl);
2016-08-07 02:02:02 -04:00
args = utils.tplCopy(copy);
2016-08-05 04:14:40 -04:00
2016-08-07 02:02:02 -04:00
if (!Array.isArray(args.domains)) {
return PromiseA.reject(new Error('args.domains should be an array of domains'));
}
if (!(args.domains.length && args.domains.every(utils.isValidDomain))) {
// NOTE: this library can't assume to handle the http loopback
// (or dns-01 validation may be used)
// so we do not check dns records or attempt a loopback here
err = new Error("invalid domain name(s): '" + args.domains + "'");
err.code = "INVALID_DOMAIN";
return PromiseA.reject(err);
}
2016-08-04 14:26:49 -04:00
// If a previous request to (re)register a certificate is already underway we need
// to return the same promise created before rather than registering things twice.
// I'm not 100% sure how to properly handle the case where someone registers domain
// lists with some but not all elements common, nor am I sure that's even a case that
// is allowed to happen anyway. But for now we act like the list is completely the
// same if any elements are the same.
var promise;
args.domains.some(function (name) {
if (pendingRegistrations.hasOwnProperty(name)) {
promise = pendingRegistrations[name];
return true;
}
});
if (promise) {
return promise;
}
promise = core.certificates._runRegistration(args);
// Now that the registration is actually underway we need to make sure any subsequent
// registration attempts return the same promise until it is completed (but not after
// it is completed).
args.domains.forEach(function (name) {
pendingRegistrations[name] = promise;
});
function clearPending() {
args.domains.forEach(function (name) {
delete pendingRegistrations[name];
});
}
promise.then(clearPending, clearPending);
return promise;
}
, _runRegistration: function (args) {
2016-08-09 15:02:10 -04:00
// TODO renewal cb
// accountId and or email
return core.accounts.getAsync(args).then(function (account) {
args.account = account;
2016-08-07 02:02:02 -04:00
if (args.certificate && args.certificate.privkey && (args.certificate.privkey.jwk || args.certificate.privkey.pem)) {
// TODO import jwk or pem and return it here
console.warn("TODO: implement certificates.checkKeypairAsync skipping");
}
var domainKeypair;
// This has been done in the getAsync already, so we skip it here
// if approveDomains doesn't set subject, we set it here
//args.subject = args.subject || args.domains[0];
2018-05-15 16:01:09 -06:00
var promise = gl.store.certificates.checkKeypairAsync(args).then(function (keypair) {
2016-08-09 15:02:10 -04:00
if (keypair) {
domainKeypair = RSA.import(keypair);
return;
2016-08-09 15:02:10 -04:00
}
2016-08-08 19:43:31 -04:00
if (args.domainKeypair) {
domainKeypair = RSA.import(args.domainKeypair);
return;
2016-08-08 19:43:31 -04:00
}
2018-07-13 04:20:57 -06:00
var keypairOpts = { bitlen: args.rsaKeySize, exp: 65537, public: true, pem: true };
return (args.generateKeypair||RSA.generateKeypairAsync)(keypairOpts).then(function (keypair) {
2016-08-07 02:02:02 -04:00
keypair.privateKeyPem = RSA.exportPrivatePem(keypair);
2016-08-08 19:14:53 -04:00
keypair.publicKeyPem = RSA.exportPublicPem(keypair);
2016-08-07 02:02:02 -04:00
keypair.privateKeyJwk = RSA.exportPrivateJwk(keypair);
domainKeypair = keypair;
2016-08-07 02:02:02 -04:00
});
}).then(function () {
return domainKeypair;
2016-08-06 02:05:04 -04:00
});
2016-08-04 14:26:49 -04:00
2016-08-07 02:02:02 -04:00
return promise.then(function (domainKeypair) {
args.domainKeypair = domainKeypair;
//args.registration = domainKey;
2016-08-08 15:17:09 -04:00
// Note: the ACME urls are always fetched fresh on purpose
// TODO is this the right place for this?
return core.getAcmeUrlsAsync(args).then(function (urls) {
args._acmeUrls = urls;
2016-08-09 15:02:10 -04:00
var certReq = {
2018-05-15 16:01:09 -06:00
debug: args.debug || gl.debug
2016-08-08 15:17:09 -04:00
, newAuthzUrl: args._acmeUrls.newAuthz
, newCertUrl: args._acmeUrls.newCert
, accountKeypair: RSA.import(account.keypair)
, domainKeypair: domainKeypair
, domains: args.domains
, challengeType: args.challengeType
2016-08-09 15:02:10 -04:00
};
//
// IMPORTANT
//
// setChallenge and removeChallenge are handed defaults
// instead of args because getChallenge does not have
// access to args
// (args is per-request, defaults is per instance)
//
// Each of these fires individually for each domain,
// even though the certificate on the whole may have many domains
//
certReq.setChallenge = function (domain, key, value, done) {
log(args.debug, "setChallenge called for '" + domain + "'");
2016-09-15 00:40:57 -06:00
var copy = utils.merge({ domains: [domain] }, args);
2018-05-15 16:01:09 -06:00
copy = utils.merge(copy, gl);
2016-08-09 15:02:10 -04:00
utils.tplCopy(copy);
// TODO need to save challengeType
2018-05-15 16:01:09 -06:00
gl.challenges[args.challengeType].set(copy, domain, key, value, done);
2016-08-09 15:02:10 -04:00
};
certReq.removeChallenge = function (domain, key, done) {
2016-09-15 00:40:57 -06:00
log(args.debug, "removeChallenge called for '" + domain + "'");
2018-05-15 16:01:09 -06:00
var copy = utils.merge({ domains: [domain] }, gl);
2016-08-09 15:02:10 -04:00
utils.tplCopy(copy);
2018-05-15 16:01:09 -06:00
gl.challenges[args.challengeType].remove(copy, domain, key, done);
2016-08-09 15:02:10 -04:00
};
2018-05-15 16:01:09 -06:00
log(args.debug, 'calling greenlock.acme.getCertificateAsync', certReq.domains);
2016-08-09 15:10:44 -04:00
// TODO acme-v2/nocompat
2018-05-15 16:01:09 -06:00
return gl.acme.getCertificateAsync(certReq).then(utils.attachCertInfo);
2016-08-08 15:17:09 -04:00
});
2016-08-07 02:02:02 -04:00
}).then(function (results) {
//var requested = {};
//var issued = {};
2016-08-13 14:35:19 -06:00
// { cert, chain, privkey /*TODO, subject, altnames, issuedAt, expiresAt */ }
2016-08-06 02:05:04 -04:00
2018-07-11 22:33:44 -06:00
// args.certs.privkey = RSA.exportPrivatePem(options.domainKeypair);
2016-08-13 14:35:19 -06:00
args.certs = results;
// args.pems is deprecated
2016-08-07 02:02:02 -04:00
args.pems = results;
// This has been done in the getAsync already, so we skip it here
// if approveDomains doesn't set subject, we set it here
//args.subject = args.subject || args.domains[0];
return gl.store.certificates.setKeypairAsync(args, domainKeypair).then(function () {
return gl.store.certificates.setAsync(args).then(function () {
return results;
});
2016-08-07 02:02:02 -04:00
});
});
2016-08-06 02:05:04 -04:00
});
2016-08-04 18:49:35 -04:00
}
2016-08-08 20:26:46 -04:00
// Certificates
, renewAsync: function (args, certs) {
var renewableAt = core.certificates._getRenewableAt(args, certs);
2016-08-09 15:51:42 -04:00
var err;
//var halfLife = (certs.expiresAt - certs.issuedAt) / 2;
//var renewable = (Date.now() - certs.issuedAt) > halfLife;
log(args.debug, "(Renew) Expires At", new Date(certs.expiresAt).toISOString());
log(args.debug, "(Renew) Renewable At", new Date(renewableAt).toISOString());
if (!args.duplicate && Date.now() < renewableAt) {
2016-08-09 15:51:42 -04:00
err = new Error(
"[ERROR] Certificate issued at '"
+ new Date(certs.issuedAt).toISOString() + "' and expires at '"
+ new Date(certs.expiresAt).toISOString() + "'. Ignoring renewal attempt until '"
+ new Date(renewableAt).toISOString() + "'. Set { duplicate: true } to force."
2016-08-09 15:51:42 -04:00
);
err.code = 'E_NOT_RENEWABLE';
return PromiseA.reject(err);
}
// Either the cert has entered its renewal period
// or we're forcing a refresh via 'dupliate: true'
log(args.debug, "Renewing!");
2018-11-05 11:10:22 -07:00
if (!args.domains || !args.domains.length) {
args.domains = args.servernames || [certs.subject].concat(certs.altnames);
}
2016-08-08 15:17:09 -04:00
return core.certificates.registerAsync(args);
}
2016-08-08 20:26:46 -04:00
// Certificates
, _isRenewable: function (args, certs) {
var renewableAt = core.certificates._getRenewableAt(args, certs);
log(args.debug, "Check Expires At", new Date(certs.expiresAt).toISOString());
log(args.debug, "Check Renewable At", new Date(renewableAt).toISOString());
if (args.duplicate || Date.now() >= renewableAt) {
log(args.debug, "certificates are renewable");
return true;
}
return false;
}
, _getRenewableAt: function (args, certs) {
2018-05-15 16:01:09 -06:00
return certs.expiresAt - (args.renewWithin || gl.renewWithin);
}
2016-08-06 02:05:04 -04:00
, checkAsync: function (args) {
2018-05-15 16:01:09 -06:00
var copy = utils.merge(args, gl);
// if approveDomains doesn't set subject, we set it here
if (!(copy.domains && copy.domains.length)) { copy.domains = [ copy.subject || copy.domain ].filter(Boolean); }
if (!copy.subject) { copy.subject = copy.domains[0]; }
if (!copy.domain) { copy.domain = copy.domains[0]; }
args = utils.tplCopy(copy);
2015-12-15 15:21:27 +00:00
2016-08-07 02:02:02 -04:00
// returns pems
return gl.store.certificates.checkAsync(args).then(function (cert) {
2016-08-08 20:26:46 -04:00
if (cert) {
cert = utils.attachCertInfo(cert);
if (utils.certHasDomain(cert, args.domain)) {
log(args.debug, 'checkAsync found existing certificates');
return cert;
}
log(args.debug, 'checkAsync found mismatched / incomplete certificates');
2016-08-08 20:26:46 -04:00
}
log(args.debug, 'checkAsync failed to find certificates');
2016-08-08 20:26:46 -04:00
return null;
});
2016-08-04 18:49:35 -04:00
}
2016-08-08 20:26:46 -04:00
// Certificates
2016-08-06 02:05:04 -04:00
, getAsync: function (args) {
2018-05-15 16:01:09 -06:00
var copy = utils.merge(args, gl);
// if approveDomains doesn't set subject, we set it here
if (!(copy.domains && copy.domains.length)) { copy.domains = [ copy.subject || copy.domain ].filter(Boolean); }
if (!copy.subject) { copy.subject = copy.domains[0]; }
if (!copy.domain) { copy.domain = copy.domains[0]; }
2016-08-07 02:02:02 -04:00
args = utils.tplCopy(copy);
2016-08-06 02:05:04 -04:00
if (args.certificate && args.certificate.privkey && args.certificate.cert && args.certificate.chain) {
// TODO skip fetching a certificate if it's fetched during approveDomains
console.warn("TODO: implement certificates.checkAsync skipping");
}
2016-08-06 02:05:04 -04:00
return core.certificates.checkAsync(args).then(function (certs) {
if (certs) { certs = utils.attachCertInfo(certs); }
if (!certs || !utils.certHasDomain(certs, args.domain)) {
2016-08-08 15:17:09 -04:00
// There is no cert available
if (false !== args.securityUpdates && !args._communityMemberAdded) {
2018-05-10 02:31:24 -06:00
try {
// We will notify all greenlock users of mandatory and security updates
2018-11-05 11:10:22 -07:00
// We'll keep track of versions and os so we can make sure things work well
// { name, version, email, domains, action, communityMember, telemetry }
require('./community').add({
name: args._communityPackage
, version: args._communityPackageVersion
, email: args.email
, domains: args.domains || args.servernames
, action: 'reg'
, communityMember: args.communityMember
, telemetry: args.telemetry
});
2018-05-10 02:31:24 -06:00
} catch(e) { /* ignore */ }
args._communityMemberAdded = true;
}
2016-08-08 15:17:09 -04:00
return core.certificates.registerAsync(args);
}
if (core.certificates._isRenewable(args, certs)) {
// it's time to renew the available cert
if (false !== args.securityUpdates && !args._communityMemberAdded) {
2018-05-10 02:31:24 -06:00
try {
// We will notify all greenlock users of mandatory and security updates
2018-11-05 11:10:22 -07:00
// We'll keep track of versions and os so we can make sure things work well
// { name, version, email, domains, action, communityMember, telemetry }
require('./community').add({
name: args._communityPackage
, version: args._communityPackageVersion
, email: args.email
, domains: args.domains || args.servernames
, action: 'renew'
, communityMember: args.communityMember
, telemetry: args.telemetry
});
2018-05-10 02:31:24 -06:00
} catch(e) { /* ignore */ }
args._communityMemberAdded = true;
}
certs.renewing = core.certificates.renewAsync(args, certs);
if (args.waitForRenewal) {
return certs.renewing;
}
2016-08-06 02:05:04 -04:00
}
2016-08-04 14:26:49 -04:00
// return existing unexpired (although potentially stale) certificates when available
// there will be an additional .renewing property if the certs are being asynchronously renewed
return certs;
2016-08-07 02:02:02 -04:00
}).then(function (results) {
// returns pems
return results;
2016-08-06 02:05:04 -04:00
});
}
}
2015-12-15 15:21:27 +00:00
2015-12-15 03:37:39 -08:00
};
2016-08-06 02:05:04 -04:00
return core;
2015-12-15 03:37:39 -08:00
};