cloudns-cloudformation-sync 1.2.0 → 1.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.prettierrc +5 -0
- package/README.md +4 -2
- package/lib/cloudns-cloudformation-sync.js +45 -30
- package/package.json +5 -5
- package/src/cloudns-cloudformation-sync.ts +76 -42
package/.prettierrc
ADDED
package/README.md
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
# ClouDNS CloudFormation Sync
|
|
2
|
-
|
|
2
|
+
|
|
3
|
+
Copyright (C) Clouden Oy 2020-2023, author Kenneth Falck <kennu@clouden.net>.
|
|
3
4
|
|
|
4
5
|
Released under the MIT license.
|
|
5
6
|
|
|
@@ -29,7 +30,7 @@ Other resource types are also allowed (A, AAAA, ALIAS, etc).
|
|
|
29
30
|
|
|
30
31
|
Use the cloudns-cloudformation-sync command to synchronize ClouDNS records.
|
|
31
32
|
|
|
32
|
-
AWS_PROFILE=xxx cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl]
|
|
33
|
+
AWS_PROFILE=xxx cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl [stackname]]
|
|
33
34
|
|
|
34
35
|
Options:
|
|
35
36
|
|
|
@@ -37,5 +38,6 @@ Options:
|
|
|
37
38
|
<cloudns-username> - ClouDNS API sub-auth-user
|
|
38
39
|
<cloudns-password-parameter-name> - SSM Parameter with the encrypted ClouDNS API password
|
|
39
40
|
[ttl] - Optional TTL for generated records (defaults to 300)
|
|
41
|
+
[stackName] - Optional CloudFormation stack name to limit the exports to scan (defaults to all stacks)
|
|
40
42
|
|
|
41
43
|
You can create your ClouDNS API credentials in the ClouDNS management console.
|
|
@@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.main = void 0;
|
|
4
4
|
/**
|
|
5
5
|
* Read AWS CloudFormation Exports and autogenerate ClouDNS records based on their names and values.
|
|
6
|
-
* Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020
|
|
6
|
+
* Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020-2023
|
|
7
7
|
*
|
|
8
8
|
* This tool can be used to autogenerate ClouDNS records for CloudFormation resources like
|
|
9
9
|
* CloudFront distributions and API Gateway domains.
|
|
@@ -19,12 +19,13 @@ exports.main = void 0;
|
|
|
19
19
|
*
|
|
20
20
|
* Other resource types are also allowed (A, AAAA, ALIAS, etc).
|
|
21
21
|
*
|
|
22
|
-
* Command line usage: AWS_PROFILE=xxx ts-node cloudns-cloudformation-sync.ts <cloudns-username> <cloudns-password-parameter-name> [ttl]
|
|
22
|
+
* Command line usage: AWS_PROFILE=xxx ts-node cloudns-cloudformation-sync.ts <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]
|
|
23
23
|
*
|
|
24
24
|
* AWS_PROFILE=xxx - Specify your AWS profile in ~/.aws/credentials as an environment variable
|
|
25
25
|
* <cloudns-username> - ClouDNS API sub-auth-user
|
|
26
26
|
* <cloudns-password-parameter-name> - SSM Parameter with the encrypted ClouDNS API password
|
|
27
27
|
* [ttl] - Optional TTL for generated records (defaults to 300)
|
|
28
|
+
* [stackName] - Optional CloudFormation stack name to limit the exports to scan (defaults to all stacks)
|
|
28
29
|
*/
|
|
29
30
|
const client_ssm_1 = require("@aws-sdk/client-ssm");
|
|
30
31
|
const client_cloudformation_1 = require("@aws-sdk/client-cloudformation");
|
|
@@ -33,17 +34,20 @@ const querystring = require("querystring");
|
|
|
33
34
|
// Load ~/.aws/config
|
|
34
35
|
process.env.AWS_SDK_LOAD_CONFIG = '1';
|
|
35
36
|
async function cloudnsRestCall(cloudnsUsername, cloudnsPassword, method, relativeUrl, queryOptions) {
|
|
36
|
-
let fullUrl = 'https://api.cloudns.net' +
|
|
37
|
-
|
|
38
|
-
'
|
|
39
|
-
|
|
37
|
+
let fullUrl = 'https://api.cloudns.net' +
|
|
38
|
+
relativeUrl +
|
|
39
|
+
'?' +
|
|
40
|
+
querystring.stringify(Object.assign({
|
|
41
|
+
'sub-auth-user': cloudnsUsername,
|
|
42
|
+
'auth-password': cloudnsPassword,
|
|
43
|
+
}, queryOptions || {}));
|
|
40
44
|
// console.log('Note: Calling', fullUrl)
|
|
41
45
|
const response = await (0, node_fetch_1.default)(fullUrl, {
|
|
42
46
|
method: method,
|
|
43
47
|
headers: {
|
|
44
48
|
'Content-Type': 'application/json',
|
|
45
|
-
|
|
46
|
-
}
|
|
49
|
+
Accept: 'application/json',
|
|
50
|
+
},
|
|
47
51
|
});
|
|
48
52
|
if (!response.ok) {
|
|
49
53
|
const errorText = await response.text();
|
|
@@ -61,18 +65,20 @@ async function autoDetectCloudnsHostAndZone(cloudnsUsername, cloudnsPassword, na
|
|
|
61
65
|
const hostName2 = nameParts.slice(0, nameParts.length - 3).join('.');
|
|
62
66
|
const zoneName2 = nameParts.slice(nameParts.length - 3).join('.');
|
|
63
67
|
// Check which zone exists
|
|
64
|
-
const zoneResponse1 = zoneCache[zoneName1] ||
|
|
65
|
-
'
|
|
66
|
-
|
|
68
|
+
const zoneResponse1 = zoneCache[zoneName1] ||
|
|
69
|
+
(await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {
|
|
70
|
+
'domain-name': zoneName1,
|
|
71
|
+
}));
|
|
67
72
|
zoneCache[zoneName1] = zoneResponse1;
|
|
68
|
-
const zoneResponse2 = zoneCache[zoneName2] ||
|
|
69
|
-
'
|
|
70
|
-
|
|
73
|
+
const zoneResponse2 = zoneCache[zoneName2] ||
|
|
74
|
+
(await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {
|
|
75
|
+
'domain-name': zoneName2,
|
|
76
|
+
}));
|
|
71
77
|
zoneCache[zoneName2] = zoneResponse2;
|
|
72
78
|
// console.log('Note: Response for host', hostName1, 'in zone', zoneName1, ':', zoneResponse1)
|
|
73
79
|
// console.log('Note: Response for host', hostName2, 'in zone', zoneName2, ':', zoneResponse2)
|
|
74
|
-
const zoneName =
|
|
75
|
-
const hostName =
|
|
80
|
+
const zoneName = zoneResponse1.status === '1' ? zoneName1 : zoneResponse2.status === '1' ? zoneName2 : '';
|
|
81
|
+
const hostName = zoneResponse1.status === '1' ? hostName1 : zoneResponse2.status === '1' ? hostName2 : '';
|
|
76
82
|
if (!zoneName) {
|
|
77
83
|
// Neither zone exists
|
|
78
84
|
throw new Error('Zone Not Found: ' + name);
|
|
@@ -87,8 +93,8 @@ async function createOrUpdateCloudnsResource(cloudnsUsername, cloudnsPassword, n
|
|
|
87
93
|
// Does the record exist?
|
|
88
94
|
const recordsResponse = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/records.json', {
|
|
89
95
|
'domain-name': zoneName,
|
|
90
|
-
|
|
91
|
-
|
|
96
|
+
host: hostName,
|
|
97
|
+
type: type,
|
|
92
98
|
});
|
|
93
99
|
const existingRecord = Object.values(recordsResponse)[0];
|
|
94
100
|
if ((existingRecord === null || existingRecord === void 0 ? void 0 : existingRecord.host) === hostName && (existingRecord === null || existingRecord === void 0 ? void 0 : existingRecord.type) === type && (existingRecord === null || existingRecord === void 0 ? void 0 : existingRecord.ttl) === ttlValue && (existingRecord === null || existingRecord === void 0 ? void 0 : existingRecord.record) === value) {
|
|
@@ -101,10 +107,10 @@ async function createOrUpdateCloudnsResource(cloudnsUsername, cloudnsPassword, n
|
|
|
101
107
|
const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/mod-record.json', {
|
|
102
108
|
'domain-name': zoneName,
|
|
103
109
|
'record-id': existingRecord === null || existingRecord === void 0 ? void 0 : existingRecord.id,
|
|
104
|
-
|
|
110
|
+
host: hostName,
|
|
105
111
|
'record-type': type,
|
|
106
|
-
|
|
107
|
-
|
|
112
|
+
record: value,
|
|
113
|
+
ttl: ttlValue,
|
|
108
114
|
});
|
|
109
115
|
if (result.status === 'Failed') {
|
|
110
116
|
throw new Error('Modify record failed: ' + (result.statusMessage || result.statusDescription));
|
|
@@ -115,10 +121,10 @@ async function createOrUpdateCloudnsResource(cloudnsUsername, cloudnsPassword, n
|
|
|
115
121
|
console.log('CREATE', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName);
|
|
116
122
|
const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/add-record.json', {
|
|
117
123
|
'domain-name': zoneName,
|
|
118
|
-
|
|
124
|
+
host: hostName,
|
|
119
125
|
'record-type': type,
|
|
120
|
-
|
|
121
|
-
|
|
126
|
+
record: value,
|
|
127
|
+
ttl: ttlValue,
|
|
122
128
|
});
|
|
123
129
|
if (result.status === 'Failed') {
|
|
124
130
|
throw new Error('Add record failed: ' + (result.statusMessage || result.statusDescription));
|
|
@@ -126,17 +132,18 @@ async function createOrUpdateCloudnsResource(cloudnsUsername, cloudnsPassword, n
|
|
|
126
132
|
}
|
|
127
133
|
}
|
|
128
134
|
async function main() {
|
|
129
|
-
var _a, _b;
|
|
130
|
-
console.log('ClouDNS CloudFormation Sync by Kenneth Falck <kennu@clouden.net> (C) Clouden Oy
|
|
135
|
+
var _a, _b, _c;
|
|
136
|
+
console.log('ClouDNS CloudFormation Sync by Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020-2023');
|
|
131
137
|
const cloudnsUsername = process.argv[2];
|
|
132
138
|
const cloudnsPasswordParameter = process.argv[3];
|
|
133
139
|
const ttlValue = process.argv[4] || '300';
|
|
140
|
+
const stackName = process.argv[5] || '';
|
|
134
141
|
if (!cloudnsUsername) {
|
|
135
|
-
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl]');
|
|
142
|
+
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]');
|
|
136
143
|
process.exit(1);
|
|
137
144
|
}
|
|
138
145
|
if (!cloudnsPasswordParameter) {
|
|
139
|
-
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl]');
|
|
146
|
+
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]');
|
|
140
147
|
process.exit(1);
|
|
141
148
|
}
|
|
142
149
|
const ssm = new client_ssm_1.SSMClient({});
|
|
@@ -153,7 +160,15 @@ async function main() {
|
|
|
153
160
|
NextToken: nextToken,
|
|
154
161
|
}));
|
|
155
162
|
for (const exportObj of response.Exports || []) {
|
|
156
|
-
if (
|
|
163
|
+
if (stackName && exportObj.ExportingStackId !== stackName) {
|
|
164
|
+
// Check if the name part of the ID matches arn:aws:cloudformation:eu-west-1:<xxx>:stack/<name>/<xxx>
|
|
165
|
+
const m = (_b = exportObj.ExportingStackId) === null || _b === void 0 ? void 0 : _b.match(/^arn:[^:]+:cloudformation:[^:]+:[^:]+:stack\/([^\/]+)\//);
|
|
166
|
+
if (!m || m[1] !== stackName) {
|
|
167
|
+
// Stack ID name part didn't match given stackName, so skip it
|
|
168
|
+
continue;
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
if ((_c = exportObj.Name) === null || _c === void 0 ? void 0 : _c.match(/^ClouDNS:/)) {
|
|
157
172
|
const nameParts = exportObj.Name.split(':');
|
|
158
173
|
const resourceType = nameParts[1];
|
|
159
174
|
const resourceName = nameParts.slice(2).join('.');
|
|
@@ -165,4 +180,4 @@ async function main() {
|
|
|
165
180
|
} while (nextToken);
|
|
166
181
|
}
|
|
167
182
|
exports.main = main;
|
|
168
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudns-cloudformation-sync.js","sourceRoot":"","sources":["../src/cloudns-cloudformation-sync.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Read AWS CloudFormation Exports and autogenerate ClouDNS records based on their names and values.\n * Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020\n *\n * This tool can be used to autogenerate ClouDNS records for CloudFormation resources like\n * CloudFront distributions and API Gateway domains.\n *\n * CloudFormation export name must specify the resource type and record hostname as follows:\n * ClouDNS:CNAME:myhost:example:org\n *\n * CloudFormation export value must specify the record value as-is (for instance, a distribution domain name):\n * xxxxxxxxxxxxxx.cloudfront.net\n *\n * The above example will generate the following record in the ClouDNS zone example.org:\n * myhost.example.org CNAME xxxxxxxxxxxxxx.cloudfront.net\n *\n * Other resource types are also allowed (A, AAAA, ALIAS, etc).\n *\n * Command line usage: AWS_PROFILE=xxx ts-node cloudns-cloudformation-sync.ts <cloudns-username> <cloudns-password-parameter-name> [ttl]\n *\n * AWS_PROFILE=xxx - Specify your AWS profile in ~/.aws/credentials as an environment variable\n * <cloudns-username> - ClouDNS API sub-auth-user\n * <cloudns-password-parameter-name> - SSM Parameter with the encrypted ClouDNS API password\n * [ttl] - Optional TTL for generated records (defaults to 300)\n */\nimport { SSMClient, GetParameterCommand } from '@aws-sdk/client-ssm'\nimport { CloudFormationClient, ListExportsCommand, ListExportsOutput } from '@aws-sdk/client-cloudformation'\nimport fetch from 'node-fetch'\nimport * as querystring from 'querystring'\n\n// Load ~/.aws/config\nprocess.env.AWS_SDK_LOAD_CONFIG = '1'\n\nasync function cloudnsRestCall(cloudnsUsername: string, cloudnsPassword: string, method: string, relativeUrl: string, queryOptions: any) {\n  let fullUrl = 'https://api.cloudns.net' + relativeUrl + '?' + querystring.stringify(Object.assign({\n    'sub-auth-user': cloudnsUsername,\n    'auth-password': cloudnsPassword,\n  }, queryOptions || {}))\n\n  // console.log('Note: Calling', fullUrl)\n\n  const response = await fetch(fullUrl, {\n    method: method,\n    headers: {\n      'Content-Type': 'application/json',\n      'Accept': 'application/json',\n    }\n  })\n  if (!response.ok) {\n    const errorText = await response.text()\n    console.error('HTTP Error', response.status, response.statusText, errorText)\n    throw new Error(errorText)\n  }\n  return response.json()\n}\n\nasync function autoDetectCloudnsHostAndZone(cloudnsUsername: string, cloudnsPassword: string, name: string, zoneCache: any) {\n  const nameParts = name.split('.')\n\n  // Zone and host name for xxx.tld\n  const hostName1 = nameParts.slice(0, nameParts.length-2).join('.')\n  const zoneName1 = nameParts.slice(nameParts.length-2).join('.')\n\n  // Zone and host name for xxx.subtld.tld\n  const hostName2 = nameParts.slice(0, nameParts.length-3).join('.')\n  const zoneName2 = nameParts.slice(nameParts.length-3).join('.')\n\n  // Check which zone exists\n  const zoneResponse1 = zoneCache[zoneName1] || await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {\n    'domain-name': zoneName1,\n  })\n  zoneCache[zoneName1] = zoneResponse1\n  const zoneResponse2 = zoneCache[zoneName2] || await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {\n    'domain-name': zoneName2,\n  })\n  zoneCache[zoneName2] = zoneResponse2\n\n  // console.log('Note: Response for host', hostName1, 'in zone', zoneName1, ':', zoneResponse1)\n  // console.log('Note: Response for host', hostName2, 'in zone', zoneName2, ':', zoneResponse2)\n\n  const zoneName = (zoneResponse1.status === '1' ? zoneName1 : zoneResponse2.status === '1' ? zoneName2 : '')\n  const hostName = (zoneResponse1.status === '1' ? hostName1 : zoneResponse2.status === '1' ? hostName2 : '')\n  if (!zoneName) {\n    // Neither zone exists\n    throw new Error('Zone Not Found: ' + name)\n  }\n  return {\n    hostName: hostName,\n    zoneName: zoneName,\n  }\n}\n\nasync function createOrUpdateCloudnsResource(cloudnsUsername: string, cloudnsPassword: string, name: string, type: string, value: string, ttlValue: string, zoneCache: any) {\n  const { zoneName, hostName } = await autoDetectCloudnsHostAndZone(cloudnsUsername, cloudnsPassword, name, zoneCache)\n  // Does the record exist?\n  const recordsResponse = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/records.json', {\n    'domain-name': zoneName,\n    'host': hostName,\n    'type': type,\n  })\n  const existingRecord: any = Object.values(recordsResponse)[0]\n  if (existingRecord?.host === hostName && existingRecord?.type === type && existingRecord?.ttl === ttlValue && existingRecord?.record === value) {\n    // Record exists already - no change\n    console.log('OK', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName)\n  } else if (existingRecord?.id) {\n    // Update record\n    console.log('UPDATE', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName)\n    const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/mod-record.json', {\n      'domain-name': zoneName,\n      'record-id': existingRecord?.id,\n      'host': hostName,\n      'record-type': type,\n      'record': value,\n      'ttl': ttlValue,\n    })\n    if (result.status === 'Failed') {\n      throw new Error( 'Modify record failed: ' + (result.statusMessage || result.statusDescription))\n    }\n  } else {\n    // Create record\n    console.log('CREATE', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName)\n    const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/add-record.json', {\n      'domain-name': zoneName,\n      'host': hostName,\n      'record-type': type,\n      'record': value,\n      'ttl': ttlValue,\n    })\n    if (result.status === 'Failed') {\n      throw new Error( 'Add record failed: ' + (result.statusMessage || result.statusDescription))\n    }\n  }\n}\n\nexport async function main() {\n  console.log('ClouDNS CloudFormation Sync by Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2021')\n  const cloudnsUsername = process.argv[2]\n  const cloudnsPasswordParameter = process.argv[3]\n  const ttlValue = process.argv[4] || '300'\n  if (!cloudnsUsername) {\n    console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl]')\n    process.exit(1)\n  }\n  if (!cloudnsPasswordParameter) {\n    console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl]')\n    process.exit(1)\n  }\n\n  const ssm = new SSMClient({})\n  const zoneCache = {}\n\n  const response = await ssm.send(new GetParameterCommand({\n    Name: cloudnsPasswordParameter,\n    WithDecryption: true,\n  }))\n  const cloudnsPassword = response.Parameter?.Value || ''\n\n  const cloudFormation = new CloudFormationClient({})\n  let nextToken\n  do {\n    const response: ListExportsOutput = await cloudFormation.send(new ListExportsCommand({\n      NextToken: nextToken,\n    }))\n    for (const exportObj of response.Exports || []) {\n      if (exportObj.Name?.match(/^ClouDNS:/)) {\n        const nameParts = exportObj.Name.split(':')\n        const resourceType = nameParts[1]\n        const resourceName = nameParts.slice(2).join('.')\n        const resourceValue = exportObj.Value!\n        await createOrUpdateCloudnsResource(cloudnsUsername, cloudnsPassword, resourceName, resourceType, resourceValue, ttlValue, zoneCache)\n      }\n    }\n    nextToken = response.NextToken\n  } while (nextToken)\n}\n\n"]}
|
|
183
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudns-cloudformation-sync.js","sourceRoot":"","sources":["../src/cloudns-cloudformation-sync.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Read AWS CloudFormation Exports and autogenerate ClouDNS records based on their names and values.\n * Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020-2023\n *\n * This tool can be used to autogenerate ClouDNS records for CloudFormation resources like\n * CloudFront distributions and API Gateway domains.\n *\n * CloudFormation export name must specify the resource type and record hostname as follows:\n * ClouDNS:CNAME:myhost:example:org\n *\n * CloudFormation export value must specify the record value as-is (for instance, a distribution domain name):\n * xxxxxxxxxxxxxx.cloudfront.net\n *\n * The above example will generate the following record in the ClouDNS zone example.org:\n * myhost.example.org CNAME xxxxxxxxxxxxxx.cloudfront.net\n *\n * Other resource types are also allowed (A, AAAA, ALIAS, etc).\n *\n * Command line usage: AWS_PROFILE=xxx ts-node cloudns-cloudformation-sync.ts <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]\n *\n * AWS_PROFILE=xxx - Specify your AWS profile in ~/.aws/credentials as an environment variable\n * <cloudns-username> - ClouDNS API sub-auth-user\n * <cloudns-password-parameter-name> - SSM Parameter with the encrypted ClouDNS API password\n * [ttl] - Optional TTL for generated records (defaults to 300)\n * [stackName] - Optional CloudFormation stack name to limit the exports to scan (defaults to all stacks)\n */\nimport { SSMClient, GetParameterCommand } from '@aws-sdk/client-ssm'\nimport { CloudFormationClient, ListExportsCommand, ListExportsOutput } from '@aws-sdk/client-cloudformation'\nimport fetch from 'node-fetch'\nimport * as querystring from 'querystring'\n\n// Load ~/.aws/config\nprocess.env.AWS_SDK_LOAD_CONFIG = '1'\n\nasync function cloudnsRestCall(cloudnsUsername: string, cloudnsPassword: string, method: string, relativeUrl: string, queryOptions: any) {\n  let fullUrl =\n    'https://api.cloudns.net' +\n    relativeUrl +\n    '?' +\n    querystring.stringify(\n      Object.assign(\n        {\n          'sub-auth-user': cloudnsUsername,\n          'auth-password': cloudnsPassword,\n        },\n        queryOptions || {}\n      )\n    )\n\n  // console.log('Note: Calling', fullUrl)\n\n  const response = await fetch(fullUrl, {\n    method: method,\n    headers: {\n      'Content-Type': 'application/json',\n      Accept: 'application/json',\n    },\n  })\n  if (!response.ok) {\n    const errorText = await response.text()\n    console.error('HTTP Error', response.status, response.statusText, errorText)\n    throw new Error(errorText)\n  }\n  return response.json()\n}\n\nasync function autoDetectCloudnsHostAndZone(cloudnsUsername: string, cloudnsPassword: string, name: string, zoneCache: any) {\n  const nameParts = name.split('.')\n\n  // Zone and host name for xxx.tld\n  const hostName1 = nameParts.slice(0, nameParts.length - 2).join('.')\n  const zoneName1 = nameParts.slice(nameParts.length - 2).join('.')\n\n  // Zone and host name for xxx.subtld.tld\n  const hostName2 = nameParts.slice(0, nameParts.length - 3).join('.')\n  const zoneName2 = nameParts.slice(nameParts.length - 3).join('.')\n\n  // Check which zone exists\n  const zoneResponse1 =\n    zoneCache[zoneName1] ||\n    (await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {\n      'domain-name': zoneName1,\n    }))\n  zoneCache[zoneName1] = zoneResponse1\n  const zoneResponse2 =\n    zoneCache[zoneName2] ||\n    (await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {\n      'domain-name': zoneName2,\n    }))\n  zoneCache[zoneName2] = zoneResponse2\n\n  // console.log('Note: Response for host', hostName1, 'in zone', zoneName1, ':', zoneResponse1)\n  // console.log('Note: Response for host', hostName2, 'in zone', zoneName2, ':', zoneResponse2)\n\n  const zoneName = zoneResponse1.status === '1' ? zoneName1 : zoneResponse2.status === '1' ? zoneName2 : ''\n  const hostName = zoneResponse1.status === '1' ? hostName1 : zoneResponse2.status === '1' ? hostName2 : ''\n  if (!zoneName) {\n    // Neither zone exists\n    throw new Error('Zone Not Found: ' + name)\n  }\n  return {\n    hostName: hostName,\n    zoneName: zoneName,\n  }\n}\n\nasync function createOrUpdateCloudnsResource(\n  cloudnsUsername: string,\n  cloudnsPassword: string,\n  name: string,\n  type: string,\n  value: string,\n  ttlValue: string,\n  zoneCache: any\n) {\n  const { zoneName, hostName } = await autoDetectCloudnsHostAndZone(cloudnsUsername, cloudnsPassword, name, zoneCache)\n  // Does the record exist?\n  const recordsResponse = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/records.json', {\n    'domain-name': zoneName,\n    host: hostName,\n    type: type,\n  })\n  const existingRecord: any = Object.values(recordsResponse)[0]\n  if (existingRecord?.host === hostName && existingRecord?.type === type && existingRecord?.ttl === ttlValue && existingRecord?.record === value) {\n    // Record exists already - no change\n    console.log('OK', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName)\n  } else if (existingRecord?.id) {\n    // Update record\n    console.log('UPDATE', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName)\n    const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/mod-record.json', {\n      'domain-name': zoneName,\n      'record-id': existingRecord?.id,\n      host: hostName,\n      'record-type': type,\n      record: value,\n      ttl: ttlValue,\n    })\n    if (result.status === 'Failed') {\n      throw new Error('Modify record failed: ' + (result.statusMessage || result.statusDescription))\n    }\n  } else {\n    // Create record\n    console.log('CREATE', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName)\n    const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/add-record.json', {\n      'domain-name': zoneName,\n      host: hostName,\n      'record-type': type,\n      record: value,\n      ttl: ttlValue,\n    })\n    if (result.status === 'Failed') {\n      throw new Error('Add record failed: ' + (result.statusMessage || result.statusDescription))\n    }\n  }\n}\n\nexport async function main() {\n  console.log('ClouDNS CloudFormation Sync by Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020-2023')\n  const cloudnsUsername = process.argv[2]\n  const cloudnsPasswordParameter = process.argv[3]\n  const ttlValue = process.argv[4] || '300'\n  const stackName = process.argv[5] || ''\n  if (!cloudnsUsername) {\n    console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]')\n    process.exit(1)\n  }\n  if (!cloudnsPasswordParameter) {\n    console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]')\n    process.exit(1)\n  }\n\n  const ssm = new SSMClient({})\n  const zoneCache = {}\n\n  const response = await ssm.send(\n    new GetParameterCommand({\n      Name: cloudnsPasswordParameter,\n      WithDecryption: true,\n    })\n  )\n  const cloudnsPassword = response.Parameter?.Value || ''\n\n  const cloudFormation = new CloudFormationClient({})\n  let nextToken\n  do {\n    const response: ListExportsOutput = await cloudFormation.send(\n      new ListExportsCommand({\n        NextToken: nextToken,\n      })\n    )\n    for (const exportObj of response.Exports || []) {\n      if (stackName && exportObj.ExportingStackId !== stackName) {\n        // Check if the name part of the ID matches arn:aws:cloudformation:eu-west-1:<xxx>:stack/<name>/<xxx>\n        const m = exportObj.ExportingStackId?.match(/^arn:[^:]+:cloudformation:[^:]+:[^:]+:stack\\/([^\\/]+)\\//)\n        if (!m || m[1] !== stackName) {\n          // Stack ID name part didn't match given stackName, so skip it\n          continue\n        }\n      }\n      if (exportObj.Name?.match(/^ClouDNS:/)) {\n        const nameParts = exportObj.Name.split(':')\n        const resourceType = nameParts[1]\n        const resourceName = nameParts.slice(2).join('.')\n        const resourceValue = exportObj.Value!\n        await createOrUpdateCloudnsResource(cloudnsUsername, cloudnsPassword, resourceName, resourceType, resourceValue, ttlValue, zoneCache)\n      }\n    }\n    nextToken = response.NextToken\n  } while (nextToken)\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "cloudns-cloudformation-sync",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.3.0",
|
|
4
4
|
"description": "Copyright (C) Clouden Oy 2023",
|
|
5
5
|
"main": "lib/cloudns-cloudformation-sync.js",
|
|
6
6
|
"bin": {
|
|
@@ -28,13 +28,13 @@
|
|
|
28
28
|
},
|
|
29
29
|
"homepage": "https://github.com/cloudeninc/cloudns-cloudformation-sync#readme",
|
|
30
30
|
"dependencies": {
|
|
31
|
-
"@aws-sdk/client-cloudformation": "^3.
|
|
32
|
-
"@aws-sdk/client-ssm": "^3.
|
|
31
|
+
"@aws-sdk/client-cloudformation": "^3.454.0",
|
|
32
|
+
"@aws-sdk/client-ssm": "^3.454.0",
|
|
33
33
|
"node-fetch": "^2.6.0"
|
|
34
34
|
},
|
|
35
35
|
"devDependencies": {
|
|
36
|
-
"@types/node": "^
|
|
36
|
+
"@types/node": "^20.10.0",
|
|
37
37
|
"@types/node-fetch": "^2.5.7",
|
|
38
|
-
"typescript": "^
|
|
38
|
+
"typescript": "^5.3.2"
|
|
39
39
|
}
|
|
40
40
|
}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Read AWS CloudFormation Exports and autogenerate ClouDNS records based on their names and values.
|
|
3
|
-
* Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020
|
|
3
|
+
* Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020-2023
|
|
4
4
|
*
|
|
5
5
|
* This tool can be used to autogenerate ClouDNS records for CloudFormation resources like
|
|
6
6
|
* CloudFront distributions and API Gateway domains.
|
|
@@ -16,12 +16,13 @@
|
|
|
16
16
|
*
|
|
17
17
|
* Other resource types are also allowed (A, AAAA, ALIAS, etc).
|
|
18
18
|
*
|
|
19
|
-
* Command line usage: AWS_PROFILE=xxx ts-node cloudns-cloudformation-sync.ts <cloudns-username> <cloudns-password-parameter-name> [ttl]
|
|
19
|
+
* Command line usage: AWS_PROFILE=xxx ts-node cloudns-cloudformation-sync.ts <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]
|
|
20
20
|
*
|
|
21
21
|
* AWS_PROFILE=xxx - Specify your AWS profile in ~/.aws/credentials as an environment variable
|
|
22
22
|
* <cloudns-username> - ClouDNS API sub-auth-user
|
|
23
23
|
* <cloudns-password-parameter-name> - SSM Parameter with the encrypted ClouDNS API password
|
|
24
24
|
* [ttl] - Optional TTL for generated records (defaults to 300)
|
|
25
|
+
* [stackName] - Optional CloudFormation stack name to limit the exports to scan (defaults to all stacks)
|
|
25
26
|
*/
|
|
26
27
|
import { SSMClient, GetParameterCommand } from '@aws-sdk/client-ssm'
|
|
27
28
|
import { CloudFormationClient, ListExportsCommand, ListExportsOutput } from '@aws-sdk/client-cloudformation'
|
|
@@ -32,10 +33,19 @@ import * as querystring from 'querystring'
|
|
|
32
33
|
process.env.AWS_SDK_LOAD_CONFIG = '1'
|
|
33
34
|
|
|
34
35
|
async function cloudnsRestCall(cloudnsUsername: string, cloudnsPassword: string, method: string, relativeUrl: string, queryOptions: any) {
|
|
35
|
-
let fullUrl =
|
|
36
|
-
'
|
|
37
|
-
|
|
38
|
-
|
|
36
|
+
let fullUrl =
|
|
37
|
+
'https://api.cloudns.net' +
|
|
38
|
+
relativeUrl +
|
|
39
|
+
'?' +
|
|
40
|
+
querystring.stringify(
|
|
41
|
+
Object.assign(
|
|
42
|
+
{
|
|
43
|
+
'sub-auth-user': cloudnsUsername,
|
|
44
|
+
'auth-password': cloudnsPassword,
|
|
45
|
+
},
|
|
46
|
+
queryOptions || {}
|
|
47
|
+
)
|
|
48
|
+
)
|
|
39
49
|
|
|
40
50
|
// console.log('Note: Calling', fullUrl)
|
|
41
51
|
|
|
@@ -43,8 +53,8 @@ async function cloudnsRestCall(cloudnsUsername: string, cloudnsPassword: string,
|
|
|
43
53
|
method: method,
|
|
44
54
|
headers: {
|
|
45
55
|
'Content-Type': 'application/json',
|
|
46
|
-
|
|
47
|
-
}
|
|
56
|
+
Accept: 'application/json',
|
|
57
|
+
},
|
|
48
58
|
})
|
|
49
59
|
if (!response.ok) {
|
|
50
60
|
const errorText = await response.text()
|
|
@@ -58,28 +68,32 @@ async function autoDetectCloudnsHostAndZone(cloudnsUsername: string, cloudnsPass
|
|
|
58
68
|
const nameParts = name.split('.')
|
|
59
69
|
|
|
60
70
|
// Zone and host name for xxx.tld
|
|
61
|
-
const hostName1 = nameParts.slice(0, nameParts.length-2).join('.')
|
|
62
|
-
const zoneName1 = nameParts.slice(nameParts.length-2).join('.')
|
|
71
|
+
const hostName1 = nameParts.slice(0, nameParts.length - 2).join('.')
|
|
72
|
+
const zoneName1 = nameParts.slice(nameParts.length - 2).join('.')
|
|
63
73
|
|
|
64
74
|
// Zone and host name for xxx.subtld.tld
|
|
65
|
-
const hostName2 = nameParts.slice(0, nameParts.length-3).join('.')
|
|
66
|
-
const zoneName2 = nameParts.slice(nameParts.length-3).join('.')
|
|
75
|
+
const hostName2 = nameParts.slice(0, nameParts.length - 3).join('.')
|
|
76
|
+
const zoneName2 = nameParts.slice(nameParts.length - 3).join('.')
|
|
67
77
|
|
|
68
78
|
// Check which zone exists
|
|
69
|
-
const zoneResponse1 =
|
|
70
|
-
|
|
71
|
-
|
|
79
|
+
const zoneResponse1 =
|
|
80
|
+
zoneCache[zoneName1] ||
|
|
81
|
+
(await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {
|
|
82
|
+
'domain-name': zoneName1,
|
|
83
|
+
}))
|
|
72
84
|
zoneCache[zoneName1] = zoneResponse1
|
|
73
|
-
const zoneResponse2 =
|
|
74
|
-
|
|
75
|
-
|
|
85
|
+
const zoneResponse2 =
|
|
86
|
+
zoneCache[zoneName2] ||
|
|
87
|
+
(await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/get-zone-info.json', {
|
|
88
|
+
'domain-name': zoneName2,
|
|
89
|
+
}))
|
|
76
90
|
zoneCache[zoneName2] = zoneResponse2
|
|
77
91
|
|
|
78
92
|
// console.log('Note: Response for host', hostName1, 'in zone', zoneName1, ':', zoneResponse1)
|
|
79
93
|
// console.log('Note: Response for host', hostName2, 'in zone', zoneName2, ':', zoneResponse2)
|
|
80
94
|
|
|
81
|
-
const zoneName =
|
|
82
|
-
const hostName =
|
|
95
|
+
const zoneName = zoneResponse1.status === '1' ? zoneName1 : zoneResponse2.status === '1' ? zoneName2 : ''
|
|
96
|
+
const hostName = zoneResponse1.status === '1' ? hostName1 : zoneResponse2.status === '1' ? hostName2 : ''
|
|
83
97
|
if (!zoneName) {
|
|
84
98
|
// Neither zone exists
|
|
85
99
|
throw new Error('Zone Not Found: ' + name)
|
|
@@ -90,13 +104,21 @@ async function autoDetectCloudnsHostAndZone(cloudnsUsername: string, cloudnsPass
|
|
|
90
104
|
}
|
|
91
105
|
}
|
|
92
106
|
|
|
93
|
-
async function createOrUpdateCloudnsResource(
|
|
107
|
+
async function createOrUpdateCloudnsResource(
|
|
108
|
+
cloudnsUsername: string,
|
|
109
|
+
cloudnsPassword: string,
|
|
110
|
+
name: string,
|
|
111
|
+
type: string,
|
|
112
|
+
value: string,
|
|
113
|
+
ttlValue: string,
|
|
114
|
+
zoneCache: any
|
|
115
|
+
) {
|
|
94
116
|
const { zoneName, hostName } = await autoDetectCloudnsHostAndZone(cloudnsUsername, cloudnsPassword, name, zoneCache)
|
|
95
117
|
// Does the record exist?
|
|
96
118
|
const recordsResponse = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'GET', '/dns/records.json', {
|
|
97
119
|
'domain-name': zoneName,
|
|
98
|
-
|
|
99
|
-
|
|
120
|
+
host: hostName,
|
|
121
|
+
type: type,
|
|
100
122
|
})
|
|
101
123
|
const existingRecord: any = Object.values(recordsResponse)[0]
|
|
102
124
|
if (existingRecord?.host === hostName && existingRecord?.type === type && existingRecord?.ttl === ttlValue && existingRecord?.record === value) {
|
|
@@ -108,60 +130,73 @@ async function createOrUpdateCloudnsResource(cloudnsUsername: string, cloudnsPas
|
|
|
108
130
|
const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/mod-record.json', {
|
|
109
131
|
'domain-name': zoneName,
|
|
110
132
|
'record-id': existingRecord?.id,
|
|
111
|
-
|
|
133
|
+
host: hostName,
|
|
112
134
|
'record-type': type,
|
|
113
|
-
|
|
114
|
-
|
|
135
|
+
record: value,
|
|
136
|
+
ttl: ttlValue,
|
|
115
137
|
})
|
|
116
138
|
if (result.status === 'Failed') {
|
|
117
|
-
throw new Error(
|
|
139
|
+
throw new Error('Modify record failed: ' + (result.statusMessage || result.statusDescription))
|
|
118
140
|
}
|
|
119
141
|
} else {
|
|
120
142
|
// Create record
|
|
121
143
|
console.log('CREATE', name, type, ttlValue, value, 'ZONE', zoneName, 'HOST', hostName)
|
|
122
144
|
const result = await cloudnsRestCall(cloudnsUsername, cloudnsPassword, 'POST', '/dns/add-record.json', {
|
|
123
145
|
'domain-name': zoneName,
|
|
124
|
-
|
|
146
|
+
host: hostName,
|
|
125
147
|
'record-type': type,
|
|
126
|
-
|
|
127
|
-
|
|
148
|
+
record: value,
|
|
149
|
+
ttl: ttlValue,
|
|
128
150
|
})
|
|
129
151
|
if (result.status === 'Failed') {
|
|
130
|
-
throw new Error(
|
|
152
|
+
throw new Error('Add record failed: ' + (result.statusMessage || result.statusDescription))
|
|
131
153
|
}
|
|
132
154
|
}
|
|
133
155
|
}
|
|
134
156
|
|
|
135
157
|
export async function main() {
|
|
136
|
-
console.log('ClouDNS CloudFormation Sync by Kenneth Falck <kennu@clouden.net> (C) Clouden Oy
|
|
158
|
+
console.log('ClouDNS CloudFormation Sync by Kenneth Falck <kennu@clouden.net> (C) Clouden Oy 2020-2023')
|
|
137
159
|
const cloudnsUsername = process.argv[2]
|
|
138
160
|
const cloudnsPasswordParameter = process.argv[3]
|
|
139
161
|
const ttlValue = process.argv[4] || '300'
|
|
162
|
+
const stackName = process.argv[5] || ''
|
|
140
163
|
if (!cloudnsUsername) {
|
|
141
|
-
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl]')
|
|
164
|
+
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]')
|
|
142
165
|
process.exit(1)
|
|
143
166
|
}
|
|
144
167
|
if (!cloudnsPasswordParameter) {
|
|
145
|
-
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl]')
|
|
168
|
+
console.error('Usage: cloudns-cloudformation-sync <cloudns-username> <cloudns-password-parameter-name> [ttl [stackName]]')
|
|
146
169
|
process.exit(1)
|
|
147
170
|
}
|
|
148
171
|
|
|
149
172
|
const ssm = new SSMClient({})
|
|
150
173
|
const zoneCache = {}
|
|
151
174
|
|
|
152
|
-
const response = await ssm.send(
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
175
|
+
const response = await ssm.send(
|
|
176
|
+
new GetParameterCommand({
|
|
177
|
+
Name: cloudnsPasswordParameter,
|
|
178
|
+
WithDecryption: true,
|
|
179
|
+
})
|
|
180
|
+
)
|
|
156
181
|
const cloudnsPassword = response.Parameter?.Value || ''
|
|
157
182
|
|
|
158
183
|
const cloudFormation = new CloudFormationClient({})
|
|
159
184
|
let nextToken
|
|
160
185
|
do {
|
|
161
|
-
const response: ListExportsOutput = await cloudFormation.send(
|
|
162
|
-
|
|
163
|
-
|
|
186
|
+
const response: ListExportsOutput = await cloudFormation.send(
|
|
187
|
+
new ListExportsCommand({
|
|
188
|
+
NextToken: nextToken,
|
|
189
|
+
})
|
|
190
|
+
)
|
|
164
191
|
for (const exportObj of response.Exports || []) {
|
|
192
|
+
if (stackName && exportObj.ExportingStackId !== stackName) {
|
|
193
|
+
// Check if the name part of the ID matches arn:aws:cloudformation:eu-west-1:<xxx>:stack/<name>/<xxx>
|
|
194
|
+
const m = exportObj.ExportingStackId?.match(/^arn:[^:]+:cloudformation:[^:]+:[^:]+:stack\/([^\/]+)\//)
|
|
195
|
+
if (!m || m[1] !== stackName) {
|
|
196
|
+
// Stack ID name part didn't match given stackName, so skip it
|
|
197
|
+
continue
|
|
198
|
+
}
|
|
199
|
+
}
|
|
165
200
|
if (exportObj.Name?.match(/^ClouDNS:/)) {
|
|
166
201
|
const nameParts = exportObj.Name.split(':')
|
|
167
202
|
const resourceType = nameParts[1]
|
|
@@ -173,4 +208,3 @@ export async function main() {
|
|
|
173
208
|
nextToken = response.NextToken
|
|
174
209
|
} while (nextToken)
|
|
175
210
|
}
|
|
176
|
-
|