2025-10-13 12:05:20 +08:00

396 lines
9.8 KiB
JavaScript

// Copyright 2022 The Ip2Region Authors. All rights reserved.
// Use of this source code is governed by a Apache2.0-style
// license that can be found in the LICENSE file.
// util functions
// @Author Lion <chenxin619315@gmail.com>
const fs = require('fs');
const XdbStructure20 = 2;
const XdbStructure30 = 3;
const XdbIPv4Id = 4;
const XdbIPv6Id = 6;
const HeaderInfoLength = 256;
const VectorIndexRows = 256;
const VectorIndexCols = 256;
const VectorIndexSize = 8;
class Header {
constructor(buff) {
this.version = buff.readUInt16LE(0);
this.indexPolicy = buff.readUInt16LE(2);
this.createdAt = buff.readUInt32LE(4);
this.startIndexPtr = buff.readUInt32LE(8);
this.endIndexPtr = buff.readUInt32LE(12);
// since IPv6 supporting
this.ipVersion = buff.readUInt16LE(16);
this.runtimePtrBytes = buff.readUInt16LE(18);
// keep the raw data
this.buff = buff;
}
toString() {
return `{
"version":${this.version},
"index_policy":${this.indexPolicy},
"start_index_ptr": ${this.startIndexPtr},
"end_index_ptr": ${this.endIndexPtr},
"ipVersion": ${this.ipVersion},
"runtime_ptr_bytes": ${this.runtimePtrBytes}
}`;
}
}
// ---
// parse ipv4 address
function _parse_ipv4_addr(v4String) {
let ps = v4String.split('.', 4);
if (ps.length != 4) {
throw new Error('invalid ipv4 address');
}
var v;
const ipBytes = Buffer.alloc(4);
for (var i = 0; i < ps.length; i++) {
v = parseInt(ps[i], 10);
if (isNaN(v)) {
throw new Error(`invalid ipv4 part '${ps[i]}', a valid number expected`);
}
if (v < 0 || v > 255) {
throw new Error(`invalid ipv4 part '${ps[i]}' should >= 0 and <= 255`);
}
ipBytes[i] = (v & 0xFF);
}
return ipBytes;
}
// parse ipv6 address
function _parse_ipv6_addr(v6String) {
let ps = v6String.split(':', 8);
if (ps.length < 3) {
throw new Error('invalid ipv6 address');
}
let dc_num = 0, offset = 0;
const ipBytes = Buffer.alloc(16);
for (var i = 0; i < ps.length; i++) {
let s = ps[i].trim();
// Double colon check and auto padding
if (s.length == 0) {
// ONLY one double colon allow
if (dc_num > 0) {
throw new Error('invalid ipv6 address: multi double colon detected');
}
let start = i, mi = ps.length - 1;
// clear all the consecutive spaces
for (i++;;) {
s = ps[i].trim();
if (s.length > 0) {
i--;
break;
}
if (i >= mi) {
break;
}
i++;
}
dc_num = 1;
// padding = 8 - start - left
let padding = 8 - start - (mi - i);
offset += 2 * padding;
continue;
}
let v = parseInt(s, 16);
if (isNaN(v)) {
throw new Error(`invalid ipv6 part '${ps[i]}', a valid hex number expected`);
}
if (v < 0 || v > 0xFFFF) {
throw new Error(`invalid ipv6 part '${ps[i]}' should >= 0 and <= 65534`);
}
ipBytes.writeUint16BE(v, offset);
offset += 2;
}
return ipBytes;
}
// parse the specified string ip and return its bytes
// @param ip string
// @return Buffer
function parseIP(ipString) {
let sDot = ipString.indexOf('.');
let cDot = ipString.indexOf(':');
if (sDot > -1 && cDot == -1) {
return _parse_ipv4_addr(ipString);
} else if (cDot > -1) {
return _parse_ipv6_addr(ipString);
} else {
throw new Error(`invalid ip address '${ipString}'`);
}
}
// ---
// ipv4 bytes to string
function _ipv4_to_string(v4Bytes) {
return v4Bytes.join('.');
}
// ipv6 bytes to string
function _ipv6_to_string(v6Bytes, compress) {
let ps = [];
for (var i = 0; i < v6Bytes.length; i += 2) {
ps.push(v6Bytes.readUint16BE(i).toString(16));
}
if (compress === false) {
return ps.join(':');
}
// auto compression of consecutive zero
let j = 0, mi = ps.length - 1;
let _ = [];
for (i = 0; i < ps.length; i++) {
if (i >= mi || j > 0) {
_.push(ps[i]);
continue;
}
if (ps[i] != '0' || ps[i+1] != '0') {
_.push(ps[i]);
continue;
}
// find the first two zero part
// and keep find all the zero part
for (i += 2; i < ps.length;) {
if (ps[i] != '0') {
i--;
break;
}
i++;
}
// make sure there is an empty head.
if (_.length == 0) {
_.push('');
}
_.push(''); // empty for double colon
// make sure there is an empty tail
if (i == ps.length && _.length == 2) {
_.push('');
}
}
return _.join(':');
}
// bytes ip to humen-readable string ip
function ipToString(ipBytes, compress) {
if (!Buffer.isBuffer(ipBytes)) {
throw new Error('invalid bytes ip, not a Buffer');
}
if (ipBytes.length == 4) {
return _ipv4_to_string(ipBytes, compress);
} else if (ipBytes.length == 16) {
return _ipv6_to_string(ipBytes, compress);
} else {
throw new Error('invalid bytes ip with length not 4 or 16');
}
}
function ipBytesString(ipBytes) {
if (!Buffer.isBuffer(ipBytes)) {
throw new Error('invalid bytes ip, not a Buffer');
}
let ps = [];
for (var i = 0; i < ipBytes.length; i++) {
ps.push(ipBytes[i] & 0xFF);
}
return ps.join('.');
}
// compare two byte ips
// ip2 = buff[offset:ip1.length]
// returns: -1 if ip1 < ip2, 1 if ip1 > ip2 or 0
function ipSubCompare(ip1, buff, offset) {
return ip1.compare(buff, offset, offset + ip1.length);
}
function ipCompare(ip1, ip2) {
return ipSubCompare(ip1, ip2, 0);
}
// ---
class Version {
constructor(id, name, bytes, indexSize, ipCompareFunc) {
this.id = id;
this.name = name;
this.bytes = bytes;
this.indexSize = indexSize;
this.ipCompareFunc = ipCompareFunc;
}
ipCompare(ip1, ip2) {
return this.ipCompareFunc(ip1, ip2, 0);
}
ipSubCompare(ip1, ip2, offset) {
return this.ipCompareFunc(ip1, ip2, offset);
}
toString() {
return `{"id": ${this.id}, "name": "${this.name}", "bytes":${this.bytes}, "index_size": ${this.indexSize}}`;
}
}
// 14 = 4 + 4 + 2 + 4
const IPv4 = new Version(XdbIPv4Id, "IPv4", 4, 14, function(ip1, buff, offset){
// ip1: Big endian byte order parsed from input
// ip2: Little endian byte order read from xdb index.
// @Note: to compatible with the old Litten endian index encode implementation.
let i, j = offset + ip1.length - 1;
for (i = 0; i < ip1.length; i++, j--) {
const i1 = ip1[i] & 0xFF;
const i2 = buff[j] & 0xFF;
if (i1 < i2) {
return -1;
}
if (i1 > i2) {
return 1;
}
}
return 0;
});
// 38 = 16 + 16 + 2 + 4
const IPv6 = new Version(XdbIPv6Id, "IPv6", 16, 38, ipSubCompare);
function versionFromName(name) {
let n = name.toUpperCase();
if (n == "V4" || n == "IPV4") {
return IPv4;
} else if (n == "V6" || n == "IPV6") {
return IPv6;
} else {
return null;
}
}
function versionFromHeader(h) {
let v = h.version();
// old structure with ONLY IPv4 supporting
if (v == XdbStructure20) {
return IPv4;
}
// structure 3.0 with IPv6 supporting
if (v != XdbStructure30) {
return null;
}
let ipVer = h.ipVersion();
if (ipVer == XdbIPv4Id) {
return IPv4;
} else if (ipVer == XdbIPv6Id) {
return IPv6;
} else {
return null;
}
}
// ---
function loadHeader(fd) {
const buffer = Buffer.alloc(HeaderInfoLength);
const rBytes = fs.readSync(fd, buffer, 0, HeaderInfoLength, 0);
if (rBytes != HeaderInfoLength) {
throw new Error(`incomplete read (${rBytes} read, ${header.HeaderInfoLength} expected)`);
}
return new Header(buffer);
}
function loadHeaderFromFile(dbPath) {
const fd = fs.openSync(dbPath, "r");
const header = loadHeader(fd);
fs.closeSync(fd);
return header;
}
function loadVectorIndex(fd) {
const vBytes = VectorIndexCols * VectorIndexRows * VectorIndexSize;
const buffer = Buffer.alloc(vBytes);
const rBytes = fs.readSync(fd, buffer, 0, vBytes, HeaderInfoLength);
if (rBytes != vBytes) {
throw new Error(`incomplete read (${rBytes} read, ${vBytes} expected)`);
}
return buffer;
}
function loadVectorIndexFromFile(dbPath) {
const fd = fs.openSync(dbPath, "r");
const vIndex = loadVectorIndex(fd);
fs.closeSync(fd);
return vIndex;
}
function loadContent(fd) {
const stats = fs.fstatSync(fd);
const buffer = Buffer.alloc(stats.size);
const rBytes = fs.readSync(fd, buffer, 0, buffer.length, 0);
if (rBytes != stats.size) {
throw new Error(`incomplete read (${rBytes} read, ${stats.size} expected)`);
}
return buffer;
}
function loadContentFromFile(dbPath) {
const fd = fs.openSync(dbPath, "r");
const content = loadContent(fd);
fs.close(fd, function(){});
return content;
}
module.exports = {
// header
XdbStructure20, XdbStructure30, XdbIPv4Id, XdbIPv6Id,
HeaderInfoLength, VectorIndexRows, VectorIndexCols, VectorIndexSize,
// version
IPv4, IPv6,
versionFromName, versionFromHeader,
// utils
parseIP, ipToString, ipBytesString,
ipSubCompare, ipCompare,
loadHeader, loadHeaderFromFile,
loadVectorIndex, loadVectorIndexFromFile,
loadContent, loadContentFromFile
}