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

133 lines
4.1 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.
// searcher implementation
// @Author Lion <chenxin619315@gmail.com>
const fs = require('fs');
const {
parseIP,
HeaderInfoLength, VectorIndexCols, VectorIndexSize,
ipToString
} = require('./util');
class Searcher {
constructor(version, dbPath, vectorIndex, cBuffer) {
this.ioCount = 0;
this.version = version;
if (cBuffer != null) {
this.handle = null;
this.vectorIndex = null;
this.cBuffer = cBuffer;
} else {
this.handle = fs.openSync(dbPath, 'r');
this.vectorIndex = vectorIndex;
this.cBuffer = null;
}
}
getIOCount() {
return this.ioCount;
}
search(ip) {
// check and parse the string ip
const ipBytes = Buffer.isBuffer(ip) ? ip : parseIP(ip);
// ip version check
if (ipBytes.length != this.version.bytes) {
throw new Error(`invalid ip address '${ipToString(ipBytes)}' (${this.version.name} expected)`);
}
// reset the global counter
this.ioCount = 0;
// located the segment index block based on the vector index
let sPtr = 0, ePtr = 0;
let il0 = ipBytes[0], il1 = ipBytes[1];
let idx = il0 * VectorIndexCols * VectorIndexSize + il1 * VectorIndexSize;
if (this.vectorIndex != null) {
sPtr = this.vectorIndex.readUint32LE(idx);
ePtr = this.vectorIndex.readUint32LE(idx + 4);
} else if (this.cBuffer != null) {
sPtr = this.cBuffer.readUint32LE(HeaderInfoLength + idx);
ePtr = this.cBuffer.readUint32LE(HeaderInfoLength + idx + 4);
} else {
const buff = Buffer.alloc(VectorIndexSize);
this.read(HeaderInfoLength + idx, buff);
sPtr = buff.readUInt32LE(0);
ePtr = buff.readUInt32LE(4);
}
// console.log(`sPtr: ${sPtr}, ePtr: ${ePtr}`);
// binary search the segment index block to get the region info
const bytes = ipBytes.length, dBytes = ipBytes.length << 1;
const indexSize = this.version.indexSize;
const buff = Buffer.alloc(indexSize);
let dLen = -1, dPtr = -1, l = 0, h = (ePtr - sPtr) / indexSize;
while (l <= h) {
const m = (l + h) >> 1;
const p = sPtr + m * indexSize;
// read the segment index block
this.read(p, buff);
if (this.version.ipSubCompare(ipBytes, buff, 0) < 0) {
h = m - 1;
} else if (this.version.ipSubCompare(ipBytes, buff, bytes) > 0) {
l = m + 1;
} else {
dLen = buff.readUint16LE(dBytes);
dPtr = buff.readUint32LE(dBytes + 2);
break;
}
}
// console.log(`dLen: ${dLen}, dPtr: ${dPtr}`);
const region = Buffer.alloc(dLen);
this.read(dPtr, region);
return region.toString('utf-8');
}
read(offset, buff, stats) {
// check the in-memory buffer first
if (this.cBuffer != null) {
this.cBuffer.copy(buff, 0, offset, offset + buff.length);
return;
}
// increase the io counts
this.ioCount++;
// read the data
let rBytes = fs.readSync(this.handle, buff, 0, buff.length, offset);
if (rBytes != buff.length) {
throw new Error(`incomplete read: read bytes should be ${buff.length}`);
}
}
// close the searcher
close() {
if (this.handle != null) {
fs.close(this.handle);
}
}
}
function newWithFileOnly(version, dbPath) {
return new Searcher(version, dbPath, null, null);
}
function newWithVectorIndex(version, dbPath, vectorIndex) {
return new Searcher(version, dbPath, vectorIndex, null);
}
function newWithBuffer(version, cBuffer) {
return new Searcher(version, null, null, cBuffer);
}
module.exports = {
newWithFileOnly,
newWithVectorIndex,
newWithBuffer
}