first commit
This commit is contained in:
commit
f3b1fb7324
|
@ -0,0 +1,2 @@
|
||||||
|
node_modules
|
||||||
|
ip_ranges
|
|
@ -0,0 +1,2 @@
|
||||||
|
node_modules
|
||||||
|
ip_ranges
|
|
@ -0,0 +1,23 @@
|
||||||
|
FROM node:21
|
||||||
|
|
||||||
|
# Install dependencies
|
||||||
|
RUN apt-get update && \
|
||||||
|
apt-get install -y bgpq3 && \
|
||||||
|
apt-get clean && \
|
||||||
|
rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# Create app directory
|
||||||
|
WORKDIR /usr/src/app
|
||||||
|
|
||||||
|
# Install app dependencies
|
||||||
|
COPY package*.json ./
|
||||||
|
RUN npm install
|
||||||
|
|
||||||
|
# Bundle app source
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
RUN node buildIpRange.js
|
||||||
|
|
||||||
|
# Expose port and start the application
|
||||||
|
EXPOSE 3000
|
||||||
|
CMD [ "node", "server.js" ]
|
|
@ -0,0 +1,66 @@
|
||||||
|
import fs from "fs";
|
||||||
|
import path from "path";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
|
||||||
|
import * as csv from "csv-parse";
|
||||||
|
|
||||||
|
import { exec } from "child_process";
|
||||||
|
import { getCloudASN } from "./getCloudASN.js";
|
||||||
|
import { generateWellKnownIPs } from "./wellKnown";
|
||||||
|
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
|
||||||
|
const getCloudASN = async () =>
|
||||||
|
new Promise(async (resolve, reject) => {
|
||||||
|
const results = [];
|
||||||
|
const cloudList = (
|
||||||
|
await fs.promises.readFile(path.join(__dirname, "./cloud_list.md"))
|
||||||
|
)
|
||||||
|
.toString()
|
||||||
|
.split("\n");
|
||||||
|
|
||||||
|
fs.createReadStream("./as.csv")
|
||||||
|
.pipe(csv.parse({ delimiter: ",", from_line: 2 }))
|
||||||
|
.on("data", function ([asn, handle, description]) {
|
||||||
|
if (cloudList.includes(handle) || cloudList.includes(description)) {
|
||||||
|
results.push({ asn, handle, description });
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.on("end", function () {
|
||||||
|
resolve(results);
|
||||||
|
})
|
||||||
|
.on("error", function (error) {
|
||||||
|
console.log(error.message);
|
||||||
|
reject(error.message);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const buildIpRange = async () => {
|
||||||
|
const dir = path.join(__dirname, "ip_ranges");
|
||||||
|
if (!fs.existsSync(path.join(__dirname, "ip_ranges"))) {
|
||||||
|
fs.mkdirSync(path.join(__dirname, "ip_ranges"));
|
||||||
|
}
|
||||||
|
const cloudASM = await getCloudASN();
|
||||||
|
for (const { asn, handle } of cloudASM) {
|
||||||
|
try {
|
||||||
|
await new Promise((resolve, reject) => {
|
||||||
|
exec(
|
||||||
|
`bgpq3 -h whois.radb.net -j AS${asn} > ip_ranges/${handle}.json`,
|
||||||
|
(error, stdout, stderr) => {
|
||||||
|
if (error) {
|
||||||
|
console.error(`exec error: ${error}`);
|
||||||
|
reject(error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`${asn}, ${handle} failed`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
generateWellKnownIPs();
|
||||||
|
};
|
||||||
|
|
||||||
|
buildIpRange();
|
|
@ -0,0 +1,25 @@
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
import ipCheck from "ip";
|
||||||
|
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
const ipDir = path.join(__dirname, "./ip_ranges");
|
||||||
|
|
||||||
|
const database = {};
|
||||||
|
for (const file of fs.readdirSync(ipDir)) {
|
||||||
|
database[path.basename(file, ".json")] = JSON.parse(
|
||||||
|
fs.readFileSync(path.join(ipDir, file)).toString()
|
||||||
|
)["NN"].map(({ prefix }) => prefix);
|
||||||
|
}
|
||||||
|
|
||||||
|
export const checkIp = (ip) => {
|
||||||
|
for (const [name, cidrs] of Object.entries(database)) {
|
||||||
|
for (const cidr of cidrs) {
|
||||||
|
if (ipCheck.cidrSubnet(cidr).contains(ip)) {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
};
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,8 @@
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
export const config = {
|
||||||
|
dbPath: path.join(__dirname, "ip_ranges.db"),
|
||||||
|
};
|
|
@ -0,0 +1,32 @@
|
||||||
|
import fs from "fs";
|
||||||
|
import path from "path";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
|
||||||
|
import * as csv from "csv-parse";
|
||||||
|
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
|
||||||
|
export const getCloudASN = async () =>
|
||||||
|
new Promise(async (resolve, reject) => {
|
||||||
|
const results = [];
|
||||||
|
const cloudList = (
|
||||||
|
await fs.promises.readFile(path.join(__dirname, "./cloud_list.md"))
|
||||||
|
)
|
||||||
|
.toString()
|
||||||
|
.split("\n");
|
||||||
|
|
||||||
|
fs.createReadStream("./as.csv")
|
||||||
|
.pipe(csv.parse({ delimiter: ",", from_line: 2 }))
|
||||||
|
.on("data", function ([asn, handle, description]) {
|
||||||
|
if (cloudList.includes(handle) || cloudList.includes(description)) {
|
||||||
|
results.push({ asn, handle, description });
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.on("end", function () {
|
||||||
|
resolve(results);
|
||||||
|
})
|
||||||
|
.on("error", function (error) {
|
||||||
|
console.log(error.message);
|
||||||
|
reject(error.message);
|
||||||
|
});
|
||||||
|
});
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,28 @@
|
||||||
|
{
|
||||||
|
"name": "datacenters-ip-addresses",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "",
|
||||||
|
"main": "index.js",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
|
},
|
||||||
|
"keywords": [],
|
||||||
|
"author": "",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"axios": "^1.6.8",
|
||||||
|
"csv": "^6.3.9",
|
||||||
|
"csv-parser": "^3.0.0",
|
||||||
|
"csv-writer": "^1.6.0",
|
||||||
|
"fastify": "^4.27.0",
|
||||||
|
"ip": "^2.0.1",
|
||||||
|
"ip-address": "^9.0.5",
|
||||||
|
"lodash": "^4.17.21",
|
||||||
|
"node-html-parser": "^6.1.13",
|
||||||
|
"progress": "^2.0.3",
|
||||||
|
"range-inclusive": "^1.0.2",
|
||||||
|
"rocksdb": "^5.2.1",
|
||||||
|
"whois": "^2.14.0"
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,53 @@
|
||||||
|
import Fastify from "fastify";
|
||||||
|
import helmet from "@fastify/helmet";
|
||||||
|
import rateLimit from "@fastify/rate-limit";
|
||||||
|
import cors from "@fastify/cors";
|
||||||
|
import { checkIp } from "./checkIp.js";
|
||||||
|
|
||||||
|
const fastify = Fastify();
|
||||||
|
|
||||||
|
await fastify.register(helmet);
|
||||||
|
|
||||||
|
await fastify.register(rateLimit, {
|
||||||
|
max: (request) => (request.headers["x-api-key"] ? 100 : 1),
|
||||||
|
timeWindow: "1 second",
|
||||||
|
keyGenerator: (request) => request.headers["x-api-key"] || request.ip,
|
||||||
|
addHeaders: {
|
||||||
|
"x-ratelimit-limit": true,
|
||||||
|
"x-ratelimit-remaining": true,
|
||||||
|
"x-ratelimit-reset": true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await fastify.register(cors);
|
||||||
|
|
||||||
|
fastify.get(
|
||||||
|
"/api/ip/:ip",
|
||||||
|
{
|
||||||
|
schema: {
|
||||||
|
params: {
|
||||||
|
type: "object",
|
||||||
|
properties: {
|
||||||
|
ip: { type: "string", format: "ipv4" },
|
||||||
|
},
|
||||||
|
required: ["ip"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
async function handler(request, reply) {
|
||||||
|
try {
|
||||||
|
const datacenter = checkIp(request.params.ip);
|
||||||
|
reply.header("Content-Type", "application/json");
|
||||||
|
reply.send({ datacenter });
|
||||||
|
} catch (err) {
|
||||||
|
reply.code(500).send({ error: "Internal Server Error" });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fastify.listen({ port: 3000 });
|
||||||
|
} catch (err) {
|
||||||
|
fastify.log.error(err);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
|
@ -0,0 +1,22 @@
|
||||||
|
import axios from "axios";
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
export const aws = async () => {
|
||||||
|
const { data } = await axios.get(
|
||||||
|
"https://ip-ranges.amazonaws.com/ip-ranges.json"
|
||||||
|
);
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
await fs.promises.writeFile(
|
||||||
|
path.join(__dirname, "../ip_ranges/AWS.json"),
|
||||||
|
JSON.stringify({
|
||||||
|
NN: [
|
||||||
|
...data.prefixes.map(({ ip_prefix }) => ip_prefix),
|
||||||
|
...data.ipv6_prefixes.map(({ ipv6_prefix }) => ipv6_prefix),
|
||||||
|
].map((prefix) => ({ prefix })),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// (async () => await aws())();
|
|
@ -0,0 +1,23 @@
|
||||||
|
import axios from "axios";
|
||||||
|
import _ from "lodash";
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
export const azure = async () => {
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
const { data } = await axios.get(
|
||||||
|
"https://download.microsoft.com/download/7/1/D/71D86715-5596-4529-9B13-DA13A5DE5B63/ServiceTags_Public_20240513.json"
|
||||||
|
);
|
||||||
|
const ips = _.flatten(
|
||||||
|
data.values.map(({ properties }) => properties.addressPrefixes)
|
||||||
|
);
|
||||||
|
await fs.promises.writeFile(
|
||||||
|
path.join(__dirname, "../ip_ranges/AZURE.json"),
|
||||||
|
JSON.stringify({
|
||||||
|
NN: ips.map((prefix) => ({ prefix })),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// (async () => console.log(await azure()))();
|
|
@ -0,0 +1,19 @@
|
||||||
|
import axios from "axios";
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
export const cloudflare = async () => {
|
||||||
|
const [{ data: ipv4 }, { data: ipv6 }] = await Promise.all([
|
||||||
|
axios.get("https://www.cloudflare.com/ips-v4"),
|
||||||
|
axios.get("https://www.cloudflare.com/ips-v6"),
|
||||||
|
]);
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
const ips = [...ipv4.split("\n"), ...ipv6.split("\n")];
|
||||||
|
await fs.promises.writeFile(
|
||||||
|
path.join(__dirname, "../ip_ranges/CLOUDFLARE.json"),
|
||||||
|
JSON.stringify({
|
||||||
|
NN: ips.map((prefix) => ({ prefix })),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
};
|
|
@ -0,0 +1,21 @@
|
||||||
|
import axios from "axios";
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
export const gcp = async () => {
|
||||||
|
const { data } = await axios.get(
|
||||||
|
"https://www.gstatic.com/ipranges/cloud.json"
|
||||||
|
);
|
||||||
|
|
||||||
|
const ips = data.prefixes.map(
|
||||||
|
({ ipv4Prefix, ipv6Prefix }) => ipv4Prefix || ipv6Prefix
|
||||||
|
);
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
await fs.promises.writeFile(
|
||||||
|
path.join(__dirname, "../ip_ranges/GCP.json"),
|
||||||
|
JSON.stringify({
|
||||||
|
NN: ips.map((prefix) => ({ prefix })),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
};
|
|
@ -0,0 +1,31 @@
|
||||||
|
import axios from "axios";
|
||||||
|
import _ from "lodash";
|
||||||
|
import htmlParser from "node-html-parser";
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
export const ibm = async () => {
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
const { data } = await axios.get(
|
||||||
|
"https://ondeck.console.cloud.ibm.com/docs/cloud-infrastructure?topic=cloud-infrastructure-ibm-cloud-ip-ranges"
|
||||||
|
);
|
||||||
|
|
||||||
|
const root = htmlParser.parse(data);
|
||||||
|
const ips = _.flattenDeep(
|
||||||
|
["#section-front-end-network", "#section-load-balancer-ips"].map(
|
||||||
|
(cssSelector) =>
|
||||||
|
root
|
||||||
|
.querySelectorAll(`${cssSelector} td:nth-child(3)`)
|
||||||
|
.map((e) => e.text.split("\n").map((l) => l.trim()))
|
||||||
|
)
|
||||||
|
).filter((e) => e);
|
||||||
|
await fs.promises.writeFile(
|
||||||
|
path.join(__dirname, "../ip_ranges/CLOUDFLARE.json"),
|
||||||
|
JSON.stringify({
|
||||||
|
NN: ips.map((prefix) => ({ prefix })),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// (async () => console.log(await ibm()))();
|
|
@ -0,0 +1,9 @@
|
||||||
|
import { aws } from "./aws.js";
|
||||||
|
import { azure } from "./azure.js";
|
||||||
|
import { cloudflare } from "./cloudflare.js";
|
||||||
|
import { gcp } from "./gcp.js";
|
||||||
|
import { ibm } from "./ibm.js";
|
||||||
|
import { oracle } from "./oracle.js";
|
||||||
|
|
||||||
|
export const generateWellKnownIPs = () =>
|
||||||
|
Promise.all([aws(), azure(), cloudflare(), gcp(), ibm(), oracle()]);
|
|
@ -0,0 +1,22 @@
|
||||||
|
import axios from "axios";
|
||||||
|
import _ from "lodash";
|
||||||
|
import fs from "fs";
|
||||||
|
import { fileURLToPath } from "url";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
export const oracle = async () => {
|
||||||
|
const { data } = await axios.get(
|
||||||
|
"https://docs.oracle.com/en-us/iaas/tools/public_ip_ranges.json"
|
||||||
|
);
|
||||||
|
|
||||||
|
const ips = _.flatten(
|
||||||
|
data.regions.map(({ cidrs }) => cidrs.map(({ cidr }) => cidr))
|
||||||
|
);
|
||||||
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
||||||
|
await fs.promises.writeFile(
|
||||||
|
path.join(__dirname, "../ip_ranges/ORACLE.json"),
|
||||||
|
JSON.stringify({
|
||||||
|
NN: ips.map((prefix) => ({ prefix })),
|
||||||
|
})
|
||||||
|
);
|
||||||
|
};
|
Loading…
Reference in New Issue