Compare commits

...

3 Commits

Author SHA1 Message Date
AJ ONeal 0b387a150f make prettier 2019-07-23 01:59:25 -06:00
AJ ONeal d93093fe00 merge l0nestar's updates 2019-07-23 01:57:10 -06:00
AJ ONeal a6a7f6b95e package cleanup 2019-07-23 01:54:22 -06:00
7 changed files with 421 additions and 244 deletions

80
.gitignore vendored
View File

@ -1,2 +1,78 @@
node_modules
package-lock.json
# ---> Node
# Logs
logs
*.log
npm-debug.log*
yarn-debug.log*
yarn-error.log*
# Runtime data
pids
*.pid
*.seed
*.pid.lock
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
# nyc test coverage
.nyc_output
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# Bower dependency directory (https://bower.io/)
bower_components
# node-waf configuration
.lock-wscript
# Compiled binary addons (https://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules/
jspm_packages/
# TypeScript v1 declaration files
typings/
# Optional npm cache directory
.npm
# Optional eslint cache
.eslintcache
# Optional REPL history
.node_repl_history
# Output of 'npm pack'
*.tgz
# Yarn Integrity file
.yarn-integrity
# dotenv environment variables file
.env
# parcel-bundler cache (https://parceljs.org/)
.cache
# next.js build output
.next
# nuxt.js build output
.nuxt
# vuepress build output
.vuepress/dist
# Serverless directories
.serverless
# FuseBox cache
.fusebox/

View File

@ -1,7 +1,6 @@
# [acme-dns-01-namecheap](https://git.rootprojects.org/root/acme-dns-01-namecheap) | a [Root](https://rootrpojects.org) project
NameCheap DNS + Let's Encrypt
NameCheap DNS + Let's Encrypt
This handles ACME dns-01 challenges, compatible with ACME.js and Greenlock.js.
Passes [acme-dns-01-test](https://git.rootprojects.org/root/acme-dns-01-test.js).
@ -18,11 +17,11 @@ First you create an instance with your credentials:
```js
var dns01 = require('acme-dns-01-namecheap').create({
apiUser:'username',
apiKey : 'xxxx',
clientIp:'public ip',
username: 'api user',
baseUrl: 'sandbox or production', // default production
apiUser: 'username',
apiKey: 'xxxx',
clientIp: 'public ip',
username: 'api user',
baseUrl: 'sandbox or production' // default production
});
```
@ -56,18 +55,18 @@ See the [ACME.js](https://git.rootprojects.org/root/acme-v2.js) for more details
```js
dns01
.set({
identifier: { value: 'foo.example.com' },
wildcard: false,
dnsHost: '_acme-challenge.foo.example.com',
dnsAuthorization: 'xxx_secret_xxx'
})
.then(function () {
console.log("TXT record set");
})
.catch(function () {
console.log("Failed to set TXT record");
});
.set({
identifier: { value: 'foo.example.com' },
wildcard: false,
dnsHost: '_acme-challenge.foo.example.com',
dnsAuthorization: 'xxx_secret_xxx'
})
.then(function() {
console.log('TXT record set');
})
.catch(function() {
console.log('Failed to set TXT record');
});
```
See [acme-dns-01-test](https://git.rootprojects.org/root/acme-dns-01-test.js)
@ -77,6 +76,5 @@ for more implementation details.
```bash
# node ./test.js domain-zone api-user api-key client-ip username [username is optional if similar to api-user]
node ./test.js example.com demo 45.77.4.126 d41474b94e7d4536baabb074a09c96bd
node ./test.js example.com demo d41474b94e7d4536baabb074a09c96bd 45.77.4.126
```

6
example.env Normal file
View File

@ -0,0 +1,6 @@
ZONE=example.co.uk
API_USER=exampleuser
API_KEY=xxxxxxxxxxxxxxx
USERNAME=exampleuser
CLIENT_IP=121.22.123.22

View File

@ -5,222 +5,263 @@ var request; // = require('@root/request');
var parseString = require('xml2js').parseString;
parseString = util.promisify(parseString);
const SANDBOX_URL = 'https://api.sandbox.namecheap.com/xml.response';
const PRODUCTION_URL = 'https://api.namecheap.com/xml.response';
var defaults = {
baseUrl: SANDBOX_URL
baseUrl: SANDBOX_URL
};
function extend(obj) {
var newObj = {};
for (var i in obj) {
if (obj.hasOwnProperty(i)) {
newObj[i] = obj[i];
}
}
return newObj;
var newObj = {};
for (var i in obj) {
if (obj.hasOwnProperty(i)) {
newObj[i] = obj[i];
}
}
return newObj;
}
function assign(obj1,obj2) {
for (var attrname in obj2) { obj1[attrname] = obj2[attrname]; }
function assign(obj1, obj2) {
for (var attrname in obj2) {
obj1[attrname] = obj2[attrname];
}
}
function requestUrl(baseUrl, params) {
var queryString = Object.keys(params).map(function (key) {
return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]);
}).join('&');
// console.debug(queryString);
return baseUrl + '?' + queryString;
var queryString = Object.keys(params)
.map(function(key) {
return (
encodeURIComponent(key) + '=' + encodeURIComponent(params[key])
);
})
.join('&');
// console.debug(queryString);
return baseUrl + '?' + queryString;
}
module.exports.create = function (config) {
// config = { baseUrl, token }
var baseUrl = config.baseUrl || defaults.baseUrl;
module.exports.create = function(config) {
// config = { baseUrl, token }
var baseUrl = config.baseUrl || defaults.baseUrl;
var globalParams = {
apiUser: config.apiUser,
apiKey: config.apiKey,
username: config.username,
ClientIp: config.clientIp
};
var globalParams = {
apiUser: config.apiUser,
apiKey: config.apiKey,
username: config.username,
ClientIp: config.clientIp
};
function api(command, params) {
var requestParams = extend(globalParams);
requestParams['Command'] = command;
assign(requestParams,params);
function api(command, params) {
var requestParams = extend(globalParams);
requestParams['Command'] = command;
assign(requestParams, params);
var url = requestUrl(baseUrl, requestParams);
console.log('DEBUG >>> url: ' + url);
console.log('DEBUG >>> requestParams: ' + JSON.stringify(requestParams, null, 2));
var url = requestUrl(baseUrl, requestParams);
console.log('DEBUG >>> url: ' + url);
console.log(
'DEBUG >>> requestParams: ' + JSON.stringify(requestParams, null, 2)
);
return request({
method: 'POST',
url: url,
}).then(function (response) {
var responseBody = response.body;
// console.log(responseBody);
return parseString(responseBody).then(function (result) {
// check response status
if (result['ApiResponse']['$']['Status'] === 'ERROR') {
for (let i = 0; i < result['ApiResponse']['Errors'].length; i++) {
console.log('DEBUG >>> error: ' + JSON.stringify(result['ApiResponse']['Errors'][i]['Error'][0], null, 2));
}
throw new Error('API Error');
} else { // Status="OK"
return result['ApiResponse']['CommandResponse'][0]
}
});
});
}
return request({
method: 'POST',
url: url
}).then(function(response) {
var responseBody = response.body;
// console.log(responseBody);
return parseString(responseBody).then(function(result) {
// check response status
if (result['ApiResponse']['$']['Status'] === 'ERROR') {
for (
let i = 0;
i < result['ApiResponse']['Errors'].length;
i++
) {
console.log(
'DEBUG >>> error: ' +
JSON.stringify(
result['ApiResponse']['Errors'][i][
'Error'
][0],
null,
2
)
);
}
throw new Error('API Error');
} else {
// Status="OK"
return result['ApiResponse']['CommandResponse'][0];
}
});
});
}
return {
init: function (deps) {
request = deps.request;
return null;
},
return {
init: function(deps) {
request = deps.request;
return null;
},
zones: function(data) {
return api('namecheap.domains.getList',{}).then(function (zonesResponse) {
// console.log('zones');
// console.log(zonesResponse);
return zonesResponse['DomainGetListResult'].map(function (x) {
return x['Domain'][0]['$']['Name'];
});
zones: function(data) {
return api('namecheap.domains.getList', {}).then(function(
zonesResponse
) {
// console.log('zones');
// console.log(zonesResponse);
return zonesResponse['DomainGetListResult'].map(function(x) {
return x['Domain'][0]['$']['Name'];
});
});
},
});
},
set: function(data) {
console.log(`DEBUG >>> data: ${JSON.stringify(data, null, 2)}`);
var ch = data.challenge;
var txt = ch.dnsAuthorization;
set: function (data) {
console.log(`DEBUG >>> data: ${JSON.stringify(data, null, 2)}`);
var ch = data.challenge;
var txt = ch.dnsAuthorization;
var params = {};
// var zone = ch.dnsZone;
var zone = ch.identifier.value;
console.log(`DEBUG >>> zone: ${zone}`);
var params = {};
// var zone = ch.dnsZone;
var zone = ch.identifier.value;
console.log(`DEBUG >>> zone: ${zone}`);
// the domain is the first part
// params['SLD'] = zone.split('.')[0];
// the rest of the components are the TLD
// params['TLD'] = zone.split('.').splice(1).join('.');
// the domain is the first part
// params['SLD'] = zone.split('.')[0];
// the rest of the components are the TLD
// params['TLD'] = zone.split('.').splice(1).join('.');
var domains = zone.split('.');
console.log('DEBUG >>> ' + domains);
var domains = zone.split('.');
console.log('DEBUG >>> ' + domains);
// if you have subdomain foo.blah.com, SLD = blah and TLD = com
params['TLD'] = domains[domains.length - 1];
params['SLD'] = domains[domains.length - 2];
// if you have subdomain foo.blah.com, SLD = blah and TLD = com
params['TLD'] = domains[domains.length - 1];
params['SLD'] = domains[domains.length - 2];
console.log(`DEBUG >>> SLD: ${params['SLD']}`);
console.log(`DEBUG >>> TLD: ${params['TLD']}`);
console.log(`DEBUG >>> SLD: ${params['SLD']}`);
console.log(`DEBUG >>> TLD: ${params['TLD']}`);
// setting a host record overwrites all existing,
// adding a new records means you've have to send back all previous records too
// setting a host record overwrites all existing,
// adding a new records means you've have to send back all previous records too
return api('namecheap.domains.dns.getHosts',params).then(function (hostsResponse) {
var currentHostRecordsCount = hostsResponse['DomainDNSGetHostsResult'][0]['host'].length;
return api('namecheap.domains.dns.getHosts', params).then(function(
hostsResponse
) {
var currentHostRecordsCount =
hostsResponse['DomainDNSGetHostsResult'][0]['host'].length;
for (var i = 0; i < currentHostRecordsCount; i++) {
// console.log(hostsResponse['DomainDNSGetHostsResult'][i]['host'][0]);
var currentEntry = hostsResponse['DomainDNSGetHostsResult'][0]['host'][i]['$'];
for (var i = 0; i < currentHostRecordsCount; i++) {
// console.log(hostsResponse['DomainDNSGetHostsResult'][i]['host'][0]);
var currentEntry =
hostsResponse['DomainDNSGetHostsResult'][0]['host'][i][
'$'
];
params['HostName'+(i+1)] = currentEntry['Name'];
params['RecordType'+(i+1)] = currentEntry['Type'];
params['Address'+(i+1)] = currentEntry['Address'];
params['TTL'+(i+1)] = currentEntry['TTL'];
}
params['HostName' + (i + 1)] = currentEntry['Name'];
params['RecordType' + (i + 1)] = currentEntry['Type'];
params['Address' + (i + 1)] = currentEntry['Address'];
params['TTL' + (i + 1)] = currentEntry['TTL'];
}
params['HostName'+(currentHostRecordsCount+1)] = ch.dnsPrefix;
params['RecordType'+(currentHostRecordsCount+1)] = 'TXT';
params['Address'+(currentHostRecordsCount+1)] = txt;
params['TTL'+(currentHostRecordsCount+1)] = 100; // in minutes
params['HostName' + (currentHostRecordsCount + 1)] =
ch.dnsPrefix;
params['RecordType' + (currentHostRecordsCount + 1)] = 'TXT';
params['Address' + (currentHostRecordsCount + 1)] = txt;
params['TTL' + (currentHostRecordsCount + 1)] = 100; // in minutes
// console.log(params);
// console.log(params);
return api('namecheap.domains.dns.setHosts',params).then(function (setHostResponse) {
// console.log('setHost');
// console.log(setHostResponse);
return true
}).catch(function (err) {
throw new Error('record did not set. check subdomain, api key, etc');
});
});
return api('namecheap.domains.dns.setHosts', params)
.then(function(setHostResponse) {
// console.log('setHost');
// console.log(setHostResponse);
return true;
})
.catch(function(err) {
throw new Error(
'record did not set. check subdomain, api key, etc'
);
});
});
},
remove: function(data) {
var ch = data.challenge;
},
remove: function (data) {
var ch = data.challenge;
var params = {};
var zone = ch.identifier.value;
var domains = zone.split('.');
var params = {};
var zone = ch.identifier.value;
var domains = zone.split('.');
params['TLD'] = domains[domains.length - 1];
params['SLD'] = domains[domains.length - 2];
params['TLD'] = domains[domains.length - 1];
params['SLD'] = domains[domains.length - 2];
// setting a host record overwrites all existing,
// removing a new records means you've have to send back all previous records without removed
// setting a host record overwrites all existing,
// removing a new records means you've have to send back all previous records without removed
return api('namecheap.domains.dns.getHosts',params).then(function (hostsResponse) {
var currentHostRecordsCount = hostsResponse['DomainDNSGetHostsResult'][0]['host'].length;
return api('namecheap.domains.dns.getHosts', params).then(function(
hostsResponse
) {
var currentHostRecordsCount =
hostsResponse['DomainDNSGetHostsResult'][0]['host'].length;
for (var i = 0; i < currentHostRecordsCount; i++) {
// console.log(hostsResponse['DomainDNSGetHostsResult'][i]['host'][0]);
var currentEntry = hostsResponse['DomainDNSGetHostsResult'][0]['host'][i]['$'];
if(currentEntry['Address'] != ch.dnsAuthorization){
params['HostName'+(i+1)] = currentEntry['Name'];
params['RecordType'+(i+1)] = currentEntry['Type'];
params['Address'+(i+1)] = currentEntry['Address'];
params['TTL'+(i+1)] = currentEntry['TTL'];
}
}
for (var i = 0; i < currentHostRecordsCount; i++) {
// console.log(hostsResponse['DomainDNSGetHostsResult'][i]['host'][0]);
var currentEntry =
hostsResponse['DomainDNSGetHostsResult'][0]['host'][i][
'$'
];
if (currentEntry['Address'] != ch.dnsAuthorization) {
params['HostName' + (i + 1)] = currentEntry['Name'];
params['RecordType' + (i + 1)] = currentEntry['Type'];
params['Address' + (i + 1)] = currentEntry['Address'];
params['TTL' + (i + 1)] = currentEntry['TTL'];
}
}
return api('namecheap.domains.dns.setHosts',params).then(function (setHostResponse) {
// console.log('setHost');
// console.log(setHostResponse);
return true
}).catch(function (err) {
throw new Error('record did not remove. check subdomain, api key, etc');
});
});
return api('namecheap.domains.dns.setHosts', params)
.then(function(setHostResponse) {
// console.log('setHost');
// console.log(setHostResponse);
return true;
})
.catch(function(err) {
throw new Error(
'record did not remove. check subdomain, api key, etc'
);
});
});
},
get: function(data) {
var ch = data.challenge;
},
get: function (data) {
var ch = data.challenge;
var params = {};
var zone = ch.identifier.value;
var domains = zone.split('.');
var params = {};
var zone = ch.identifier.value;
var domains = zone.split('.');
params['TLD'] = domains[domains.length - 1];
params['SLD'] = domains[domains.length - 2];
params['TLD'] = domains[domains.length - 1];
params['SLD'] = domains[domains.length - 2];
return api('namecheap.domains.dns.getHosts', params).then(function(
hostsResponse
) {
// console.log('hosts');
// console.log(hostsResponse);
var currentHostRecords =
hostsResponse['DomainDNSGetHostsResult'][0]['host'];
return api('namecheap.domains.dns.getHosts',params).then(function (hostsResponse) {
// console.log('hosts');
// console.log(hostsResponse);
var currentHostRecords = hostsResponse['DomainDNSGetHostsResult'][0]['host'];
var entries = currentHostRecords.filter(function(x) {
return x['$']['Type'] === 'TXT';
});
var entries = currentHostRecords.filter(function (x) {
return x['$']['Type'] === 'TXT';
});
var entry = entries.filter(function(x) {
// console.log('data', x.data);
// console.log('dnsAuth', ch.dnsAuthorization, ch);
return x['$']['Address'] === ch.dnsAuthorization;
})[0];
var entry = entries.filter(function (x) {
// console.log('data', x.data);
// console.log('dnsAuth', ch.dnsAuthorization, ch);
return x['$']['Address'] === ch.dnsAuthorization;
})[0];
if (entry) {
return {dnsAuthorization: entry['$']['Address']};
} else {
return null;
}
});
}
};
if (entry) {
return { dnsAuthorization: entry['$']['Address'] };
} else {
return null;
}
});
}
};
};

49
package-lock.json generated Normal file
View File

@ -0,0 +1,49 @@
{
"name": "acme-dns-01-namecheap",
"version": "3.0.0",
"lockfileVersion": 1,
"requires": true,
"dependencies": {
"@root/request": {
"version": "1.3.11",
"resolved": "https://registry.npmjs.org/@root/request/-/request-1.3.11.tgz",
"integrity": "sha512-3a4Eeghcjsfe6zh7EJ+ni1l8OK9Fz2wL1OjP4UCa0YdvtH39kdXB9RGWuzyNv7dZi0+Ffkc83KfH0WbPMiuJFw=="
},
"acme-challenge-test": {
"version": "3.3.2",
"resolved": "https://registry.npmjs.org/acme-challenge-test/-/acme-challenge-test-3.3.2.tgz",
"integrity": "sha512-0AbMcaON20wpI5vzFDAqwcv2VerY4xIlNCqX0w1xEJUIu/EQtQNmkje+rKNuy2TUl2KBMdIaR6YBbJUdaEiC4w==",
"dev": true,
"requires": {
"@root/request": "^1.3.11"
},
"dependencies": {
"@root/request": {
"version": "1.3.11",
"resolved": "https://registry.npmjs.org/@root/request/-/request-1.3.11.tgz",
"integrity": "sha512-3a4Eeghcjsfe6zh7EJ+ni1l8OK9Fz2wL1OjP4UCa0YdvtH39kdXB9RGWuzyNv7dZi0+Ffkc83KfH0WbPMiuJFw==",
"dev": true
}
}
},
"sax": {
"version": "1.2.4",
"resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz",
"integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw=="
},
"xml2js": {
"version": "0.4.19",
"resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.19.tgz",
"integrity": "sha512-esZnJZJOiJR9wWKMyuvSE1y6Dq5LCuJanqhxslH2bxM6duahNZ+HMpCLhBQGZkbX6xRf8x1Y2eJlgt2q3qo49Q==",
"requires": {
"sax": ">=0.6.0",
"xmlbuilder": "~9.0.1"
}
},
"xmlbuilder": {
"version": "9.0.7",
"resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-9.0.7.tgz",
"integrity": "sha1-Ey7mPS7FVlxVfiD0wi35rKaGsQ0="
}
}
}

View File

@ -1,35 +1,39 @@
{
"name": "acme-dns-01-namecheap",
"version": "3.0.0",
"description": "Namecheap DNS for Let's Encrypt / ACME dns-01 challenges with ACME.js and Greenlock.js",
"main": "index.js",
"scripts": {
"test": "node ./test.js"
},
"repository": {
"type": "git",
"url": "https://git.coolaj86.com/coolaj86/acme-dns-01-namecheap.js.git"
},
"keywords": [
"namecheap",
"name-cheap",
"dns",
"dns-01",
"letsencrypt",
"acme",
"greenlock"
],
"author": "AJ ONeal <coolaj86@gmail.com> (https://coolaj86.com/)",
"contributors": [
"Nyaundi Brian <danleyb2@gmail.com> (https://git.coolaj86.com/danleyb2/)",
"Archie Baer <archie@abaer.dev> (https://abaer.dev)"
],
"license": "MPL-2.0",
"dependencies": {
"@root/request": "^1.3.11",
"xml2js": "^0.4.19"
},
"devDependencies": {
"acme-dns-01-test": "^3.2.1"
}
"name": "acme-dns-01-namecheap",
"version": "3.0.0",
"description": "Namecheap DNS for Let's Encrypt / ACME dns-01 challenges with ACME.js and Greenlock.js",
"main": "index.js",
"files": [
"lib",
"test.js"
],
"scripts": {
"test": "node ./test.js"
},
"repository": {
"type": "git",
"url": "https://git.coolaj86.com/coolaj86/acme-dns-01-namecheap.js.git"
},
"keywords": [
"namecheap",
"name-cheap",
"dns",
"dns-01",
"letsencrypt",
"acme",
"greenlock"
],
"author": "AJ ONeal <coolaj86@gmail.com> (https://coolaj86.com/)",
"contributors": [
"Nyaundi Brian <danleyb2@gmail.com> (https://git.coolaj86.com/danleyb2/)",
"Archie Baer <archie@abaer.dev> (https://abaer.dev)"
],
"license": "MPL-2.0",
"dependencies": {
"@root/request": "^1.3.11",
"xml2js": "^0.4.19"
},
"devDependencies": {
"acme-challenge-test": "^3.3.2"
}
}

33
test.js
View File

@ -3,28 +3,31 @@
// https://git.rootprojects.org/root/acme-dns-01-test.js
var tester = require('acme-challenge-test');
require('dotenv').config();
// Usage: node ./test.js example.com xxxxxxxxx
var zone = process.argv[2];
var zone = process.argv[2] || process.env.ZONE;
var challenger = require('./index.js').create({
apiUser:process.argv[3],
apiKey : process.argv[4],
clientIp:process.argv[5],
username: process.argv[6] || process.argv[3]
apiUser: process.argv[3] || process.env.API_USER,
apiKey: process.argv[4] || process.env.API_KEY || process.env.TOKEN,
clientIp: process.argv[5] || process.env.CLIENT_IP,
username:
process.argv[6] ||
process.env.USERNAME ||
process.argv[3] ||
process.env.API_USER
});
// The dry-run tests can pass on, literally, 'example.com'
// but the integration tests require that you have control over the domain
tester
.testZone('dns-01', zone, challenger)
.then(function() {
console.info('PASS', zone);
})
.catch(function(e) {
console.info('FAIL', zone);
console.error(e.message);
console.error(e.stack);
});
.then(function() {
console.info('PASS', zone);
})
.catch(function(e) {
console.info('FAIL', zone);
console.error(e.message);
console.error(e.stack);
});