range_check
Version:
This is a simple module to validate IP address, check ip address version, check if ip is within a range.
193 lines (192 loc) • 4.86 kB
JavaScript
// src/index.ts
import ipaddr from "ipaddr.js";
import ip6 from "ip6";
function isIP(addr) {
const ver = version(addr);
return ver === 4 || ver === 6;
}
function version(addr) {
try {
const parse_addr = ipaddr.parse(addr);
const kind = parse_addr.kind();
if (kind === "ipv4") {
if (ipaddr.IPv4.isValidFourPartDecimal(addr)) {
return 4;
} else {
return 0;
}
} else if (kind === "ipv6") {
return 6;
} else {
return 0;
}
} catch (err) {
return 0;
}
}
function isV4(addr) {
return version(addr) === 4;
}
function isV6(addr) {
return version(addr) === 6;
}
function isRange(range) {
try {
ipaddr.parseCIDR(range);
return true;
} catch (err) {
return false;
}
}
function inRange(addr, range) {
if (typeof range === "string") {
if (range.indexOf("/") !== -1) {
try {
const range_data = range.split("/");
const parse_addr = ipaddr.parse(addr);
const parse_range = ipaddr.parse(range_data[0]);
return parse_addr.match(parse_range, range_data[1]);
} catch (err) {
return false;
}
} else {
addr = isV6(addr) ? ip6.normalize(addr) : addr;
range = isV6(range) ? ip6.normalize(range) : range;
return isIP(range) && addr === range;
}
} else if (range && typeof range === "object") {
for (const check_range in range) {
if (inRange(addr, range[check_range]) === true) {
return true;
}
}
return false;
} else {
return false;
}
}
function isPrivateIP(ip) {
try {
const addr = ipaddr.parse(ip);
const kind = addr.kind();
const range = addr.range();
if (kind === "ipv4") {
return range === "private" || range === "loopback" || ip === "127.0.0.1";
} else if (kind === "ipv6") {
if (range === "ipv4Mapped") {
const ipv4 = addr.toIPv4Address();
return ipv4.range() === "private" || ipv4.range() === "loopback" || ip === "127.0.0.1";
} else {
return range === "uniqueLocal" || range === "loopback" || ip === "::1";
}
}
return false;
} catch (err) {
return false;
}
}
function isIPInRangeOrPrivate(ip, options = { allowAnyPrivate: true }) {
if (options.allowAnyPrivate !== false && isPrivateIP(ip)) {
return true;
}
if (options.ranges) {
return inRange(ip, options.ranges);
}
return false;
}
function storeIP(addr) {
try {
var parse_addr = ipaddr.parse(addr);
var kind = parse_addr.kind();
if (kind === "ipv4") {
return addr;
} else if (kind === "ipv6") {
if (parse_addr.isIPv4MappedAddress()) {
return parse_addr.toIPv4Address().toString();
} else {
return ip6.abbreviate(addr);
}
} else {
return null;
}
} catch (err) {
return null;
}
}
function displayIP(addr) {
try {
var parse_addr = ipaddr.parse(addr);
var kind = parse_addr.kind();
if (kind === "ipv4") {
return addr;
} else if (kind === "ipv6") {
if (parse_addr.isIPv4MappedAddress()) {
return parse_addr.toIPv4Address().toString();
} else {
return ip6.normalize(addr);
}
} else {
return "";
}
} catch (err) {
return "";
}
}
function IPFingerprint(addr) {
const ver = version(addr);
if (ver === 4) {
return `v4:${addr}`;
} else if (ver === 6) {
try {
const parsedAddr = ipaddr.parse(addr);
if (parsedAddr.kind() === "ipv6") {
if (
//@ts-ignore: it exists!
parsedAddr.isIPv4MappedAddress()
) {
const ipv4Addr = parsedAddr.toIPv4Address().toString();
return `v4:${ipv4Addr}`;
}
const parts = parsedAddr.parts;
const prefix = parts.slice(0, 4).map((part) => part.toString(16)).join(":");
return `v6:${prefix}::`;
}
throw new Error("Invalid IPv6 address");
} catch (err) {
throw new Error("Invalid IPv6 address");
}
} else {
throw new Error("Invalid IP address");
}
}
async function IPFingerprintHashed(addr) {
const fingerprint = IPFingerprint(addr);
const [prefix, ipPart] = fingerprint.split(":", 2);
const encoder = new TextEncoder();
const data = encoder.encode(ipPart);
try {
const hashBuffer = await crypto.subtle.digest("SHA-256", data);
const hashArray = Array.from(new Uint8Array(hashBuffer));
const hashHex = hashArray.map((b) => b.toString(16).padStart(2, "0")).join("");
return `${prefix}:${hashHex}`;
} catch (error) {
throw new Error(
"Crypto functionality not available - hashing failed. Use the non-hashed version instead."
);
}
}
export {
IPFingerprint,
IPFingerprintHashed,
displayIP,
inRange,
isIP,
isIPInRangeOrPrivate,
isPrivateIP,
isRange,
isV4,
isV6,
storeIP as searchIP,
storeIP,
version
};