2020-11-08 20:33:07 +00:00
|
|
|
import * as faceapi from "@vladmandic/face-api";
|
|
|
|
import canvas from "canvas";
|
2020-11-09 01:57:57 +00:00
|
|
|
import fs, { lstatSync } from "fs";
|
2020-11-08 20:33:07 +00:00
|
|
|
import * as path from "path";
|
2020-11-27 06:15:46 +00:00
|
|
|
import { LabeledFaceDescriptors, TNetInput } from "@vladmandic/face-api";
|
2020-11-09 01:57:57 +00:00
|
|
|
import * as mime from "mime-types";
|
|
|
|
import dotenv from "dotenv-extended";
|
2020-11-09 03:14:45 +00:00
|
|
|
import { getFaceDetectorOptions } from "../src/common";
|
2020-11-08 20:33:07 +00:00
|
|
|
require("@tensorflow/tfjs-node");
|
|
|
|
|
|
|
|
const { Canvas, Image, ImageData } = canvas;
|
|
|
|
//@ts-ignore
|
|
|
|
faceapi.env.monkeyPatch({ Canvas, Image, ImageData });
|
|
|
|
|
|
|
|
const main = async () => {
|
2020-11-09 01:57:57 +00:00
|
|
|
dotenv.load({
|
|
|
|
silent: false,
|
|
|
|
errorOnMissing: true,
|
|
|
|
});
|
|
|
|
const inputDir = process.env.REF_IMAGE_DIR as string;
|
|
|
|
const outDir = process.env.TRAINED_MODEL_DIR as string;
|
|
|
|
|
2020-11-08 20:33:07 +00:00
|
|
|
const faceDetectionNet = faceapi.nets.ssdMobilenetv1;
|
|
|
|
await faceDetectionNet.loadFromDisk(path.join(__dirname, "../weights"));
|
|
|
|
await faceapi.nets.faceLandmark68Net.loadFromDisk(
|
|
|
|
path.join(__dirname, "../weights")
|
|
|
|
);
|
|
|
|
await faceapi.nets.faceRecognitionNet.loadFromDisk(
|
|
|
|
path.join(__dirname, "../weights")
|
|
|
|
);
|
|
|
|
|
|
|
|
const options = getFaceDetectorOptions(faceDetectionNet);
|
|
|
|
|
2020-11-09 01:57:57 +00:00
|
|
|
const dirs = fs.readdirSync(inputDir);
|
|
|
|
|
2020-11-27 06:15:46 +00:00
|
|
|
const refs: Array<LabeledFaceDescriptors> = [];
|
2020-11-09 01:57:57 +00:00
|
|
|
for (const dir of dirs) {
|
|
|
|
if (!lstatSync(path.join(inputDir, dir)).isDirectory()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const files = fs.readdirSync(path.join(inputDir, dir));
|
|
|
|
let referenceResults = await Promise.all(
|
|
|
|
files.map(async (file: string) => {
|
|
|
|
const mimeType = mime.contentType(
|
|
|
|
path.extname(path.join(inputDir, dir, file))
|
|
|
|
);
|
|
|
|
if (!mimeType || !mimeType.startsWith("image")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
console.log(path.join(inputDir, dir, file));
|
|
|
|
|
|
|
|
try {
|
|
|
|
const referenceImage = (await canvas.loadImage(
|
|
|
|
path.join(inputDir, dir, file)
|
|
|
|
)) as unknown;
|
|
|
|
|
|
|
|
const descriptor = await faceapi
|
2020-11-27 06:15:46 +00:00
|
|
|
.detectSingleFace(referenceImage as TNetInput, options)
|
2020-11-09 01:57:57 +00:00
|
|
|
.withFaceLandmarks()
|
2020-11-27 06:15:46 +00:00
|
|
|
.withFaceDescriptor();
|
|
|
|
if (!descriptor || !descriptor.descriptor) {
|
|
|
|
throw new Error("No face found");
|
|
|
|
}
|
2020-11-09 01:57:57 +00:00
|
|
|
|
2020-11-27 06:15:46 +00:00
|
|
|
const faceDescriptors = [descriptor.descriptor];
|
|
|
|
return new faceapi.LabeledFaceDescriptors(dir, faceDescriptors);
|
2020-11-09 01:57:57 +00:00
|
|
|
} catch (err) {
|
|
|
|
console.log(
|
|
|
|
"An error occurred loading image at path: " +
|
|
|
|
path.join(inputDir, dir, file)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
2020-11-27 06:15:46 +00:00
|
|
|
if (referenceResults) {
|
|
|
|
refs.push(
|
|
|
|
...(referenceResults.filter((e) => e) as LabeledFaceDescriptors[])
|
|
|
|
);
|
2020-11-09 01:57:57 +00:00
|
|
|
}
|
2020-11-27 06:15:46 +00:00
|
|
|
}
|
2020-11-09 01:57:57 +00:00
|
|
|
|
2020-11-27 06:15:46 +00:00
|
|
|
const faceMatcher = new faceapi.FaceMatcher(refs);
|
|
|
|
|
|
|
|
fs.writeFile(
|
|
|
|
path.join(outDir, "data.json"),
|
|
|
|
JSON.stringify(faceMatcher.toJSON()),
|
|
|
|
"utf8",
|
|
|
|
(err) => {
|
|
|
|
if (err) {
|
|
|
|
console.log(`An error occurred while writing data model to file`);
|
2020-11-09 01:57:57 +00:00
|
|
|
}
|
2020-11-27 06:15:46 +00:00
|
|
|
|
|
|
|
console.log(`Successfully wrote data model to file`);
|
|
|
|
}
|
|
|
|
);
|
2020-11-08 20:33:07 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
main();
|