| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | require('dotenv').config(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | var ACME = require('../'); | 
					
						
							|  |  |  | var Keypairs = require('../lib/keypairs.js'); | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | var acme = ACME.create({ | 
					
						
							|  |  |  | 	// debug: true
 | 
					
						
							|  |  |  | }); | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | // TODO exec npm install --save-dev CHALLENGE_MODULE
 | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | var config = { | 
					
						
							|  |  |  | 	env: process.env.ENV, | 
					
						
							|  |  |  | 	email: process.env.SUBSCRIBER_EMAIL, | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | 	domain: process.env.BASE_DOMAIN, | 
					
						
							|  |  |  | 	challengeType: process.env.CHALLENGE_TYPE, | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | 	challengeModule: process.env.CHALLENGE_PLUGIN, | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | 	challengeOptions: JSON.parse(process.env.CHALLENGE_OPTIONS) | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | }; | 
					
						
							|  |  |  | config.debug = !/^PROD/i.test(config.env); | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | var pluginPrefix = 'acme-' + config.challengeType + '-'; | 
					
						
							|  |  |  | var pluginName = config.challengeModule; | 
					
						
							|  |  |  | var plugin; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function badPlugin(err) { | 
					
						
							|  |  |  | 	if ('MODULE_NOT_FOUND' !== err.code) { | 
					
						
							|  |  |  | 		console.error(err); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	console.error("Couldn't find '" + pluginName + "'. Is it installed?"); | 
					
						
							|  |  |  | 	console.error("\tnpm install --save-dev '" + pluginName + "'"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | try { | 
					
						
							|  |  |  | 	plugin = require(pluginName); | 
					
						
							|  |  |  | } catch (err) { | 
					
						
							|  |  |  | 	if ( | 
					
						
							|  |  |  | 		'MODULE_NOT_FOUND' !== err.code || | 
					
						
							|  |  |  | 		0 === pluginName.indexOf(pluginPrefix) | 
					
						
							|  |  |  | 	) { | 
					
						
							|  |  |  | 		badPlugin(err); | 
					
						
							|  |  |  | 		process.exit(1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	try { | 
					
						
							|  |  |  | 		pluginName = pluginPrefix + pluginName; | 
					
						
							|  |  |  | 		plugin = require(pluginName); | 
					
						
							|  |  |  | 	} catch (e) { | 
					
						
							|  |  |  | 		badPlugin(e); | 
					
						
							|  |  |  | 		process.exit(1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | config.challenger = plugin.create(config.challengeOptions); | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | if (!config.challengeType || !config.domain) { | 
					
						
							|  |  |  | 	console.error( | 
					
						
							|  |  |  | 		new Error('Missing config variables. Check you .env and the docs') | 
					
						
							|  |  |  | 			.message | 
					
						
							|  |  |  | 	); | 
					
						
							|  |  |  | 	console.error(config); | 
					
						
							|  |  |  | 	process.exit(1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var challenges = {}; | 
					
						
							|  |  |  | challenges[config.challengeType] = config.challenger; | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | async function happyPath(accKty, srvKty, rnd) { | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 	var agreed = false; | 
					
						
							|  |  |  | 	var metadata = await acme.init( | 
					
						
							|  |  |  | 		'https://acme-staging-v02.api.letsencrypt.org/directory' | 
					
						
							|  |  |  | 	); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Ready to use, show page
 | 
					
						
							|  |  |  | 	if (config.debug) { | 
					
						
							|  |  |  | 		console.info('ACME.js initialized'); | 
					
						
							|  |  |  | 		console.info(metadata); | 
					
						
							|  |  |  | 		console.info(''); | 
					
						
							|  |  |  | 		console.info(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | 	var accountKeypair = await Keypairs.generate({ kty: accKty }); | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 	if (config.debug) { | 
					
						
							|  |  |  | 		console.info('Account Key Created'); | 
					
						
							|  |  |  | 		console.info(JSON.stringify(accountKeypair, null, 2)); | 
					
						
							|  |  |  | 		console.info(''); | 
					
						
							|  |  |  | 		console.info(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var account = await acme.accounts.create({ | 
					
						
							|  |  |  | 		agreeToTerms: agree, | 
					
						
							|  |  |  | 		// TODO detect jwk/pem/der?
 | 
					
						
							|  |  |  | 		accountKeypair: { privateKeyJwk: accountKeypair.private }, | 
					
						
							|  |  |  | 		email: config.email | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 	// TODO top-level agree
 | 
					
						
							|  |  |  | 	function agree(tos) { | 
					
						
							|  |  |  | 		if (config.debug) { | 
					
						
							|  |  |  | 			console.info('Agreeing to Terms of Service:'); | 
					
						
							|  |  |  | 			console.info(tos); | 
					
						
							|  |  |  | 			console.info(''); | 
					
						
							|  |  |  | 			console.info(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		agreed = true; | 
					
						
							|  |  |  | 		return Promise.resolve(tos); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (config.debug) { | 
					
						
							|  |  |  | 		console.info('New Subscriber Account'); | 
					
						
							|  |  |  | 		console.info(JSON.stringify(account, null, 2)); | 
					
						
							|  |  |  | 		console.info(); | 
					
						
							|  |  |  | 		console.info(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (!agreed) { | 
					
						
							|  |  |  | 		throw new Error('Failed to ask the user to agree to terms'); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | 	var serverKeypair = await Keypairs.generate({ kty: srvKty }); | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 	if (config.debug) { | 
					
						
							|  |  |  | 		console.info('Server Key Created'); | 
					
						
							|  |  |  | 		console.info(JSON.stringify(serverKeypair, null, 2)); | 
					
						
							|  |  |  | 		console.info(); | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | 		console.info(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | 	var domains = randomDomains(rnd); | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | 	if (config.debug) { | 
					
						
							|  |  |  | 		console.info('Get certificates for random domains:'); | 
					
						
							|  |  |  | 		console.info(domains); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	var results = await acme.certificates.create({ | 
					
						
							|  |  |  | 		account: account, | 
					
						
							|  |  |  | 		accountKeypair: { privateKeyJwk: accountKeypair.private }, | 
					
						
							|  |  |  | 		serverKeypair: { privateKeyJwk: serverKeypair.private }, | 
					
						
							|  |  |  | 		domains: domains, | 
					
						
							|  |  |  | 		challenges: challenges, // must be implemented
 | 
					
						
							|  |  |  | 		skipDryRun: true | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (config.debug) { | 
					
						
							|  |  |  | 		console.info('Got SSL Certificate:'); | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | 		console.info(Object.keys(results)); | 
					
						
							| 
									
										
										
										
											2019-10-05 05:21:07 -06:00
										 |  |  | 		console.info(results.expires); | 
					
						
							|  |  |  | 		console.info(results.cert); | 
					
						
							|  |  |  | 		console.info(results.chain); | 
					
						
							|  |  |  | 		console.info(''); | 
					
						
							|  |  |  | 		console.info(''); | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | // Try EC + RSA
 | 
					
						
							|  |  |  | var rnd = random(); | 
					
						
							|  |  |  | happyPath('EC', 'RSA', rnd) | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 	.then(function() { | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | 		// Now try RSA + EC
 | 
					
						
							|  |  |  | 		rnd = random(); | 
					
						
							|  |  |  | 		return happyPath('RSA', 'EC', rnd).then(function() { | 
					
						
							|  |  |  | 			console.info('success'); | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 	.catch(function(err) { | 
					
						
							|  |  |  | 		console.error('Error:'); | 
					
						
							|  |  |  | 		console.error(err.stack); | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-06 01:22:18 -06:00
										 |  |  | function randomDomains(rnd) { | 
					
						
							| 
									
										
										
										
											2019-10-04 17:35:59 -06:00
										 |  |  | 	return ['foo-acmejs', 'bar-acmejs', '*.baz-acmejs', 'baz-acmejs'].map( | 
					
						
							|  |  |  | 		function(pre) { | 
					
						
							|  |  |  | 			return pre + '-' + rnd + '.' + config.domain; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function random() { | 
					
						
							|  |  |  | 	return parseInt( | 
					
						
							|  |  |  | 		Math.random() | 
					
						
							|  |  |  | 			.toString() | 
					
						
							|  |  |  | 			.slice(2, 99), | 
					
						
							|  |  |  | 		10 | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 		.toString(16) | 
					
						
							|  |  |  | 		.slice(0, 4); | 
					
						
							|  |  |  | } |