mirror of https://github.com/lukechilds/node.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
208 lines
5.7 KiB
208 lines
5.7 KiB
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var util = require('util');
|
|
var constants = require('constants');
|
|
var tls = require('tls');
|
|
|
|
// Lazily loaded
|
|
var crypto = null;
|
|
|
|
var binding = process.binding('crypto');
|
|
var NativeSecureContext = binding.SecureContext;
|
|
|
|
var CONTEXT_DEFAULT_OPTIONS = undefined;
|
|
|
|
function getSecureOptions(secureProtocol, secureOptions) {
|
|
if (CONTEXT_DEFAULT_OPTIONS === undefined) {
|
|
CONTEXT_DEFAULT_OPTIONS = 0;
|
|
|
|
if (!binding.SSL3_ENABLE)
|
|
CONTEXT_DEFAULT_OPTIONS |= constants.SSL_OP_NO_SSLv3;
|
|
|
|
if (!binding.SSL2_ENABLE)
|
|
CONTEXT_DEFAULT_OPTIONS |= constants.SSL_OP_NO_SSLv2;
|
|
}
|
|
|
|
if (secureOptions === undefined) {
|
|
if (secureProtocol === undefined ||
|
|
secureProtocol === 'SSLv23_method' ||
|
|
secureProtocol === 'SSLv23_server_method' ||
|
|
secureProtocol === 'SSLv23_client_method') {
|
|
secureOptions |= CONTEXT_DEFAULT_OPTIONS;
|
|
}
|
|
}
|
|
|
|
return secureOptions;
|
|
}
|
|
exports._getSecureOptions = getSecureOptions;
|
|
|
|
function SecureContext(secureProtocol, flags, context) {
|
|
if (!(this instanceof SecureContext)) {
|
|
return new SecureContext(secureProtocol, flags, context);
|
|
}
|
|
|
|
if (context) {
|
|
this.context = context;
|
|
} else {
|
|
this.context = new NativeSecureContext();
|
|
|
|
if (secureProtocol) {
|
|
this.context.init(secureProtocol);
|
|
} else {
|
|
this.context.init();
|
|
}
|
|
}
|
|
|
|
flags = getSecureOptions(secureProtocol, flags);
|
|
|
|
this.context.setOptions(flags);
|
|
}
|
|
|
|
exports.SecureContext = SecureContext;
|
|
|
|
|
|
exports.createSecureContext = function createSecureContext(options, context) {
|
|
if (!options) options = {};
|
|
|
|
var secureOptions = options.secureOptions;
|
|
if (options.honorCipherOrder)
|
|
secureOptions |= constants.SSL_OP_CIPHER_SERVER_PREFERENCE;
|
|
|
|
var c = new SecureContext(options.secureProtocol, secureOptions, context);
|
|
|
|
if (context) return c;
|
|
|
|
if (options.key) {
|
|
if (Array.isArray(options.key)) {
|
|
for (var i = 0; i < options.key.length; i++) {
|
|
var key = options.key[i];
|
|
|
|
if (key.passphrase)
|
|
c.context.setKey(key.pem, key.passphrase);
|
|
else
|
|
c.context.setKey(key);
|
|
}
|
|
} else {
|
|
if (options.passphrase) {
|
|
c.context.setKey(options.key, options.passphrase);
|
|
} else {
|
|
c.context.setKey(options.key);
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: It's important to add CA before the cert to be able to load
|
|
// cert's issuer in C++ code.
|
|
if (options.ca) {
|
|
if (util.isArray(options.ca)) {
|
|
for (var i = 0, len = options.ca.length; i < len; i++) {
|
|
c.context.addCACert(options.ca[i]);
|
|
}
|
|
} else {
|
|
c.context.addCACert(options.ca);
|
|
}
|
|
} else {
|
|
c.context.addRootCerts();
|
|
}
|
|
|
|
if (options.cert) {
|
|
if (Array.isArray(options.cert)) {
|
|
for (var i = 0; i < options.cert.length; i++)
|
|
c.context.setCert(options.cert[i]);
|
|
} else {
|
|
c.context.setCert(options.cert);
|
|
}
|
|
}
|
|
|
|
if (options.ciphers)
|
|
c.context.setCiphers(options.ciphers);
|
|
else
|
|
c.context.setCiphers(tls.DEFAULT_CIPHERS);
|
|
|
|
if (util.isUndefined(options.ecdhCurve))
|
|
c.context.setECDHCurve(tls.DEFAULT_ECDH_CURVE);
|
|
else if (options.ecdhCurve)
|
|
c.context.setECDHCurve(options.ecdhCurve);
|
|
|
|
if (options.dhparam) c.context.setDHParam(options.dhparam);
|
|
|
|
if (options.crl) {
|
|
if (util.isArray(options.crl)) {
|
|
for (var i = 0, len = options.crl.length; i < len; i++) {
|
|
c.context.addCRL(options.crl[i]);
|
|
}
|
|
} else {
|
|
c.context.addCRL(options.crl);
|
|
}
|
|
}
|
|
|
|
if (options.sessionIdContext) {
|
|
c.context.setSessionIdContext(options.sessionIdContext);
|
|
}
|
|
|
|
if (options.pfx) {
|
|
var pfx = options.pfx;
|
|
var passphrase = options.passphrase;
|
|
|
|
if (!crypto)
|
|
crypto = require('crypto');
|
|
|
|
pfx = crypto._toBuf(pfx);
|
|
if (passphrase)
|
|
passphrase = crypto._toBuf(passphrase);
|
|
|
|
if (passphrase) {
|
|
c.context.loadPKCS12(pfx, passphrase);
|
|
} else {
|
|
c.context.loadPKCS12(pfx);
|
|
}
|
|
}
|
|
|
|
return c;
|
|
};
|
|
|
|
exports.translatePeerCertificate = function translatePeerCertificate(c) {
|
|
if (!c)
|
|
return null;
|
|
|
|
if (c.issuer) c.issuer = tls.parseCertString(c.issuer);
|
|
if (c.issuerCertificate && c.issuerCertificate !== c) {
|
|
c.issuerCertificate = translatePeerCertificate(c.issuerCertificate);
|
|
}
|
|
if (c.subject) c.subject = tls.parseCertString(c.subject);
|
|
if (c.infoAccess) {
|
|
var info = c.infoAccess;
|
|
c.infoAccess = {};
|
|
|
|
// XXX: More key validation?
|
|
info.replace(/([^\n:]*):([^\n]*)(?:\n|$)/g, function(all, key, val) {
|
|
if (key === '__proto__')
|
|
return;
|
|
|
|
if (c.infoAccess.hasOwnProperty(key))
|
|
c.infoAccess[key].push(val);
|
|
else
|
|
c.infoAccess[key] = [val];
|
|
});
|
|
}
|
|
return c;
|
|
};
|
|
|