|
@ -10,18 +10,20 @@ const port = 2000; |
|
|
app.use(bodyParser.json()); |
|
|
app.use(bodyParser.json()); |
|
|
|
|
|
|
|
|
// Function to load JSON data
|
|
|
// Function to load JSON data
|
|
|
function loadJSON(filePath, callback) { |
|
|
function loadJSON(filePath) { |
|
|
|
|
|
return new Promise((resolve, reject) => { |
|
|
fs.readFile(filePath, 'utf8', (err, data) => { |
|
|
fs.readFile(filePath, 'utf8', (err, data) => { |
|
|
if (err) { |
|
|
if (err) { |
|
|
return callback(err, null); |
|
|
return reject(err); |
|
|
} |
|
|
} |
|
|
try { |
|
|
try { |
|
|
const jsonData = JSON.parse(data); |
|
|
const jsonData = JSON.parse(data); |
|
|
callback(null, jsonData); |
|
|
resolve(jsonData); |
|
|
} catch (parseError) { |
|
|
} catch (parseError) { |
|
|
callback(parseError, null); |
|
|
reject(parseError); |
|
|
} |
|
|
} |
|
|
}); |
|
|
}); |
|
|
|
|
|
}); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
function cosineDistanceMatching(vector1, vector2) { |
|
|
function cosineDistanceMatching(vector1, vector2) { |
|
@ -30,112 +32,105 @@ function cosineDistanceMatching(vector1, vector2) { |
|
|
return Math.sqrt(distance); |
|
|
return Math.sqrt(distance); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
let tree, vectors, jsonEmbeddings |
|
|
let tree, vectors, jsonEmbeddings, filteredEmbeddings; |
|
|
|
|
|
|
|
|
buildVPTree() |
|
|
|
|
|
loadVPTree() |
|
|
|
|
|
|
|
|
|
|
|
// Load tree endpoint
|
|
|
|
|
|
app.post('/search/', (req, res) => { |
|
|
|
|
|
const { vector } = req.body; |
|
|
|
|
|
|
|
|
|
|
|
console.log(vector) |
|
|
|
|
|
|
|
|
|
|
|
if (!Array.isArray(vector) || vector.length !== 7 || !vector.every(num => typeof num === 'number')) { |
|
|
|
|
|
return res.status(400).send({ success: false, message: 'Invalid vector provided. It must be an array of 7 floating-point numbers.' }); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
let idx = queryVPTree(vector) |
|
|
|
|
|
|
|
|
|
|
|
res.send({folder: jsonEmbeddings[idx].folder, image: jsonEmbeddings[idx].image, video: jsonEmbeddings[idx].video, frame: jsonEmbeddings[idx].frame}); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
function queryVPTree(value){ |
|
|
|
|
|
let nearest = tree.search(value) |
|
|
|
|
|
let index = nearest[0].i |
|
|
|
|
|
return index |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
function buildVPTree(){ |
|
|
async function buildVPTree() { |
|
|
// Load JSON data from embeddings.json
|
|
|
// Load JSON data from embeddings.json
|
|
|
const jsonFilePath = path.join(__dirname, 'embeddings.json'); |
|
|
const jsonFilePath = path.join(__dirname, 'embeddings.json'); |
|
|
loadJSON(jsonFilePath, (err, jsonData) => { |
|
|
try { |
|
|
if (err) { |
|
|
jsonEmbeddings = await loadJSON(jsonFilePath); |
|
|
console.error('Error loading JSON:', err); |
|
|
|
|
|
return; |
|
|
|
|
|
} |
|
|
|
|
|
// Extract vectors from the JSON data
|
|
|
|
|
|
jsonEmbeddings = jsonData |
|
|
|
|
|
vectors = jsonEmbeddings.map(item => item.vector); |
|
|
|
|
|
tree = vptree.build(vectors, cosineDistanceMatching) |
|
|
|
|
|
const treeString = tree.stringify(); |
|
|
|
|
|
const fileName = "tree" |
|
|
|
|
|
const filePath = path.join(__dirname, `${fileName}.txt`); |
|
|
|
|
|
|
|
|
|
|
|
fs.writeFile(filePath, treeString, 'utf8', (err) => { |
|
|
|
|
|
if (err) { |
|
|
|
|
|
console.log("Tree did not save to file.") |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
console.log("Tree saved successfully.") |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
function saveVPTree(fileName){ |
|
|
// Filter embeddings
|
|
|
|
|
|
filteredEmbeddings = jsonEmbeddings.filter(item => item.lost !== 1); |
|
|
|
|
|
vectors = filteredEmbeddings.map(item => item.vector); |
|
|
|
|
|
|
|
|
|
|
|
tree = vptree.build(vectors, cosineDistanceMatching); |
|
|
const treeString = tree.stringify(); |
|
|
const treeString = tree.stringify(); |
|
|
|
|
|
const fileName = "sv-tree"; |
|
|
const filePath = path.join(__dirname, `${fileName}.txt`); |
|
|
const filePath = path.join(__dirname, `${fileName}.txt`); |
|
|
|
|
|
|
|
|
fs.writeFile(filePath, treeString, 'utf8', (err) => { |
|
|
await fs.promises.writeFile(filePath, treeString, 'utf8'); |
|
|
if (err) { |
|
|
console.log("Tree saved successfully."); |
|
|
console.log("Tree did not save to file.") |
|
|
} catch (err) { |
|
|
return |
|
|
console.error('Error building VP-tree:', err); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
console.log("Tree saved successfully.") |
|
|
|
|
|
}); |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
function loadVPTree(){ |
|
|
async function loadVPTree() { |
|
|
loadTreeFromDisk("tree", (err, treeData) => { |
|
|
try { |
|
|
if (err) { |
|
|
const treeData = await loadTreeFromDisk("sv-tree"); |
|
|
console.error('Failed to load tree from disk:', err); |
|
|
|
|
|
return; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Rebuild the VP-tree using the saved structure
|
|
|
// Rebuild the VP-tree using the saved structure
|
|
|
|
|
|
const jsonFilePath = path.join(__dirname, 'embeddings.json'); |
|
|
try { |
|
|
try { |
|
|
|
|
|
jsonEmbeddings = await loadJSON(jsonFilePath); |
|
|
|
|
|
filteredEmbeddings = jsonEmbeddings.filter(item => item.lost !== 1); // Filter again after loading
|
|
|
|
|
|
vectors = filteredEmbeddings.map(item => item.vector); |
|
|
|
|
|
|
|
|
|
|
|
console.log(vectors.length); |
|
|
tree = vptree.load(vectors, cosineDistanceMatching, treeData); |
|
|
tree = vptree.load(vectors, cosineDistanceMatching, treeData); |
|
|
console.log('Tree loaded successfully.'); |
|
|
console.log('Tree loaded successfully.'); |
|
|
|
|
|
|
|
|
} catch (loadError) { |
|
|
} catch (loadError) { |
|
|
console.error('Error loading the VP-tree:', loadError); |
|
|
console.error('Error loading the VP-tree:', loadError); |
|
|
} |
|
|
} |
|
|
}); |
|
|
} catch (err) { |
|
|
|
|
|
console.error('Failed to load tree from disk:', err); |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
function loadTreeFromDisk(fileName, callback){ |
|
|
function loadTreeFromDisk(fileName) { |
|
|
const filePath = path.join(__dirname, `${fileName}.txt`); |
|
|
const filePath = path.join(__dirname, `${fileName}.txt`); |
|
|
|
|
|
return new Promise((resolve, reject) => { |
|
|
fs.readFile(filePath, 'utf8', (err, data) => { |
|
|
fs.readFile(filePath, 'utf8', (err, data) => { |
|
|
if (err) { |
|
|
if (err) { |
|
|
console.error('Error reading the file:', err); |
|
|
console.error('Error reading the file:', err); |
|
|
return callback(err, null); |
|
|
return reject(err); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
// Parse the data as a JavaScript object
|
|
|
// Parse the data as a JavaScript object
|
|
|
try { |
|
|
try { |
|
|
const treeData = eval(`(${data})`); // Using eval to convert to object
|
|
|
const treeData = eval(`(${data})`); // Using eval to convert to object
|
|
|
callback(null, treeData); |
|
|
resolve(treeData); |
|
|
} catch (parseError) { |
|
|
} catch (parseError) { |
|
|
console.error('Error parsing the tree data:', parseError); |
|
|
console.error('Error parsing the tree data:', parseError); |
|
|
callback(parseError, null); |
|
|
reject(parseError); |
|
|
|
|
|
} |
|
|
|
|
|
}); |
|
|
|
|
|
}); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
async function initialize() { |
|
|
|
|
|
await buildVPTree(); |
|
|
|
|
|
await loadVPTree(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// Initialize the VP-tree building and loading process
|
|
|
|
|
|
initialize(); |
|
|
|
|
|
|
|
|
|
|
|
app.post('/search/', (req, res) => { |
|
|
|
|
|
const { vector } = req.body; |
|
|
|
|
|
|
|
|
|
|
|
if (!Array.isArray(vector) || vector.length !== 7 || !vector.every(num => typeof num === 'number')) { |
|
|
|
|
|
return res.status(400).send({ success: false, message: 'Invalid vector provided. It must be an array of 7 floating-point numbers.' }); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
let idx = queryVPTree(vector); |
|
|
|
|
|
if (idx !== undefined && idx < filteredEmbeddings.length) { |
|
|
|
|
|
res.send({ |
|
|
|
|
|
folder: filteredEmbeddings[idx].folder, |
|
|
|
|
|
image: filteredEmbeddings[idx].image, |
|
|
|
|
|
video: filteredEmbeddings[idx].video, |
|
|
|
|
|
frame: filteredEmbeddings[idx].frame, |
|
|
|
|
|
lost: filteredEmbeddings[idx].lost |
|
|
}); |
|
|
}); |
|
|
|
|
|
} else { |
|
|
|
|
|
res.status(404).send({ success: false, message: 'No matching entry found.' }); |
|
|
|
|
|
} |
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
function queryVPTree(value) { |
|
|
|
|
|
let nearest = tree.search(value); |
|
|
|
|
|
let index = nearest[0].i; |
|
|
|
|
|
return index; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
app.listen(port, () => { |
|
|
app.listen(port, () => { |
|
|