mirror of
https://github.com/NginxProxyManager/nginx-proxy-manager.git
synced 2025-08-23 01:10:23 +00:00
Fix remote execution bug where email address can contain malicious code
Some checks failed
Close stale issues and PRs / stale (push) Has been cancelled
Some checks failed
Close stale issues and PRs / stale (push) Has been cancelled
also convert almost all cmd execs for certificates to properly escape arguments
This commit is contained in:
@@ -1,5 +1,5 @@
|
|||||||
const _ = require('lodash');
|
const _ = require('lodash');
|
||||||
const fs = require('fs');
|
const fs = require('node:fs');
|
||||||
const batchflow = require('batchflow');
|
const batchflow = require('batchflow');
|
||||||
const logger = require('../logger').access;
|
const logger = require('../logger').access;
|
||||||
const error = require('../lib/error');
|
const error = require('../lib/error');
|
||||||
@@ -38,7 +38,7 @@ const internalAccessList = {
|
|||||||
.then((row) => {
|
.then((row) => {
|
||||||
data.id = row.id;
|
data.id = row.id;
|
||||||
|
|
||||||
let promises = [];
|
const promises = [];
|
||||||
|
|
||||||
// Now add the items
|
// Now add the items
|
||||||
data.items.map((item) => {
|
data.items.map((item) => {
|
||||||
@@ -116,7 +116,7 @@ const internalAccessList = {
|
|||||||
.then((row) => {
|
.then((row) => {
|
||||||
if (row.id !== data.id) {
|
if (row.id !== data.id) {
|
||||||
// Sanity check that something crazy hasn't happened
|
// Sanity check that something crazy hasn't happened
|
||||||
throw new error.InternalValidationError('Access List could not be updated, IDs do not match: ' + row.id + ' !== ' + data.id);
|
throw new error.InternalValidationError(`Access List could not be updated, IDs do not match: ${row.id} !== ${data.id}`);
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
@@ -135,10 +135,10 @@ const internalAccessList = {
|
|||||||
.then(() => {
|
.then(() => {
|
||||||
// Check for items and add/update/remove them
|
// Check for items and add/update/remove them
|
||||||
if (typeof data.items !== 'undefined' && data.items) {
|
if (typeof data.items !== 'undefined' && data.items) {
|
||||||
let promises = [];
|
const promises = [];
|
||||||
let items_to_keep = [];
|
const items_to_keep = [];
|
||||||
|
|
||||||
data.items.map(function (item) {
|
data.items.map((item) => {
|
||||||
if (item.password) {
|
if (item.password) {
|
||||||
promises.push(accessListAuthModel
|
promises.push(accessListAuthModel
|
||||||
.query()
|
.query()
|
||||||
@@ -154,7 +154,7 @@ const internalAccessList = {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let query = accessListAuthModel
|
const query = accessListAuthModel
|
||||||
.query()
|
.query()
|
||||||
.delete()
|
.delete()
|
||||||
.where('access_list_id', data.id);
|
.where('access_list_id', data.id);
|
||||||
@@ -175,9 +175,9 @@ const internalAccessList = {
|
|||||||
.then(() => {
|
.then(() => {
|
||||||
// Check for clients and add/update/remove them
|
// Check for clients and add/update/remove them
|
||||||
if (typeof data.clients !== 'undefined' && data.clients) {
|
if (typeof data.clients !== 'undefined' && data.clients) {
|
||||||
let promises = [];
|
const promises = [];
|
||||||
|
|
||||||
data.clients.map(function (client) {
|
data.clients.map((client) => {
|
||||||
if (client.address) {
|
if (client.address) {
|
||||||
promises.push(accessListClientModel
|
promises.push(accessListClientModel
|
||||||
.query()
|
.query()
|
||||||
@@ -190,7 +190,7 @@ const internalAccessList = {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let query = accessListClientModel
|
const query = accessListClientModel
|
||||||
.query()
|
.query()
|
||||||
.delete()
|
.delete()
|
||||||
.where('access_list_id', data.id);
|
.where('access_list_id', data.id);
|
||||||
@@ -249,7 +249,7 @@ const internalAccessList = {
|
|||||||
|
|
||||||
return access.can('access_lists:get', data.id)
|
return access.can('access_lists:get', data.id)
|
||||||
.then((access_data) => {
|
.then((access_data) => {
|
||||||
let query = accessListModel
|
const query = accessListModel
|
||||||
.query()
|
.query()
|
||||||
.select('access_list.*', accessListModel.raw('COUNT(proxy_host.id) as proxy_host_count'))
|
.select('access_list.*', accessListModel.raw('COUNT(proxy_host.id) as proxy_host_count'))
|
||||||
.leftJoin('proxy_host', function() {
|
.leftJoin('proxy_host', function() {
|
||||||
@@ -267,7 +267,7 @@ const internalAccessList = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (typeof data.expand !== 'undefined' && data.expand !== null) {
|
if (typeof data.expand !== 'undefined' && data.expand !== null) {
|
||||||
query.withGraphFetched('[' + data.expand.join(', ') + ']');
|
query.withGraphFetched(`[${data.expand.join(', ')}]`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return query.then(utils.omitRow(omissions()));
|
return query.then(utils.omitRow(omissions()));
|
||||||
@@ -327,7 +327,7 @@ const internalAccessList = {
|
|||||||
// 3. reconfigure those hosts, then reload nginx
|
// 3. reconfigure those hosts, then reload nginx
|
||||||
|
|
||||||
// set the access_list_id to zero for these items
|
// set the access_list_id to zero for these items
|
||||||
row.proxy_hosts.map(function (val, idx) {
|
row.proxy_hosts.map((_val, idx) => {
|
||||||
row.proxy_hosts[idx].access_list_id = 0;
|
row.proxy_hosts[idx].access_list_id = 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -340,11 +340,11 @@ const internalAccessList = {
|
|||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
// delete the htpasswd file
|
// delete the htpasswd file
|
||||||
let htpasswd_file = internalAccessList.getFilename(row);
|
const htpasswd_file = internalAccessList.getFilename(row);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
fs.unlinkSync(htpasswd_file);
|
fs.unlinkSync(htpasswd_file);
|
||||||
} catch (err) {
|
} catch (_err) {
|
||||||
// do nothing
|
// do nothing
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -374,7 +374,7 @@ const internalAccessList = {
|
|||||||
getAll: (access, expand, search_query) => {
|
getAll: (access, expand, search_query) => {
|
||||||
return access.can('access_lists:list')
|
return access.can('access_lists:list')
|
||||||
.then((access_data) => {
|
.then((access_data) => {
|
||||||
let query = accessListModel
|
const query = accessListModel
|
||||||
.query()
|
.query()
|
||||||
.select('access_list.*', accessListModel.raw('COUNT(proxy_host.id) as proxy_host_count'))
|
.select('access_list.*', accessListModel.raw('COUNT(proxy_host.id) as proxy_host_count'))
|
||||||
.leftJoin('proxy_host', function() {
|
.leftJoin('proxy_host', function() {
|
||||||
@@ -393,19 +393,19 @@ const internalAccessList = {
|
|||||||
// Query is used for searching
|
// Query is used for searching
|
||||||
if (typeof search_query === 'string') {
|
if (typeof search_query === 'string') {
|
||||||
query.where(function () {
|
query.where(function () {
|
||||||
this.where('name', 'like', '%' + search_query + '%');
|
this.where('name', 'like', `%${search_query}%`);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof expand !== 'undefined' && expand !== null) {
|
if (typeof expand !== 'undefined' && expand !== null) {
|
||||||
query.withGraphFetched('[' + expand.join(', ') + ']');
|
query.withGraphFetched(`[${expand.join(', ')}]`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return query.then(utils.omitRows(omissions()));
|
return query.then(utils.omitRows(omissions()));
|
||||||
})
|
})
|
||||||
.then((rows) => {
|
.then((rows) => {
|
||||||
if (rows) {
|
if (rows) {
|
||||||
rows.map(function (row, idx) {
|
rows.map((row, idx) => {
|
||||||
if (typeof row.items !== 'undefined' && row.items) {
|
if (typeof row.items !== 'undefined' && row.items) {
|
||||||
rows[idx] = internalAccessList.maskItems(row);
|
rows[idx] = internalAccessList.maskItems(row);
|
||||||
}
|
}
|
||||||
@@ -424,7 +424,7 @@ const internalAccessList = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
getCount: (user_id, visibility) => {
|
getCount: (user_id, visibility) => {
|
||||||
let query = accessListModel
|
const query = accessListModel
|
||||||
.query()
|
.query()
|
||||||
.count('id as count')
|
.count('id as count')
|
||||||
.where('is_deleted', 0);
|
.where('is_deleted', 0);
|
||||||
@@ -445,7 +445,7 @@ const internalAccessList = {
|
|||||||
*/
|
*/
|
||||||
maskItems: (list) => {
|
maskItems: (list) => {
|
||||||
if (list && typeof list.items !== 'undefined') {
|
if (list && typeof list.items !== 'undefined') {
|
||||||
list.items.map(function (val, idx) {
|
list.items.map((val, idx) => {
|
||||||
let repeat_for = 8;
|
let repeat_for = 8;
|
||||||
let first_char = '*';
|
let first_char = '*';
|
||||||
|
|
||||||
@@ -468,7 +468,7 @@ const internalAccessList = {
|
|||||||
* @returns {String}
|
* @returns {String}
|
||||||
*/
|
*/
|
||||||
getFilename: (list) => {
|
getFilename: (list) => {
|
||||||
return '/data/access/' + list.id;
|
return `/data/access/${list.id}`;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -479,15 +479,15 @@ const internalAccessList = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
build: (list) => {
|
build: (list) => {
|
||||||
logger.info('Building Access file #' + list.id + ' for: ' + list.name);
|
logger.info(`Building Access file #${list.id} for: ${list.name}`);
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
let htpasswd_file = internalAccessList.getFilename(list);
|
const htpasswd_file = internalAccessList.getFilename(list);
|
||||||
|
|
||||||
// 1. remove any existing access file
|
// 1. remove any existing access file
|
||||||
try {
|
try {
|
||||||
fs.unlinkSync(htpasswd_file);
|
fs.unlinkSync(htpasswd_file);
|
||||||
} catch (err) {
|
} catch (_err) {
|
||||||
// do nothing
|
// do nothing
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -504,14 +504,14 @@ const internalAccessList = {
|
|||||||
if (list.items.length) {
|
if (list.items.length) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
batchflow(list.items).sequential()
|
batchflow(list.items).sequential()
|
||||||
.each((i, item, next) => {
|
.each((_i, item, next) => {
|
||||||
if (typeof item.password !== 'undefined' && item.password.length) {
|
if (typeof item.password !== 'undefined' && item.password.length) {
|
||||||
logger.info('Adding: ' + item.username);
|
logger.info(`Adding: ${item.username}`);
|
||||||
|
|
||||||
utils.execFile('openssl', ['passwd', '-apr1', item.password])
|
utils.execFile('openssl', ['passwd', '-apr1', item.password])
|
||||||
.then((res) => {
|
.then((res) => {
|
||||||
try {
|
try {
|
||||||
fs.appendFileSync(htpasswd_file, item.username + ':' + res + '\n', {encoding: 'utf8'});
|
fs.appendFileSync(htpasswd_file, `${item.username}:${res}\n`, {encoding: 'utf8'});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
reject(err);
|
reject(err);
|
||||||
}
|
}
|
||||||
@@ -528,7 +528,7 @@ const internalAccessList = {
|
|||||||
reject(err);
|
reject(err);
|
||||||
})
|
})
|
||||||
.end((results) => {
|
.end((results) => {
|
||||||
logger.success('Built Access file #' + list.id + ' for: ' + list.name);
|
logger.success(`Built Access file #${list.id} for: ${list.name}`);
|
||||||
resolve(results);
|
resolve(results);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
const _ = require('lodash');
|
const _ = require('lodash');
|
||||||
const fs = require('fs');
|
const fs = require('node:fs');
|
||||||
const https = require('https');
|
const https = require('node:https');
|
||||||
const tempWrite = require('temp-write');
|
const tempWrite = require('temp-write');
|
||||||
const moment = require('moment');
|
const moment = require('moment');
|
||||||
const archiver = require('archiver');
|
const archiver = require('archiver');
|
||||||
@@ -49,7 +49,7 @@ const internalCertificate = {
|
|||||||
processExpiringHosts: () => {
|
processExpiringHosts: () => {
|
||||||
if (!internalCertificate.intervalProcessing) {
|
if (!internalCertificate.intervalProcessing) {
|
||||||
internalCertificate.intervalProcessing = true;
|
internalCertificate.intervalProcessing = true;
|
||||||
logger.info('Renewing SSL certs expiring within ' + internalCertificate.renewBeforeExpirationBy[0] + ' ' + internalCertificate.renewBeforeExpirationBy[1] + ' ...');
|
logger.info(`Renewing SSL certs expiring within ${internalCertificate.renewBeforeExpirationBy[0]} ${internalCertificate.renewBeforeExpirationBy[1]} ...`);
|
||||||
|
|
||||||
const expirationThreshold = moment().add(internalCertificate.renewBeforeExpirationBy[0], internalCertificate.renewBeforeExpirationBy[1]).format('YYYY-MM-DD HH:mm:ss');
|
const expirationThreshold = moment().add(internalCertificate.renewBeforeExpirationBy[0], internalCertificate.renewBeforeExpirationBy[1]).format('YYYY-MM-DD HH:mm:ss');
|
||||||
|
|
||||||
@@ -70,7 +70,7 @@ const internalCertificate = {
|
|||||||
*/
|
*/
|
||||||
let sequence = Promise.resolve();
|
let sequence = Promise.resolve();
|
||||||
|
|
||||||
certificates.forEach(function (certificate) {
|
certificates.forEach((certificate) => {
|
||||||
sequence = sequence.then(() =>
|
sequence = sequence.then(() =>
|
||||||
internalCertificate
|
internalCertificate
|
||||||
.renew(
|
.renew(
|
||||||
@@ -202,7 +202,7 @@ const internalCertificate = {
|
|||||||
.then(() => {
|
.then(() => {
|
||||||
// At this point, the letsencrypt cert should exist on disk.
|
// At this point, the letsencrypt cert should exist on disk.
|
||||||
// Lets get the expiry date from the file and update the row silently
|
// Lets get the expiry date from the file and update the row silently
|
||||||
return internalCertificate.getCertificateInfoFromFile('/etc/letsencrypt/live/npm-' + certificate.id + '/fullchain.pem')
|
return internalCertificate.getCertificateInfoFromFile(`${internalCertificate.getLiveCertPath(certificate.id)}/fullchain.pem`)
|
||||||
.then((cert_info) => {
|
.then((cert_info) => {
|
||||||
return certificateModel
|
return certificateModel
|
||||||
.query()
|
.query()
|
||||||
@@ -263,7 +263,7 @@ const internalCertificate = {
|
|||||||
.then((row) => {
|
.then((row) => {
|
||||||
if (row.id !== data.id) {
|
if (row.id !== data.id) {
|
||||||
// Sanity check that something crazy hasn't happened
|
// Sanity check that something crazy hasn't happened
|
||||||
throw new error.InternalValidationError('Certificate could not be updated, IDs do not match: ' + row.id + ' !== ' + data.id);
|
throw new error.InternalValidationError(`Certificate could not be updated, IDs do not match: ${row.id} !== ${data.id}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return certificateModel
|
return certificateModel
|
||||||
@@ -308,7 +308,7 @@ const internalCertificate = {
|
|||||||
|
|
||||||
return access.can('certificates:get', data.id)
|
return access.can('certificates:get', data.id)
|
||||||
.then((access_data) => {
|
.then((access_data) => {
|
||||||
let query = certificateModel
|
const query = certificateModel
|
||||||
.query()
|
.query()
|
||||||
.where('is_deleted', 0)
|
.where('is_deleted', 0)
|
||||||
.andWhere('id', data.id)
|
.andWhere('id', data.id)
|
||||||
@@ -323,7 +323,7 @@ const internalCertificate = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (typeof data.expand !== 'undefined' && data.expand !== null) {
|
if (typeof data.expand !== 'undefined' && data.expand !== null) {
|
||||||
query.withGraphFetched('[' + data.expand.join(', ') + ']');
|
query.withGraphFetched(`[${data.expand.join(', ')}]`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return query.then(utils.omitRow(omissions()));
|
return query.then(utils.omitRow(omissions()));
|
||||||
@@ -354,17 +354,17 @@ const internalCertificate = {
|
|||||||
})
|
})
|
||||||
.then((certificate) => {
|
.then((certificate) => {
|
||||||
if (certificate.provider === 'letsencrypt') {
|
if (certificate.provider === 'letsencrypt') {
|
||||||
const zipDirectory = '/etc/letsencrypt/live/npm-' + data.id;
|
const zipDirectory = internalCertificate.getLiveCertPath(data.id);
|
||||||
|
|
||||||
if (!fs.existsSync(zipDirectory)) {
|
if (!fs.existsSync(zipDirectory)) {
|
||||||
throw new error.ItemNotFoundError('Certificate ' + certificate.nice_name + ' does not exists');
|
throw new error.ItemNotFoundError(`Certificate ${certificate.nice_name} does not exists`);
|
||||||
}
|
}
|
||||||
|
|
||||||
let certFiles = fs.readdirSync(zipDirectory)
|
const certFiles = fs.readdirSync(zipDirectory)
|
||||||
.filter((fn) => fn.endsWith('.pem'))
|
.filter((fn) => fn.endsWith('.pem'))
|
||||||
.map((fn) => fs.realpathSync(path.join(zipDirectory, fn)));
|
.map((fn) => fs.realpathSync(path.join(zipDirectory, fn)));
|
||||||
const downloadName = 'npm-' + data.id + '-' + `${Date.now()}.zip`;
|
const downloadName = `npm-${data.id}-${Date.now()}.zip`;
|
||||||
const opName = '/tmp/' + downloadName;
|
const opName = `/tmp/${downloadName}`;
|
||||||
internalCertificate.zipFiles(certFiles, opName)
|
internalCertificate.zipFiles(certFiles, opName)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
logger.debug('zip completed : ', opName);
|
logger.debug('zip completed : ', opName);
|
||||||
@@ -392,7 +392,7 @@ const internalCertificate = {
|
|||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
source
|
source
|
||||||
.map((fl) => {
|
.map((fl) => {
|
||||||
let fileName = path.basename(fl);
|
const fileName = path.basename(fl);
|
||||||
logger.debug(fl, 'added to certificate zip');
|
logger.debug(fl, 'added to certificate zip');
|
||||||
archive.file(fl, { name: fileName });
|
archive.file(fl, { name: fileName });
|
||||||
});
|
});
|
||||||
@@ -462,7 +462,7 @@ const internalCertificate = {
|
|||||||
getAll: (access, expand, search_query) => {
|
getAll: (access, expand, search_query) => {
|
||||||
return access.can('certificates:list')
|
return access.can('certificates:list')
|
||||||
.then((access_data) => {
|
.then((access_data) => {
|
||||||
let query = certificateModel
|
const query = certificateModel
|
||||||
.query()
|
.query()
|
||||||
.where('is_deleted', 0)
|
.where('is_deleted', 0)
|
||||||
.groupBy('id')
|
.groupBy('id')
|
||||||
@@ -479,12 +479,12 @@ const internalCertificate = {
|
|||||||
// Query is used for searching
|
// Query is used for searching
|
||||||
if (typeof search_query === 'string') {
|
if (typeof search_query === 'string') {
|
||||||
query.where(function () {
|
query.where(function () {
|
||||||
this.where('nice_name', 'like', '%' + search_query + '%');
|
this.where('nice_name', 'like', `%${search_query}%`);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof expand !== 'undefined' && expand !== null) {
|
if (typeof expand !== 'undefined' && expand !== null) {
|
||||||
query.withGraphFetched('[' + expand.join(', ') + ']');
|
query.withGraphFetched(`[${expand.join(', ')}]`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return query.then(utils.omitRows(omissions()));
|
return query.then(utils.omitRows(omissions()));
|
||||||
@@ -499,7 +499,7 @@ const internalCertificate = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
getCount: (user_id, visibility) => {
|
getCount: (user_id, visibility) => {
|
||||||
let query = certificateModel
|
const query = certificateModel
|
||||||
.query()
|
.query()
|
||||||
.count('id as count')
|
.count('id as count')
|
||||||
.where('is_deleted', 0);
|
.where('is_deleted', 0);
|
||||||
@@ -521,7 +521,7 @@ const internalCertificate = {
|
|||||||
writeCustomCert: (certificate) => {
|
writeCustomCert: (certificate) => {
|
||||||
logger.info('Writing Custom Certificate:', certificate);
|
logger.info('Writing Custom Certificate:', certificate);
|
||||||
|
|
||||||
const dir = '/data/custom_ssl/npm-' + certificate.id;
|
const dir = `/data/custom_ssl/npm-${certificate.id}`;
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
if (certificate.provider === 'letsencrypt') {
|
if (certificate.provider === 'letsencrypt') {
|
||||||
@@ -531,7 +531,7 @@ const internalCertificate = {
|
|||||||
|
|
||||||
let certData = certificate.meta.certificate;
|
let certData = certificate.meta.certificate;
|
||||||
if (typeof certificate.meta.intermediate_certificate !== 'undefined') {
|
if (typeof certificate.meta.intermediate_certificate !== 'undefined') {
|
||||||
certData = certData + '\n' + certificate.meta.intermediate_certificate;
|
certData = `${certData}\n${certificate.meta.intermediate_certificate}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -543,7 +543,7 @@ const internalCertificate = {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
fs.writeFile(dir + '/fullchain.pem', certData, function (err) {
|
fs.writeFile(`${dir}/fullchain.pem`, certData, (err) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
reject(err);
|
reject(err);
|
||||||
} else {
|
} else {
|
||||||
@@ -553,7 +553,7 @@ const internalCertificate = {
|
|||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
fs.writeFile(dir + '/privkey.pem', certificate.meta.certificate_key, function (err) {
|
fs.writeFile(`${dir}/privkey.pem`, certificate.meta.certificate_key, (err) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
reject(err);
|
reject(err);
|
||||||
} else {
|
} else {
|
||||||
@@ -591,7 +591,7 @@ const internalCertificate = {
|
|||||||
validate: (data) => {
|
validate: (data) => {
|
||||||
return new Promise((resolve) => {
|
return new Promise((resolve) => {
|
||||||
// Put file contents into an object
|
// Put file contents into an object
|
||||||
let files = {};
|
const files = {};
|
||||||
_.map(data.files, (file, name) => {
|
_.map(data.files, (file, name) => {
|
||||||
if (internalCertificate.allowedSslFiles.indexOf(name) !== -1) {
|
if (internalCertificate.allowedSslFiles.indexOf(name) !== -1) {
|
||||||
files[name] = file.data.toString();
|
files[name] = file.data.toString();
|
||||||
@@ -603,7 +603,7 @@ const internalCertificate = {
|
|||||||
.then((files) => {
|
.then((files) => {
|
||||||
// For each file, create a temp file and write the contents to it
|
// For each file, create a temp file and write the contents to it
|
||||||
// Then test it depending on the file type
|
// Then test it depending on the file type
|
||||||
let promises = [];
|
const promises = [];
|
||||||
_.map(files, (content, type) => {
|
_.map(files, (content, type) => {
|
||||||
promises.push(new Promise((resolve) => {
|
promises.push(new Promise((resolve) => {
|
||||||
if (type === 'certificate_key') {
|
if (type === 'certificate_key') {
|
||||||
@@ -688,11 +688,11 @@ const internalCertificate = {
|
|||||||
reject(new error.ValidationError('Result Validation Error: Validation timed out. This could be due to the key being passphrase-protected.'));
|
reject(new error.ValidationError('Result Validation Error: Validation timed out. This could be due to the key being passphrase-protected.'));
|
||||||
}, 10000);
|
}, 10000);
|
||||||
utils
|
utils
|
||||||
.exec('openssl pkey -in ' + filepath + ' -check -noout 2>&1 ')
|
.exec(`openssl pkey -in ${filepath} -check -noout 2>&1 `)
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
clearTimeout(failTimeout);
|
clearTimeout(failTimeout);
|
||||||
if (!result.toLowerCase().includes('key is valid')) {
|
if (!result.toLowerCase().includes('key is valid')) {
|
||||||
reject(new error.ValidationError('Result Validation Error: ' + result));
|
reject(new error.ValidationError(`Result Validation Error: ${result}`));
|
||||||
}
|
}
|
||||||
fs.unlinkSync(filepath);
|
fs.unlinkSync(filepath);
|
||||||
resolve(true);
|
resolve(true);
|
||||||
@@ -700,7 +700,7 @@ const internalCertificate = {
|
|||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
clearTimeout(failTimeout);
|
clearTimeout(failTimeout);
|
||||||
fs.unlinkSync(filepath);
|
fs.unlinkSync(filepath);
|
||||||
reject(new error.ValidationError('Certificate Key is not valid (' + err.message + ')', err));
|
reject(new error.ValidationError(`Certificate Key is not valid (${err.message})`, err));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -735,9 +735,9 @@ const internalCertificate = {
|
|||||||
* @param {Boolean} [throw_expired] Throw when the certificate is out of date
|
* @param {Boolean} [throw_expired] Throw when the certificate is out of date
|
||||||
*/
|
*/
|
||||||
getCertificateInfoFromFile: (certificate_file, throw_expired) => {
|
getCertificateInfoFromFile: (certificate_file, throw_expired) => {
|
||||||
let certData = {};
|
const certData = {};
|
||||||
|
|
||||||
return utils.exec('openssl x509 -in ' + certificate_file + ' -subject -noout')
|
return utils.execFile('openssl', ['x509', '-in', certificate_file, '-subject', '-noout'])
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
// Examples:
|
// Examples:
|
||||||
// subject=CN = *.jc21.com
|
// subject=CN = *.jc21.com
|
||||||
@@ -745,11 +745,11 @@ const internalCertificate = {
|
|||||||
const regex = /(?:subject=)?[^=]+=\s+(\S+)/gim;
|
const regex = /(?:subject=)?[^=]+=\s+(\S+)/gim;
|
||||||
const match = regex.exec(result);
|
const match = regex.exec(result);
|
||||||
if (match && typeof match[1] !== 'undefined') {
|
if (match && typeof match[1] !== 'undefined') {
|
||||||
certData['cn'] = match[1];
|
certData.cn = match[1];
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
return utils.exec('openssl x509 -in ' + certificate_file + ' -issuer -noout');
|
return utils.execFile('openssl', ['x509', '-in', certificate_file, '-issuer', '-noout']);
|
||||||
})
|
})
|
||||||
|
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
@@ -760,11 +760,11 @@ const internalCertificate = {
|
|||||||
const regex = /^(?:issuer=)?(.*)$/gim;
|
const regex = /^(?:issuer=)?(.*)$/gim;
|
||||||
const match = regex.exec(result);
|
const match = regex.exec(result);
|
||||||
if (match && typeof match[1] !== 'undefined') {
|
if (match && typeof match[1] !== 'undefined') {
|
||||||
certData['issuer'] = match[1];
|
certData.issuer = match[1];
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
return utils.exec('openssl x509 -in ' + certificate_file + ' -dates -noout');
|
return utils.execFile('openssl', ['x509', '-in', certificate_file, '-dates', '-noout']);
|
||||||
})
|
})
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
// notBefore=Jul 14 04:04:29 2018 GMT
|
// notBefore=Jul 14 04:04:29 2018 GMT
|
||||||
@@ -773,7 +773,7 @@ const internalCertificate = {
|
|||||||
let validTo = null;
|
let validTo = null;
|
||||||
|
|
||||||
const lines = result.split('\n');
|
const lines = result.split('\n');
|
||||||
lines.map(function (str) {
|
lines.map((str) => {
|
||||||
const regex = /^(\S+)=(.*)$/gim;
|
const regex = /^(\S+)=(.*)$/gim;
|
||||||
const match = regex.exec(str.trim());
|
const match = regex.exec(str.trim());
|
||||||
|
|
||||||
@@ -789,21 +789,21 @@ const internalCertificate = {
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (!validFrom || !validTo) {
|
if (!validFrom || !validTo) {
|
||||||
throw new error.ValidationError('Could not determine dates from certificate: ' + result);
|
throw new error.ValidationError(`Could not determine dates from certificate: ${result}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (throw_expired && validTo < parseInt(moment().format('X'), 10)) {
|
if (throw_expired && validTo < parseInt(moment().format('X'), 10)) {
|
||||||
throw new error.ValidationError('Certificate has expired');
|
throw new error.ValidationError('Certificate has expired');
|
||||||
}
|
}
|
||||||
|
|
||||||
certData['dates'] = {
|
certData.dates = {
|
||||||
from: validFrom,
|
from: validFrom,
|
||||||
to: validTo
|
to: validTo
|
||||||
};
|
};
|
||||||
|
|
||||||
return certData;
|
return certData;
|
||||||
}).catch((err) => {
|
}).catch((err) => {
|
||||||
throw new error.ValidationError('Certificate is not valid (' + err.message + ')', err);
|
throw new error.ValidationError(`Certificate is not valid (${err.message})`, err);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -814,7 +814,7 @@ const internalCertificate = {
|
|||||||
* @param {Boolean} [remove]
|
* @param {Boolean} [remove]
|
||||||
* @returns {Object}
|
* @returns {Object}
|
||||||
*/
|
*/
|
||||||
cleanMeta: function (meta, remove) {
|
cleanMeta: (meta, remove) => {
|
||||||
internalCertificate.allowedSslFiles.map((key) => {
|
internalCertificate.allowedSslFiles.map((key) => {
|
||||||
if (typeof meta[key] !== 'undefined' && meta[key]) {
|
if (typeof meta[key] !== 'undefined' && meta[key]) {
|
||||||
if (remove) {
|
if (remove) {
|
||||||
@@ -834,24 +834,35 @@ const internalCertificate = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
requestLetsEncryptSsl: (certificate) => {
|
requestLetsEncryptSsl: (certificate) => {
|
||||||
logger.info('Requesting Let\'sEncrypt certificates for Cert #' + certificate.id + ': ' + certificate.domain_names.join(', '));
|
logger.info(`Requesting LetsEncrypt certificates for Cert #${certificate.id}: ${certificate.domain_names.join(', ')}`);
|
||||||
|
|
||||||
const cmd = `${certbotCommand} certonly ` +
|
const args = [
|
||||||
`--config '${letsencryptConfig}' ` +
|
'certonly',
|
||||||
'--work-dir "/tmp/letsencrypt-lib" ' +
|
'--config',
|
||||||
'--logs-dir "/tmp/letsencrypt-log" ' +
|
letsencryptConfig,
|
||||||
`--cert-name "npm-${certificate.id}" ` +
|
'--work-dir',
|
||||||
'--agree-tos ' +
|
'/tmp/letsencrypt-lib',
|
||||||
'--authenticator webroot ' +
|
'--logs-dir',
|
||||||
`--email '${certificate.meta.letsencrypt_email}' ` +
|
'/tmp/letsencrypt-log',
|
||||||
'--preferred-challenges "dns,http" ' +
|
'--cert-name',
|
||||||
`--domains "${certificate.domain_names.join(',')}" ` +
|
`npm-${certificate.id}`,
|
||||||
(letsencryptServer !== null ? `--server '${letsencryptServer}' ` : '') +
|
'--agree-tos',
|
||||||
(letsencryptStaging && letsencryptServer === null ? '--staging ' : '');
|
'--authenticator',
|
||||||
|
'webroot',
|
||||||
|
'--email',
|
||||||
|
certificate.meta.letsencrypt_email,
|
||||||
|
'--preferred-challenges',
|
||||||
|
'dns,http',
|
||||||
|
'--domains',
|
||||||
|
certificate.domain_names.join(','),
|
||||||
|
];
|
||||||
|
|
||||||
logger.info('Command:', cmd);
|
const adds = internalCertificate.getAdditionalCertbotArgs(certificate.id);
|
||||||
|
args.push(...adds.args);
|
||||||
|
|
||||||
return utils.exec(cmd)
|
logger.info(`Command: ${certbotCommand} ${args ? args.join(' ') : ''}`);
|
||||||
|
|
||||||
|
return utils.execFile(certbotCommand, args, adds.opts)
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
logger.success(result);
|
logger.success(result);
|
||||||
return result;
|
return result;
|
||||||
@@ -868,50 +879,48 @@ const internalCertificate = {
|
|||||||
requestLetsEncryptSslWithDnsChallenge: async (certificate) => {
|
requestLetsEncryptSslWithDnsChallenge: async (certificate) => {
|
||||||
await certbot.installPlugin(certificate.meta.dns_provider);
|
await certbot.installPlugin(certificate.meta.dns_provider);
|
||||||
const dnsPlugin = dnsPlugins[certificate.meta.dns_provider];
|
const dnsPlugin = dnsPlugins[certificate.meta.dns_provider];
|
||||||
logger.info(`Requesting Let'sEncrypt certificates via ${dnsPlugin.name} for Cert #${certificate.id}: ${certificate.domain_names.join(', ')}`);
|
logger.info(`Requesting LetsEncrypt certificates via ${dnsPlugin.name} for Cert #${certificate.id}: ${certificate.domain_names.join(', ')}`);
|
||||||
|
|
||||||
const credentialsLocation = '/etc/letsencrypt/credentials/credentials-' + certificate.id;
|
const credentialsLocation = `/etc/letsencrypt/credentials/credentials-${certificate.id}`;
|
||||||
fs.mkdirSync('/etc/letsencrypt/credentials', { recursive: true });
|
fs.mkdirSync('/etc/letsencrypt/credentials', { recursive: true });
|
||||||
fs.writeFileSync(credentialsLocation, certificate.meta.dns_provider_credentials, {mode: 0o600});
|
fs.writeFileSync(credentialsLocation, certificate.meta.dns_provider_credentials, {mode: 0o600});
|
||||||
|
|
||||||
// Whether the plugin has a --<name>-credentials argument
|
// Whether the plugin has a --<name>-credentials argument
|
||||||
const hasConfigArg = certificate.meta.dns_provider !== 'route53';
|
const hasConfigArg = certificate.meta.dns_provider !== 'route53';
|
||||||
|
|
||||||
let mainCmd = certbotCommand + ' certonly ' +
|
const args = [
|
||||||
`--config '${letsencryptConfig}' ` +
|
'certonly',
|
||||||
'--work-dir "/tmp/letsencrypt-lib" ' +
|
'--config',
|
||||||
'--logs-dir "/tmp/letsencrypt-log" ' +
|
letsencryptConfig,
|
||||||
`--cert-name 'npm-${certificate.id}' ` +
|
'--work-dir',
|
||||||
'--agree-tos ' +
|
'/tmp/letsencrypt-lib',
|
||||||
`--email '${certificate.meta.letsencrypt_email}' ` +
|
'--logs-dir',
|
||||||
`--domains '${certificate.domain_names.join(',')}' ` +
|
'/tmp/letsencrypt-log',
|
||||||
`--authenticator '${dnsPlugin.full_plugin_name}' ` +
|
'--cert-name',
|
||||||
(
|
`npm-${certificate.id}`,
|
||||||
hasConfigArg
|
'--agree-tos',
|
||||||
? `--${dnsPlugin.full_plugin_name}-credentials '${credentialsLocation}' `
|
'--email',
|
||||||
: ''
|
certificate.meta.letsencrypt_email,
|
||||||
) +
|
'--domains',
|
||||||
(
|
certificate.domain_names.join(','),
|
||||||
certificate.meta.propagation_seconds !== undefined
|
'--authenticator',
|
||||||
? `--${dnsPlugin.full_plugin_name}-propagation-seconds '${certificate.meta.propagation_seconds}' `
|
dnsPlugin.full_plugin_name,
|
||||||
: ''
|
];
|
||||||
) +
|
|
||||||
(letsencryptServer !== null ? `--server '${letsencryptServer}' ` : '') +
|
|
||||||
(letsencryptStaging && letsencryptServer === null ? '--staging ' : '');
|
|
||||||
|
|
||||||
// Prepend the path to the credentials file as an environment variable
|
if (hasConfigArg) {
|
||||||
if (certificate.meta.dns_provider === 'route53') {
|
args.push(`--${dnsPlugin.full_plugin_name}-credentials`, credentialsLocation);
|
||||||
mainCmd = 'AWS_CONFIG_FILE=\'' + credentialsLocation + '\' ' + mainCmd;
|
}
|
||||||
|
if (certificate.meta.propagation_seconds !== undefined) {
|
||||||
|
args.push(`--${dnsPlugin.full_plugin_name}-propagation-seconds`, certificate.meta.propagation_seconds.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (certificate.meta.dns_provider === 'duckdns') {
|
const adds = internalCertificate.getAdditionalCertbotArgs(certificate.id, certificate.meta.dns_provider);
|
||||||
mainCmd = mainCmd + ' --dns-duckdns-no-txt-restore';
|
args.push(...adds.args);
|
||||||
}
|
|
||||||
|
|
||||||
logger.info('Command:', mainCmd);
|
logger.info(`Command: ${certbotCommand} ${args ? args.join(' ') : ''}`);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const result = await utils.exec(mainCmd);
|
const result = await utils.execFile(certbotCommand, args, adds.opts);
|
||||||
logger.info(result);
|
logger.info(result);
|
||||||
return result;
|
return result;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@@ -939,7 +948,7 @@ const internalCertificate = {
|
|||||||
|
|
||||||
return renewMethod(certificate)
|
return renewMethod(certificate)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
return internalCertificate.getCertificateInfoFromFile('/etc/letsencrypt/live/npm-' + certificate.id + '/fullchain.pem');
|
return internalCertificate.getCertificateInfoFromFile(`${internalCertificate.getLiveCertPath(certificate.id)}/fullchain.pem`);
|
||||||
})
|
})
|
||||||
.then((cert_info) => {
|
.then((cert_info) => {
|
||||||
return certificateModel
|
return certificateModel
|
||||||
@@ -971,22 +980,31 @@ const internalCertificate = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
renewLetsEncryptSsl: (certificate) => {
|
renewLetsEncryptSsl: (certificate) => {
|
||||||
logger.info('Renewing Let\'sEncrypt certificates for Cert #' + certificate.id + ': ' + certificate.domain_names.join(', '));
|
logger.info(`Renewing LetsEncrypt certificates for Cert #${certificate.id}: ${certificate.domain_names.join(', ')}`);
|
||||||
|
|
||||||
const cmd = certbotCommand + ' renew --force-renewal ' +
|
const args = [
|
||||||
`--config '${letsencryptConfig}' ` +
|
'renew',
|
||||||
'--work-dir "/tmp/letsencrypt-lib" ' +
|
'--force-renewal',
|
||||||
'--logs-dir "/tmp/letsencrypt-log" ' +
|
'--config',
|
||||||
`--cert-name 'npm-${certificate.id}' ` +
|
letsencryptConfig,
|
||||||
'--preferred-challenges "dns,http" ' +
|
'--work-dir',
|
||||||
'--no-random-sleep-on-renew ' +
|
'/tmp/letsencrypt-lib',
|
||||||
'--disable-hook-validation ' +
|
'--logs-dir',
|
||||||
(letsencryptServer !== null ? `--server '${letsencryptServer}' ` : '') +
|
'/tmp/letsencrypt-log',
|
||||||
(letsencryptStaging && letsencryptServer === null ? '--staging ' : '');
|
'--cert-name',
|
||||||
|
`npm-${certificate.id}`,
|
||||||
|
'--preferred-challenges',
|
||||||
|
'dns,http',
|
||||||
|
'--no-random-sleep-on-renew',
|
||||||
|
'--disable-hook-validation',
|
||||||
|
];
|
||||||
|
|
||||||
logger.info('Command:', cmd);
|
const adds = internalCertificate.getAdditionalCertbotArgs(certificate.id, certificate.meta.dns_provider);
|
||||||
|
args.push(...adds.args);
|
||||||
|
|
||||||
return utils.exec(cmd)
|
logger.info(`Command: ${certbotCommand} ${args ? args.join(' ') : ''}`);
|
||||||
|
|
||||||
|
return utils.execFile(certbotCommand, args, adds.opts)
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
logger.info(result);
|
logger.info(result);
|
||||||
return result;
|
return result;
|
||||||
@@ -1004,27 +1022,29 @@ const internalCertificate = {
|
|||||||
throw Error(`Unknown DNS provider '${certificate.meta.dns_provider}'`);
|
throw Error(`Unknown DNS provider '${certificate.meta.dns_provider}'`);
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info(`Renewing Let'sEncrypt certificates via ${dnsPlugin.name} for Cert #${certificate.id}: ${certificate.domain_names.join(', ')}`);
|
logger.info(`Renewing LetsEncrypt certificates via ${dnsPlugin.name} for Cert #${certificate.id}: ${certificate.domain_names.join(', ')}`);
|
||||||
|
|
||||||
let mainCmd = certbotCommand + ' renew --force-renewal ' +
|
const args = [
|
||||||
`--config "${letsencryptConfig}" ` +
|
'renew',
|
||||||
'--work-dir "/tmp/letsencrypt-lib" ' +
|
'--force-renewal',
|
||||||
'--logs-dir "/tmp/letsencrypt-log" ' +
|
'--config',
|
||||||
`--cert-name 'npm-${certificate.id}' ` +
|
letsencryptConfig,
|
||||||
'--disable-hook-validation ' +
|
'--work-dir',
|
||||||
'--no-random-sleep-on-renew ' +
|
'/tmp/letsencrypt-lib',
|
||||||
(letsencryptServer !== null ? `--server '${letsencryptServer}' ` : '') +
|
'--logs-dir',
|
||||||
(letsencryptStaging && letsencryptServer === null ? '--staging ' : '');
|
'/tmp/letsencrypt-log',
|
||||||
|
'--cert-name',
|
||||||
|
`npm-${certificate.id}`,
|
||||||
|
'--disable-hook-validation',
|
||||||
|
'--no-random-sleep-on-renew',
|
||||||
|
];
|
||||||
|
|
||||||
// Prepend the path to the credentials file as an environment variable
|
const adds = internalCertificate.getAdditionalCertbotArgs(certificate.id, certificate.meta.dns_provider);
|
||||||
if (certificate.meta.dns_provider === 'route53') {
|
args.push(...adds.args);
|
||||||
const credentialsLocation = '/etc/letsencrypt/credentials/credentials-' + certificate.id;
|
|
||||||
mainCmd = 'AWS_CONFIG_FILE=\'' + credentialsLocation + '\' ' + mainCmd;
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.info('Command:', mainCmd);
|
logger.info(`Command: ${certbotCommand} ${args ? args.join(' ') : ''}`);
|
||||||
|
|
||||||
return utils.exec(mainCmd)
|
return utils.execFile(certbotCommand, args, adds.opts)
|
||||||
.then(async (result) => {
|
.then(async (result) => {
|
||||||
logger.info(result);
|
logger.info(result);
|
||||||
return result;
|
return result;
|
||||||
@@ -1037,25 +1057,29 @@ const internalCertificate = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
revokeLetsEncryptSsl: (certificate, throw_errors) => {
|
revokeLetsEncryptSsl: (certificate, throw_errors) => {
|
||||||
logger.info('Revoking Let\'sEncrypt certificates for Cert #' + certificate.id + ': ' + certificate.domain_names.join(', '));
|
logger.info(`Revoking LetsEncrypt certificates for Cert #${certificate.id}: ${certificate.domain_names.join(', ')}`);
|
||||||
|
|
||||||
const mainCmd = certbotCommand + ' revoke ' +
|
const args = [
|
||||||
`--config '${letsencryptConfig}' ` +
|
'revoke',
|
||||||
'--work-dir "/tmp/letsencrypt-lib" ' +
|
'--config',
|
||||||
'--logs-dir "/tmp/letsencrypt-log" ' +
|
letsencryptConfig,
|
||||||
`--cert-path '/etc/letsencrypt/live/npm-${certificate.id}/fullchain.pem' ` +
|
'--work-dir',
|
||||||
'--delete-after-revoke ' +
|
'/tmp/letsencrypt-lib',
|
||||||
(letsencryptServer !== null ? `--server '${letsencryptServer}' ` : '') +
|
'--logs-dir',
|
||||||
(letsencryptStaging && letsencryptServer === null ? '--staging ' : '');
|
'/tmp/letsencrypt-log',
|
||||||
|
'--cert-path',
|
||||||
|
`${internalCertificate.getLiveCertPath(certificate.id)}/fullchain.pem`,
|
||||||
|
'--delete-after-revoke',
|
||||||
|
];
|
||||||
|
|
||||||
// Don't fail command if file does not exist
|
const adds = internalCertificate.getAdditionalCertbotArgs(certificate.id);
|
||||||
const delete_credentialsCmd = `rm -f '/etc/letsencrypt/credentials/credentials-${certificate.id}' || true`;
|
args.push(...adds.args);
|
||||||
|
|
||||||
logger.info('Command:', mainCmd + '; ' + delete_credentialsCmd);
|
logger.info(`Command: ${certbotCommand} ${args ? args.join(' ') : ''}`);
|
||||||
|
|
||||||
return utils.exec(mainCmd)
|
return utils.execFile(certbotCommand, args, adds.opts)
|
||||||
.then(async (result) => {
|
.then(async (result) => {
|
||||||
await utils.exec(delete_credentialsCmd);
|
await utils.exec(`rm -f '/etc/letsencrypt/credentials/credentials-${certificate.id}' || true`);
|
||||||
logger.info(result);
|
logger.info(result);
|
||||||
return result;
|
return result;
|
||||||
})
|
})
|
||||||
@@ -1073,9 +1097,8 @@ const internalCertificate = {
|
|||||||
* @returns {Boolean}
|
* @returns {Boolean}
|
||||||
*/
|
*/
|
||||||
hasLetsEncryptSslCerts: (certificate) => {
|
hasLetsEncryptSslCerts: (certificate) => {
|
||||||
const letsencryptPath = '/etc/letsencrypt/live/npm-' + certificate.id;
|
const letsencryptPath = internalCertificate.getLiveCertPath(certificate.id);
|
||||||
|
return fs.existsSync(`${letsencryptPath}/fullchain.pem`) && fs.existsSync(`${letsencryptPath}/privkey.pem`);
|
||||||
return fs.existsSync(letsencryptPath + '/fullchain.pem') && fs.existsSync(letsencryptPath + '/privkey.pem');
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1087,7 +1110,7 @@ const internalCertificate = {
|
|||||||
*/
|
*/
|
||||||
disableInUseHosts: (in_use_result) => {
|
disableInUseHosts: (in_use_result) => {
|
||||||
if (in_use_result.total_count) {
|
if (in_use_result.total_count) {
|
||||||
let promises = [];
|
const promises = [];
|
||||||
|
|
||||||
if (in_use_result.proxy_hosts.length) {
|
if (in_use_result.proxy_hosts.length) {
|
||||||
promises.push(internalNginx.bulkDeleteConfigs('proxy_host', in_use_result.proxy_hosts));
|
promises.push(internalNginx.bulkDeleteConfigs('proxy_host', in_use_result.proxy_hosts));
|
||||||
@@ -1117,7 +1140,7 @@ const internalCertificate = {
|
|||||||
*/
|
*/
|
||||||
enableInUseHosts: (in_use_result) => {
|
enableInUseHosts: (in_use_result) => {
|
||||||
if (in_use_result.total_count) {
|
if (in_use_result.total_count) {
|
||||||
let promises = [];
|
const promises = [];
|
||||||
|
|
||||||
if (in_use_result.proxy_hosts.length) {
|
if (in_use_result.proxy_hosts.length) {
|
||||||
promises.push(internalNginx.bulkGenerateConfigs('proxy_host', in_use_result.proxy_hosts));
|
promises.push(internalNginx.bulkGenerateConfigs('proxy_host', in_use_result.proxy_hosts));
|
||||||
@@ -1150,12 +1173,12 @@ const internalCertificate = {
|
|||||||
|
|
||||||
// Create a test challenge file
|
// Create a test challenge file
|
||||||
const testChallengeDir = '/data/letsencrypt-acme-challenge/.well-known/acme-challenge';
|
const testChallengeDir = '/data/letsencrypt-acme-challenge/.well-known/acme-challenge';
|
||||||
const testChallengeFile = testChallengeDir + '/test-challenge';
|
const testChallengeFile = `${testChallengeDir}/test-challenge`;
|
||||||
fs.mkdirSync(testChallengeDir, {recursive: true});
|
fs.mkdirSync(testChallengeDir, {recursive: true});
|
||||||
fs.writeFileSync(testChallengeFile, 'Success', {encoding: 'utf8'});
|
fs.writeFileSync(testChallengeFile, 'Success', {encoding: 'utf8'});
|
||||||
|
|
||||||
async function performTestForDomain (domain) {
|
async function performTestForDomain (domain) {
|
||||||
logger.info('Testing http challenge for ' + domain);
|
logger.info(`Testing http challenge for ${domain}`);
|
||||||
const url = `http://${domain}/.well-known/acme-challenge/test-challenge`;
|
const url = `http://${domain}/.well-known/acme-challenge/test-challenge`;
|
||||||
const formBody = `method=G&url=${encodeURI(url)}&bodytype=T&requestbody=&headername=User-Agent&headervalue=None&locationid=1&ch=false&cc=false`;
|
const formBody = `method=G&url=${encodeURI(url)}&bodytype=T&requestbody=&headername=User-Agent&headervalue=None&locationid=1&ch=false&cc=false`;
|
||||||
const options = {
|
const options = {
|
||||||
@@ -1169,13 +1192,16 @@ const internalCertificate = {
|
|||||||
|
|
||||||
const result = await new Promise((resolve) => {
|
const result = await new Promise((resolve) => {
|
||||||
|
|
||||||
const req = https.request('https://www.site24x7.com/tools/restapi-tester', options, function (res) {
|
const req = https.request('https://www.site24x7.com/tools/restapi-tester', options, (res) => {
|
||||||
let responseBody = '';
|
let responseBody = '';
|
||||||
|
|
||||||
res.on('data', (chunk) => responseBody = responseBody + chunk);
|
res.on('data', (chunk) => {
|
||||||
res.on('end', function () {
|
responseBody = responseBody + chunk;
|
||||||
|
});
|
||||||
|
|
||||||
|
res.on('end', () => {
|
||||||
try {
|
try {
|
||||||
const parsedBody = JSON.parse(responseBody + '');
|
const parsedBody = JSON.parse(`${responseBody}`);
|
||||||
if (res.statusCode !== 200) {
|
if (res.statusCode !== 200) {
|
||||||
logger.warn(`Failed to test HTTP challenge for domain ${domain} because HTTP status code ${res.statusCode} was returned: ${parsedBody.message}`);
|
logger.warn(`Failed to test HTTP challenge for domain ${domain} because HTTP status code ${res.statusCode} was returned: ${parsedBody.message}`);
|
||||||
resolve(undefined);
|
resolve(undefined);
|
||||||
@@ -1196,7 +1222,7 @@ const internalCertificate = {
|
|||||||
// Make sure to write the request body.
|
// Make sure to write the request body.
|
||||||
req.write(formBody);
|
req.write(formBody);
|
||||||
req.end();
|
req.end();
|
||||||
req.on('error', function (e) { logger.warn(`Failed to test HTTP challenge for domain ${domain}`, e);
|
req.on('error', (e) => { logger.warn(`Failed to test HTTP challenge for domain ${domain}`, e);
|
||||||
resolve(undefined); });
|
resolve(undefined); });
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1238,6 +1264,34 @@ const internalCertificate = {
|
|||||||
fs.unlinkSync(testChallengeFile);
|
fs.unlinkSync(testChallengeFile);
|
||||||
|
|
||||||
return results;
|
return results;
|
||||||
|
},
|
||||||
|
|
||||||
|
getAdditionalCertbotArgs: (certificate_id, dns_provider) => {
|
||||||
|
const args = [];
|
||||||
|
if (letsencryptServer !== null) {
|
||||||
|
args.push('--server', letsencryptServer);
|
||||||
|
}
|
||||||
|
if (letsencryptStaging && letsencryptServer === null) {
|
||||||
|
args.push('--staging');
|
||||||
|
}
|
||||||
|
|
||||||
|
// For route53, add the credentials file as an environment variable,
|
||||||
|
// inheriting the process env
|
||||||
|
const opts = {};
|
||||||
|
if (certificate_id && dns_provider === 'route53') {
|
||||||
|
opts.env = process.env;
|
||||||
|
opts.env.AWS_CONFIG_FILE = `/etc/letsencrypt/credentials/credentials-${certificate_id}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dns_provider === 'duckdns') {
|
||||||
|
args.push('--dns-duckdns-no-txt-restore');
|
||||||
|
}
|
||||||
|
|
||||||
|
return {args: args, opts: opts};
|
||||||
|
},
|
||||||
|
|
||||||
|
getLiveCertPath: (certificate_id) => {
|
||||||
|
return `/etc/letsencrypt/live/npm-${certificate_id}`;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
const _ = require('lodash');
|
const _ = require('lodash');
|
||||||
const fs = require('fs');
|
const fs = require('node:fs');
|
||||||
const logger = require('../logger').nginx;
|
const logger = require('../logger').nginx;
|
||||||
const config = require('../lib/config');
|
const config = require('../lib/config');
|
||||||
const utils = require('../lib/utils');
|
const utils = require('../lib/utils');
|
||||||
@@ -57,9 +57,9 @@ const internalNginx = {
|
|||||||
// It will always look like this:
|
// It will always look like this:
|
||||||
// nginx: [alert] could not open error log file: open() "/var/log/nginx/error.log" failed (6: No such device or address)
|
// nginx: [alert] could not open error log file: open() "/var/log/nginx/error.log" failed (6: No such device or address)
|
||||||
|
|
||||||
let valid_lines = [];
|
const valid_lines = [];
|
||||||
let err_lines = err.message.split('\n');
|
const err_lines = err.message.split('\n');
|
||||||
err_lines.map(function (line) {
|
err_lines.map((line) => {
|
||||||
if (line.indexOf('/var/log/nginx/error.log') === -1) {
|
if (line.indexOf('/var/log/nginx/error.log') === -1) {
|
||||||
valid_lines.push(line);
|
valid_lines.push(line);
|
||||||
}
|
}
|
||||||
@@ -105,7 +105,7 @@ const internalNginx = {
|
|||||||
logger.info('Testing Nginx configuration');
|
logger.info('Testing Nginx configuration');
|
||||||
}
|
}
|
||||||
|
|
||||||
return utils.exec('/usr/sbin/nginx -t -g "error_log off;"');
|
return utils.execFile('/usr/sbin/nginx', ['-t', '-g', 'error_log off;']);
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -115,7 +115,7 @@ const internalNginx = {
|
|||||||
return internalNginx.test()
|
return internalNginx.test()
|
||||||
.then(() => {
|
.then(() => {
|
||||||
logger.info('Reloading Nginx');
|
logger.info('Reloading Nginx');
|
||||||
return utils.exec('/usr/sbin/nginx -s reload');
|
return utils.execFile('/usr/sbin/nginx', ['-s', 'reload']);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -128,7 +128,7 @@ const internalNginx = {
|
|||||||
if (host_type === 'default') {
|
if (host_type === 'default') {
|
||||||
return '/data/nginx/default_host/site.conf';
|
return '/data/nginx/default_host/site.conf';
|
||||||
}
|
}
|
||||||
return '/data/nginx/' + internalNginx.getFileFriendlyHostType(host_type) + '/' + host_id + '.conf';
|
return `/data/nginx/${internalNginx.getFileFriendlyHostType(host_type)}/${host_id}.conf`;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -141,7 +141,7 @@ const internalNginx = {
|
|||||||
let template;
|
let template;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
template = fs.readFileSync(__dirname + '/../templates/_location.conf', {encoding: 'utf8'});
|
template = fs.readFileSync(`${__dirname}/../templates/_location.conf`, {encoding: 'utf8'});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
reject(new error.ConfigurationError(err.message));
|
reject(new error.ConfigurationError(err.message));
|
||||||
return;
|
return;
|
||||||
@@ -152,7 +152,7 @@ const internalNginx = {
|
|||||||
|
|
||||||
const locationRendering = async () => {
|
const locationRendering = async () => {
|
||||||
for (let i = 0; i < host.locations.length; i++) {
|
for (let i = 0; i < host.locations.length; i++) {
|
||||||
let locationCopy = Object.assign({}, {access_list_id: host.access_list_id}, {certificate_id: host.certificate_id},
|
const locationCopy = Object.assign({}, {access_list_id: host.access_list_id}, {certificate_id: host.certificate_id},
|
||||||
{ssl_forced: host.ssl_forced}, {caching_enabled: host.caching_enabled}, {block_exploits: host.block_exploits},
|
{ssl_forced: host.ssl_forced}, {caching_enabled: host.caching_enabled}, {block_exploits: host.block_exploits},
|
||||||
{allow_websocket_upgrade: host.allow_websocket_upgrade}, {http2_support: host.http2_support},
|
{allow_websocket_upgrade: host.allow_websocket_upgrade}, {http2_support: host.http2_support},
|
||||||
{hsts_enabled: host.hsts_enabled}, {hsts_subdomains: host.hsts_subdomains}, {access_list: host.access_list},
|
{hsts_enabled: host.hsts_enabled}, {hsts_subdomains: host.hsts_subdomains}, {access_list: host.access_list},
|
||||||
@@ -183,21 +183,21 @@ const internalNginx = {
|
|||||||
*/
|
*/
|
||||||
generateConfig: (host_type, host_row) => {
|
generateConfig: (host_type, host_row) => {
|
||||||
// Prevent modifying the original object:
|
// Prevent modifying the original object:
|
||||||
let host = JSON.parse(JSON.stringify(host_row));
|
const host = JSON.parse(JSON.stringify(host_row));
|
||||||
const nice_host_type = internalNginx.getFileFriendlyHostType(host_type);
|
const nice_host_type = internalNginx.getFileFriendlyHostType(host_type);
|
||||||
|
|
||||||
if (config.debug()) {
|
if (config.debug()) {
|
||||||
logger.info('Generating ' + nice_host_type + ' Config:', JSON.stringify(host, null, 2));
|
logger.info(`Generating ${nice_host_type} Config:`, JSON.stringify(host, null, 2));
|
||||||
}
|
}
|
||||||
|
|
||||||
const renderEngine = utils.getRenderEngine();
|
const renderEngine = utils.getRenderEngine();
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
let template = null;
|
let template = null;
|
||||||
let filename = internalNginx.getConfigName(nice_host_type, host.id);
|
const filename = internalNginx.getConfigName(nice_host_type, host.id);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
template = fs.readFileSync(__dirname + '/../templates/' + nice_host_type + '.conf', {encoding: 'utf8'});
|
template = fs.readFileSync(`${__dirname}/../templates/${nice_host_type}.conf`, {encoding: 'utf8'});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
reject(new error.ConfigurationError(err.message));
|
reject(new error.ConfigurationError(err.message));
|
||||||
return;
|
return;
|
||||||
@@ -252,7 +252,7 @@ const internalNginx = {
|
|||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
if (config.debug()) {
|
if (config.debug()) {
|
||||||
logger.warn('Could not write ' + filename + ':', err.message);
|
logger.warn(`Could not write ${filename}:`, err.message);
|
||||||
}
|
}
|
||||||
|
|
||||||
reject(new error.ConfigurationError(err.message));
|
reject(new error.ConfigurationError(err.message));
|
||||||
@@ -277,11 +277,11 @@ const internalNginx = {
|
|||||||
const renderEngine = utils.getRenderEngine();
|
const renderEngine = utils.getRenderEngine();
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
let template = null;
|
let template = null;
|
||||||
let filename = '/data/nginx/temp/letsencrypt_' + certificate.id + '.conf';
|
const filename = `/data/nginx/temp/letsencrypt_${certificate.id}.conf`;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
template = fs.readFileSync(__dirname + '/../templates/letsencrypt-request.conf', {encoding: 'utf8'});
|
template = fs.readFileSync(`${__dirname}/../templates/letsencrypt-request.conf`, {encoding: 'utf8'});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
reject(new error.ConfigurationError(err.message));
|
reject(new error.ConfigurationError(err.message));
|
||||||
return;
|
return;
|
||||||
@@ -302,7 +302,7 @@ const internalNginx = {
|
|||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
if (config.debug()) {
|
if (config.debug()) {
|
||||||
logger.warn('Could not write ' + filename + ':', err.message);
|
logger.warn(`Could not write ${filename}:`, err.message);
|
||||||
}
|
}
|
||||||
|
|
||||||
reject(new error.ConfigurationError(err.message));
|
reject(new error.ConfigurationError(err.message));
|
||||||
@@ -316,7 +316,7 @@ const internalNginx = {
|
|||||||
* @param {String} filename
|
* @param {String} filename
|
||||||
*/
|
*/
|
||||||
deleteFile: (filename) => {
|
deleteFile: (filename) => {
|
||||||
logger.debug('Deleting file: ' + filename);
|
logger.debug(`Deleting file: ${filename}`);
|
||||||
try {
|
try {
|
||||||
fs.unlinkSync(filename);
|
fs.unlinkSync(filename);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@@ -330,7 +330,7 @@ const internalNginx = {
|
|||||||
* @returns String
|
* @returns String
|
||||||
*/
|
*/
|
||||||
getFileFriendlyHostType: (host_type) => {
|
getFileFriendlyHostType: (host_type) => {
|
||||||
return host_type.replace(new RegExp('-', 'g'), '_');
|
return host_type.replace(/-/g, '_');
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -340,7 +340,7 @@ const internalNginx = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
deleteLetsEncryptRequestConfig: (certificate) => {
|
deleteLetsEncryptRequestConfig: (certificate) => {
|
||||||
const config_file = '/data/nginx/temp/letsencrypt_' + certificate.id + '.conf';
|
const config_file = `/data/nginx/temp/letsencrypt_${certificate.id}.conf`;
|
||||||
return new Promise((resolve/*, reject*/) => {
|
return new Promise((resolve/*, reject*/) => {
|
||||||
internalNginx.deleteFile(config_file);
|
internalNginx.deleteFile(config_file);
|
||||||
resolve();
|
resolve();
|
||||||
@@ -355,7 +355,7 @@ const internalNginx = {
|
|||||||
*/
|
*/
|
||||||
deleteConfig: (host_type, host, delete_err_file) => {
|
deleteConfig: (host_type, host, delete_err_file) => {
|
||||||
const config_file = internalNginx.getConfigName(internalNginx.getFileFriendlyHostType(host_type), typeof host === 'undefined' ? 0 : host.id);
|
const config_file = internalNginx.getConfigName(internalNginx.getFileFriendlyHostType(host_type), typeof host === 'undefined' ? 0 : host.id);
|
||||||
const config_file_err = config_file + '.err';
|
const config_file_err = `${config_file}.err`;
|
||||||
|
|
||||||
return new Promise((resolve/*, reject*/) => {
|
return new Promise((resolve/*, reject*/) => {
|
||||||
internalNginx.deleteFile(config_file);
|
internalNginx.deleteFile(config_file);
|
||||||
@@ -373,7 +373,7 @@ const internalNginx = {
|
|||||||
*/
|
*/
|
||||||
renameConfigAsError: (host_type, host) => {
|
renameConfigAsError: (host_type, host) => {
|
||||||
const config_file = internalNginx.getConfigName(internalNginx.getFileFriendlyHostType(host_type), typeof host === 'undefined' ? 0 : host.id);
|
const config_file = internalNginx.getConfigName(internalNginx.getFileFriendlyHostType(host_type), typeof host === 'undefined' ? 0 : host.id);
|
||||||
const config_file_err = config_file + '.err';
|
const config_file_err = `${config_file}.err`;
|
||||||
|
|
||||||
return new Promise((resolve/*, reject*/) => {
|
return new Promise((resolve/*, reject*/) => {
|
||||||
fs.unlink(config_file, () => {
|
fs.unlink(config_file, () => {
|
||||||
@@ -392,8 +392,8 @@ const internalNginx = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
bulkGenerateConfigs: (host_type, hosts) => {
|
bulkGenerateConfigs: (host_type, hosts) => {
|
||||||
let promises = [];
|
const promises = [];
|
||||||
hosts.map(function (host) {
|
hosts.map((host) => {
|
||||||
promises.push(internalNginx.generateConfig(host_type, host));
|
promises.push(internalNginx.generateConfig(host_type, host));
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -406,8 +406,8 @@ const internalNginx = {
|
|||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
bulkDeleteConfigs: (host_type, hosts) => {
|
bulkDeleteConfigs: (host_type, hosts) => {
|
||||||
let promises = [];
|
const promises = [];
|
||||||
hosts.map(function (host) {
|
hosts.map((host) => {
|
||||||
promises.push(internalNginx.deleteConfig(host_type, host, true));
|
promises.push(internalNginx.deleteConfig(host_type, host, true));
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -418,14 +418,12 @@ const internalNginx = {
|
|||||||
* @param {string} config
|
* @param {string} config
|
||||||
* @returns {boolean}
|
* @returns {boolean}
|
||||||
*/
|
*/
|
||||||
advancedConfigHasDefaultLocation: function (cfg) {
|
advancedConfigHasDefaultLocation: (cfg) => !!cfg.match(/^(?:.*;)?\s*?location\s*?\/\s*?{/im),
|
||||||
return !!cfg.match(/^(?:.*;)?\s*?location\s*?\/\s*?{/im);
|
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @returns {boolean}
|
* @returns {boolean}
|
||||||
*/
|
*/
|
||||||
ipv6Enabled: function () {
|
ipv6Enabled: () => {
|
||||||
if (typeof process.env.DISABLE_IPV6 !== 'undefined') {
|
if (typeof process.env.DISABLE_IPV6 !== 'undefined') {
|
||||||
const disabled = process.env.DISABLE_IPV6.toLowerCase();
|
const disabled = process.env.DISABLE_IPV6.toLowerCase();
|
||||||
return !(disabled === 'on' || disabled === 'true' || disabled === '1' || disabled === 'yes');
|
return !(disabled === 'on' || disabled === 'true' || disabled === '1' || disabled === 'yes');
|
||||||
|
@@ -29,15 +29,19 @@ module.exports = {
|
|||||||
/**
|
/**
|
||||||
* @param {String} cmd
|
* @param {String} cmd
|
||||||
* @param {Array} args
|
* @param {Array} args
|
||||||
|
* @param {Object|undefined} options
|
||||||
* @returns {Promise}
|
* @returns {Promise}
|
||||||
*/
|
*/
|
||||||
execFile: (cmd, args) => {
|
execFile: (cmd, args, options) => {
|
||||||
// logger.debug('CMD: ' + cmd + ' ' + (args ? args.join(' ') : ''));
|
logger.debug(`CMD: ${cmd} ${args ? args.join(' ') : ''}`);
|
||||||
|
if (typeof options === 'undefined') {
|
||||||
|
options = {};
|
||||||
|
}
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
execFile(cmd, args, (err, stdout, /*stderr*/) => {
|
execFile(cmd, args, options, (err, stdout, stderr) => {
|
||||||
if (err && typeof err === 'object') {
|
if (err && typeof err === 'object') {
|
||||||
reject(err);
|
reject(new error.CommandError(stderr, 1, err));
|
||||||
} else {
|
} else {
|
||||||
resolve(stdout.trim());
|
resolve(stdout.trim());
|
||||||
}
|
}
|
||||||
|
@@ -6,7 +6,7 @@ const apiValidator = require('../../lib/validator/api');
|
|||||||
const internalCertificate = require('../../internal/certificate');
|
const internalCertificate = require('../../internal/certificate');
|
||||||
const schema = require('../../schema');
|
const schema = require('../../schema');
|
||||||
|
|
||||||
let router = express.Router({
|
const router = express.Router({
|
||||||
caseSensitive: true,
|
caseSensitive: true,
|
||||||
strict: true,
|
strict: true,
|
||||||
mergeParams: true
|
mergeParams: true
|
||||||
@@ -231,7 +231,7 @@ router
|
|||||||
*/
|
*/
|
||||||
router
|
router
|
||||||
.route('/:certificate_id/download')
|
.route('/:certificate_id/download')
|
||||||
.options((req, res) => {
|
.options((_req, res) => {
|
||||||
res.sendStatus(204);
|
res.sendStatus(204);
|
||||||
})
|
})
|
||||||
.all(jwtdecode())
|
.all(jwtdecode())
|
||||||
|
@@ -110,6 +110,11 @@
|
|||||||
"caching_enabled": {
|
"caching_enabled": {
|
||||||
"description": "Should we cache assets",
|
"description": "Should we cache assets",
|
||||||
"type": "boolean"
|
"type": "boolean"
|
||||||
|
},
|
||||||
|
"email": {
|
||||||
|
"description": "Email address",
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\\.[A-Za-z]{2,}$"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -69,7 +69,7 @@
|
|||||||
"type": "object"
|
"type": "object"
|
||||||
},
|
},
|
||||||
"letsencrypt_email": {
|
"letsencrypt_email": {
|
||||||
"type": "string"
|
"$ref": "../common.json#/properties/email"
|
||||||
},
|
},
|
||||||
"propagation_seconds": {
|
"propagation_seconds": {
|
||||||
"type": "integer",
|
"type": "integer",
|
||||||
|
@@ -24,7 +24,7 @@ const setupDefaultUser = () => {
|
|||||||
const email = process.env.INITIAL_ADMIN_EMAIL || 'admin@example.com';
|
const email = process.env.INITIAL_ADMIN_EMAIL || 'admin@example.com';
|
||||||
const password = process.env.INITIAL_ADMIN_PASSWORD || 'changeme';
|
const password = process.env.INITIAL_ADMIN_PASSWORD || 'changeme';
|
||||||
|
|
||||||
logger.info('Creating a new user: ' + email + ' with password: ' + password);
|
logger.info(`Creating a new user: ${email} with password: ${password}`);
|
||||||
|
|
||||||
const data = {
|
const data = {
|
||||||
is_deleted: 0,
|
is_deleted: 0,
|
||||||
@@ -113,20 +113,20 @@ const setupCertbotPlugins = () => {
|
|||||||
.andWhere('provider', 'letsencrypt')
|
.andWhere('provider', 'letsencrypt')
|
||||||
.then((certificates) => {
|
.then((certificates) => {
|
||||||
if (certificates && certificates.length) {
|
if (certificates && certificates.length) {
|
||||||
let plugins = [];
|
const plugins = [];
|
||||||
let promises = [];
|
const promises = [];
|
||||||
|
|
||||||
certificates.map(function (certificate) {
|
certificates.map((certificate) => {
|
||||||
if (certificate.meta && certificate.meta.dns_challenge === true) {
|
if (certificate.meta && certificate.meta.dns_challenge === true) {
|
||||||
if (plugins.indexOf(certificate.meta.dns_provider) === -1) {
|
if (plugins.indexOf(certificate.meta.dns_provider) === -1) {
|
||||||
plugins.push(certificate.meta.dns_provider);
|
plugins.push(certificate.meta.dns_provider);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure credentials file exists
|
// Make sure credentials file exists
|
||||||
const credentials_loc = '/etc/letsencrypt/credentials/credentials-' + certificate.id;
|
const credentials_loc = `/etc/letsencrypt/credentials/credentials-${certificate.id}`;
|
||||||
// Escape single quotes and backslashes
|
// Escape single quotes and backslashes
|
||||||
const escapedCredentials = certificate.meta.dns_provider_credentials.replaceAll('\'', '\\\'').replaceAll('\\', '\\\\');
|
const escapedCredentials = certificate.meta.dns_provider_credentials.replaceAll('\'', '\\\'').replaceAll('\\', '\\\\');
|
||||||
const credentials_cmd = '[ -f \'' + credentials_loc + '\' ] || { mkdir -p /etc/letsencrypt/credentials 2> /dev/null; echo \'' + escapedCredentials + '\' > \'' + credentials_loc + '\' && chmod 600 \'' + credentials_loc + '\'; }';
|
const credentials_cmd = `[ -f '${credentials_loc}' ] || { mkdir -p /etc/letsencrypt/credentials 2> /dev/null; echo '${escapedCredentials}' > '${credentials_loc}' && chmod 600 '${credentials_loc}'; }`;
|
||||||
promises.push(utils.exec(credentials_cmd));
|
promises.push(utils.exec(credentials_cmd));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -136,7 +136,7 @@ const setupCertbotPlugins = () => {
|
|||||||
if (promises.length) {
|
if (promises.length) {
|
||||||
return Promise.all(promises)
|
return Promise.all(promises)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
logger.info('Added Certbot plugins ' + plugins.join(', '));
|
logger.info(`Added Certbot plugins ${plugins.join(', ')}`);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -165,9 +165,7 @@ const setupLogrotation = () => {
|
|||||||
return runLogrotate();
|
return runLogrotate();
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = function () {
|
module.exports = () => setupDefaultUser()
|
||||||
return setupDefaultUser()
|
.then(setupDefaultSettings)
|
||||||
.then(setupDefaultSettings)
|
.then(setupCertbotPlugins)
|
||||||
.then(setupCertbotPlugins)
|
.then(setupLogrotation);
|
||||||
.then(setupLogrotation);
|
|
||||||
};
|
|
||||||
|
@@ -96,4 +96,28 @@ describe('Certificates endpoints', () => {
|
|||||||
expect(data.error.message).to.contain('data/domain_names/0 must match pattern');
|
expect(data.error.message).to.contain('data/domain_names/0 must match pattern');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Request Certificate - LE Email Escaped', () => {
|
||||||
|
cy.task('backendApiPost', {
|
||||||
|
token: token,
|
||||||
|
path: '/api/nginx/certificates',
|
||||||
|
data: {
|
||||||
|
domain_names: ['test.com"||echo hello-world||\\\\n test.com"'],
|
||||||
|
meta: {
|
||||||
|
dns_challenge: false,
|
||||||
|
letsencrypt_agree: true,
|
||||||
|
letsencrypt_email: "admin@example.com' --version;echo hello-world",
|
||||||
|
},
|
||||||
|
provider: 'letsencrypt',
|
||||||
|
},
|
||||||
|
returnOnError: true,
|
||||||
|
}).then((data) => {
|
||||||
|
cy.validateSwaggerSchema('post', 400, '/nginx/certificates', data);
|
||||||
|
expect(data).to.have.property('error');
|
||||||
|
expect(data.error).to.have.property('message');
|
||||||
|
expect(data.error).to.have.property('code');
|
||||||
|
expect(data.error.code).to.equal(400);
|
||||||
|
expect(data.error.message).to.contain('data/meta/letsencrypt_email must match pattern');
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
Reference in New Issue
Block a user