mirror of https://github.com/vladmandic/human
188 lines
6.6 KiB
TypeScript
188 lines
6.6 KiB
TypeScript
import * as tf from 'dist/tfjs.esm.js';
|
|
import * as image from '../image/image';
|
|
|
|
/** Env class that holds detected capabilities */
|
|
export class Env {
|
|
/** Running in Browser */
|
|
browser: boolean;
|
|
/** Running in NodeJS */
|
|
node: boolean;
|
|
/** Running in WebWorker thread */
|
|
worker: boolean;
|
|
/** Detected platform */
|
|
platform: string = '';
|
|
/** Detected agent */
|
|
agent: string = '';
|
|
/** List of supported backends */
|
|
backends: string[] = [];
|
|
/** Has any work been performed so far */
|
|
initial: boolean;
|
|
/** Are image filters supported? */
|
|
filter: boolean | undefined;
|
|
/** TFJS instance details */
|
|
tfjs: {
|
|
version: undefined | string,
|
|
};
|
|
/** Is offscreenCanvas supported? */
|
|
offscreen: undefined | boolean;
|
|
/** Are performance counter instant values or additive */
|
|
perfadd: boolean = false;
|
|
/** If using tfjs-node get version of underlying tensorflow shared library and if gpu acceleration is enabled */
|
|
tensorflow: {
|
|
version: undefined | string,
|
|
gpu: undefined | boolean,
|
|
} = {
|
|
version: undefined,
|
|
gpu: undefined,
|
|
};
|
|
/** WASM detected capabilities */
|
|
wasm: {
|
|
supported: undefined | boolean,
|
|
backend: undefined | boolean,
|
|
simd: undefined | boolean,
|
|
multithread: undefined | boolean,
|
|
} = {
|
|
supported: undefined,
|
|
backend: undefined,
|
|
simd: undefined,
|
|
multithread: undefined,
|
|
};
|
|
/** WebGL detected capabilities */
|
|
webgl: {
|
|
supported: undefined | boolean,
|
|
backend: undefined | boolean,
|
|
version: undefined | string,
|
|
renderer: undefined | string,
|
|
shader: undefined | string,
|
|
vendor: undefined | string,
|
|
} = {
|
|
supported: undefined,
|
|
backend: undefined,
|
|
version: undefined,
|
|
renderer: undefined,
|
|
shader: undefined,
|
|
vendor: undefined,
|
|
};
|
|
/** WebGPU detected capabilities */
|
|
webgpu: {
|
|
supported: undefined | boolean,
|
|
backend: undefined | boolean,
|
|
adapter: undefined | GPUAdapterInfo,
|
|
} = {
|
|
supported: undefined,
|
|
backend: undefined,
|
|
adapter: undefined,
|
|
};
|
|
/** CPU info */
|
|
cpu: {
|
|
model: undefined | string,
|
|
flags: string[],
|
|
} = {
|
|
model: undefined,
|
|
flags: [],
|
|
};
|
|
/** List of supported kernels for current backend */
|
|
kernels: string[] = [];
|
|
|
|
/** MonkeyPatch for Canvas/Image/ImageData */
|
|
#canvas: undefined;
|
|
#image: undefined;
|
|
#imageData: undefined;
|
|
|
|
get Canvas() { return this.#canvas; }
|
|
set Canvas(val) { this.#canvas = val; globalThis.Canvas = val; }
|
|
get Image() { return this.#image; }
|
|
// @ts-ignore monkey-patch;
|
|
set Image(val) { this.#image = val; globalThis.Image = val; }
|
|
get ImageData() { return this.#imageData; }
|
|
// @ts-ignore monkey-patch;
|
|
set ImageData(val) { this.#imageData = val; globalThis.ImageData = val; }
|
|
|
|
constructor() {
|
|
this.browser = (typeof navigator !== 'undefined') && (typeof navigator.appVersion !== 'undefined');
|
|
this.node = (typeof process !== 'undefined') && (typeof process.versions !== 'undefined') && (typeof process.versions.node !== 'undefined');
|
|
this.tfjs = { version: tf.version['tfjs-core'] };
|
|
this.offscreen = typeof OffscreenCanvas !== 'undefined';
|
|
this.initial = true;
|
|
|
|
// @ts-ignore WorkerGlobalScope evaluated in browser only
|
|
this.worker = this.browser && this.offscreen ? (typeof WorkerGlobalScope !== 'undefined') : undefined;
|
|
if ((typeof navigator !== 'undefined') && (typeof navigator.userAgent !== 'undefined')) { // TBD replace with navigator.userAgentData once in mainline
|
|
const agent = navigator.userAgent || '';
|
|
const raw = agent.match(/\(([^()]+)\)/g);
|
|
if (raw?.[0]) {
|
|
const platformMatch = raw[0].match(/\(([^()]+)\)/g);
|
|
this.platform = (platformMatch?.[0]) ? platformMatch[0].replace(/\(|\)/g, '') : '';
|
|
this.agent = agent.replace(raw[0], '');
|
|
if (this.platform[1]) this.agent = this.agent.replace(raw[1], '');
|
|
this.agent = this.agent.replace(/ /g, ' ');
|
|
}
|
|
} else if (typeof process !== 'undefined') {
|
|
this.platform = `${process.platform} ${process.arch}`;
|
|
this.agent = `NodeJS ${process.version}`;
|
|
}
|
|
}
|
|
|
|
/** update backend information */
|
|
async updateBackend() {
|
|
// analyze backends
|
|
this.backends = Object.keys(tf.engine().registryFactory);
|
|
try { // backend may not be initialized
|
|
this.tensorflow = {
|
|
version: (tf.backend()['binding'] ? tf.backend()['binding'].TF_Version : undefined),
|
|
gpu: (tf.backend()['binding'] ? tf.backend()['binding'].isUsingGpuDevice() : undefined),
|
|
};
|
|
} catch { /**/ }
|
|
this.wasm.supported = typeof WebAssembly !== 'undefined';
|
|
this.wasm.backend = this.backends.includes('wasm');
|
|
if (this.wasm.supported && this.wasm.backend) {
|
|
this.wasm.simd = await tf.env().getAsync('WASM_HAS_SIMD_SUPPORT') as boolean;
|
|
this.wasm.multithread = await tf.env().getAsync('WASM_HAS_MULTITHREAD_SUPPORT') as boolean;
|
|
}
|
|
const c = image.canvas(100, 100);
|
|
const gl = c ? c.getContext('webgl2') as WebGL2RenderingContext : undefined; // causes too many gl contexts
|
|
this.webgl.supported = typeof gl !== 'undefined';
|
|
this.webgl.backend = this.backends.includes('webgl');
|
|
if (this.webgl.supported && this.webgl.backend && gl) {
|
|
this.webgl.version = gl.getParameter(gl.VERSION);
|
|
this.webgl.vendor = gl.getParameter(gl.VENDOR);
|
|
this.webgl.renderer = gl.getParameter(gl.RENDERER);
|
|
this.webgl.shader = gl.getParameter(gl.SHADING_LANGUAGE_VERSION);
|
|
}
|
|
this.webgpu.supported = this.browser && typeof navigator !== 'undefined' && typeof navigator.gpu !== 'undefined';
|
|
this.webgpu.backend = this.backends.includes('webgpu');
|
|
try {
|
|
if (this.webgpu.supported) {
|
|
const adapter = await navigator.gpu.requestAdapter();
|
|
this.webgpu.adapter = await adapter?.requestAdapterInfo();
|
|
}
|
|
} catch {
|
|
this.webgpu.supported = false;
|
|
}
|
|
try {
|
|
this.kernels = tf.getKernelsForBackend(tf.getBackend()).map((kernel) => kernel.kernelName.toLowerCase());
|
|
} catch { /**/ }
|
|
}
|
|
|
|
/** update cpu information */
|
|
updateCPU() {
|
|
const cpu = { model: '', flags: [] };
|
|
if (this.node && this.platform.startsWith('linux')) {
|
|
/*
|
|
const fs = require('fs');
|
|
try {
|
|
const data = fs.readFileSync('/proc/cpuinfo').toString();
|
|
for (const line of data.split('\n')) {
|
|
if (line.startsWith('model name')) cpu.model = line.match(/:(.*)/g)[0].replace(':', '').trim();
|
|
if (line.startsWith('flags')) cpu.flags = line.match(/:(.*)/g)[0].replace(':', '').trim().split(' ').sort();
|
|
}
|
|
} catch { }
|
|
*/
|
|
}
|
|
if (!this.cpu) Object.defineProperty(this, 'cpu', { value: cpu });
|
|
else this.cpu = cpu;
|
|
}
|
|
}
|
|
|
|
export const env = new Env();
|