-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathclear-unused-dns.ts
124 lines (105 loc) · 3.64 KB
/
clear-unused-dns.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
async function deleteTXTRecords(zoneId: string) {
const baseUrl = `https://api.cloudflare.com/client/v4/zones/${zoneId}/dns_records`;
// Fetch all DNS records
const getRecords = async () => {
const response = await fetch(`${baseUrl}?per_page=1000`, {
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${api_token}`,
},
});
const data = await response.json();
return data.result.filter(
(record: any) =>
record.type === "TXT" && record.name.startsWith("_acme-challenge")
);
};
const recordsToDelete = await getRecords();
await doBatchDelete(baseUrl, recordsToDelete);
}
async function deleteTunnelCNAMERecords() {
const baseUrl = `https://api.cloudflare.com/client/v4/zones/${zone_id}/dns_records`;
// Fetch all DNS records
const getRecords = async () => {
const response = await fetch(`${baseUrl}?per_page=1000`, {
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${api_token}`,
},
});
const data = await response.json();
return data.result.filter(
(record: any) =>
record.type === "CNAME" && record.name.startsWith("tunnel-")
);
};
const recordsToDelete = await getRecords();
await doBatchDelete(baseUrl, recordsToDelete);
}
async function deleteTunnels() {
const baseUrl = `https://api.cloudflare.com/client/v4/accounts/${account_id}/cfd_tunnel`;
// Fetch all tunnels
const getRecords = async () => {
const response = await fetch(`${baseUrl}?per_page=1000`, {
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${api_token}`,
},
});
const data = await response.json();
return data.result.filter(
(tunnel: any) => tunnel.name.startsWith("tunnel-") && !tunnel.deleted_at
);
};
const recordsToDelete = await getRecords();
await doBatchDelete(baseUrl, recordsToDelete);
}
async function doBatchDelete(url: string, recordsToDelete: any[]) {
const deleteRecord = async (id: string) => {
const response = await fetch(`${url}/${id}`, {
method: "DELETE",
headers: {
"Content-Type": "application/json",
Authorization: `Bearer ${api_token}`,
},
});
if (!response.ok) {
console.error("Failed to delete record:", await response.text());
return;
}
return response.json();
};
console.log("Records to delete:", recordsToDelete.length);
// Batch deletion, 20 records at a time
const batchDelete = async (records: any[]) => {
for (let i = 0; i < records.length; i += 20) {
const batch = records.slice(i, i + 20);
const deletePromises = batch.map((record) => deleteRecord(record.id));
const results = await Promise.all(deletePromises);
results.forEach((result) => console.log("Batch delete result:", result));
}
};
await batchDelete(recordsToDelete);
}
const api_token = process.env.LINKUP_CF_API_TOKEN;
const account_id = process.env.LINKUP_CLOUDFLARE_ACCOUNT_ID;
const zone_id = process.env.LINKUP_CLOUDFLARE_ZONE_ID;
if (!api_token || !account_id || !zone_id) {
console.error("Missing Cloudflare API Token, Account ID or Zone ID");
console.error(
"Please set LINKUP_CF_API_TOKEN, LINKUP_CLOUDFLARE_ACCOUNT_ID, and LINKUP_CLOUDFLARE_ZONE_ID environment variables"
);
process.exit(1);
}
deleteTunnelCNAMERecords();
deleteTunnels();
const zones = process.argv.slice(2);
if (zones.length === 0) {
console.error("No zones specified");
console.error("Usage: clear-unused-dns.ts zone1 zone2 ...");
process.exit(1);
}
for (const zone of zones) {
console.log("Deleting records for zone:", zone);
deleteTXTRecords(zone);
}