UNPKG

@napi-rs/cli

Version:

Cli tools for napi-rs

281 lines (266 loc) 8.51 kB
export function createCjsBinding( localName: string, pkgName: string, idents: string[], packageVersion?: string, ): string { return `${bindingHeader} ${createCommonBinding(localName, pkgName, packageVersion)} module.exports = nativeBinding ${idents .map((ident) => `module.exports.${ident} = nativeBinding.${ident}`) .join('\n')} ` } export function createEsmBinding( localName: string, pkgName: string, idents: string[], packageVersion?: string, ): string { return `${bindingHeader} import { createRequire } from 'node:module' const require = createRequire(import.meta.url) const __dirname = new URL('.', import.meta.url).pathname ${createCommonBinding(localName, pkgName, packageVersion)} const { ${idents.join(', ')} } = nativeBinding ${idents.map((ident) => `export { ${ident} }`).join('\n')} ` } const bindingHeader = `// prettier-ignore /* eslint-disable */ // @ts-nocheck /* auto-generated by NAPI-RS */ ` function createCommonBinding( localName: string, pkgName: string, packageVersion?: string, ): string { function requireTuple(tuple: string, identSize = 8) { const identLow = ' '.repeat(identSize - 2) const ident = ' '.repeat(identSize) const versionCheck = packageVersion ? ` ${identLow}try { ${ident}const binding = require('${pkgName}-${tuple}') ${ident}const bindingPackageVersion = require('${pkgName}-${tuple}/package.json').version ${ident}if (bindingPackageVersion !== '${packageVersion}' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') { ${ident} throw new Error(\`Native binding package version mismatch, expected ${packageVersion} but got \${bindingPackageVersion}. You can reinstall dependencies to fix this issue.\`) ${ident}} ${ident}return binding ${identLow}} catch (e) { ${ident}loadErrors.push(e) ${identLow}}` : ` ${identLow}try { ${ident}return require('${pkgName}-${tuple}') ${identLow}} catch (e) { ${ident}loadErrors.push(e) ${identLow}}` return `try { ${ident}return require('./${localName}.${tuple}.node') ${identLow}} catch (e) { ${ident}loadErrors.push(e) ${identLow}}${versionCheck}` } return `const { readFileSync } = require('node:fs') let nativeBinding = null const loadErrors = [] const isMusl = () => { let musl = false if (process.platform === 'linux') { musl = isMuslFromFilesystem() if (musl === null) { musl = isMuslFromReport() } if (musl === null) { musl = isMuslFromChildProcess() } } return musl } const isFileMusl = (f) => f.includes('libc.musl-') || f.includes('ld-musl-') const isMuslFromFilesystem = () => { try { return readFileSync('/usr/bin/ldd', 'utf-8').includes('musl') } catch { return null } } const isMuslFromReport = () => { let report = null if (typeof process.report?.getReport === 'function') { process.report.excludeNetwork = true report = process.report.getReport() } if (!report) { return null } if (report.header && report.header.glibcVersionRuntime) { return false } if (Array.isArray(report.sharedObjects)) { if (report.sharedObjects.some(isFileMusl)) { return true } } return false } const isMuslFromChildProcess = () => { try { return require('child_process').execSync('ldd --version', { encoding: 'utf8' }).includes('musl') } catch (e) { // If we reach this case, we don't know if the system is musl or not, so is better to just fallback to false return false } } function requireNative() { if (process.env.NAPI_RS_NATIVE_LIBRARY_PATH) { try { return require(process.env.NAPI_RS_NATIVE_LIBRARY_PATH); } catch (err) { loadErrors.push(err) } } else if (process.platform === 'android') { if (process.arch === 'arm64') { ${requireTuple('android-arm64')} } else if (process.arch === 'arm') { ${requireTuple('android-arm-eabi')} } else { loadErrors.push(new Error(\`Unsupported architecture on Android \${process.arch}\`)) } } else if (process.platform === 'win32') { if (process.arch === 'x64') { if (process.config?.variables?.shlib_suffix === 'dll.a' || process.config?.variables?.node_target_type === 'shared_library') { ${requireTuple('win32-x64-gnu')} } else { ${requireTuple('win32-x64-msvc')} } } else if (process.arch === 'ia32') { ${requireTuple('win32-ia32-msvc')} } else if (process.arch === 'arm64') { ${requireTuple('win32-arm64-msvc')} } else { loadErrors.push(new Error(\`Unsupported architecture on Windows: \${process.arch}\`)) } } else if (process.platform === 'darwin') { ${requireTuple('darwin-universal', 6)} if (process.arch === 'x64') { ${requireTuple('darwin-x64')} } else if (process.arch === 'arm64') { ${requireTuple('darwin-arm64')} } else { loadErrors.push(new Error(\`Unsupported architecture on macOS: \${process.arch}\`)) } } else if (process.platform === 'freebsd') { if (process.arch === 'x64') { ${requireTuple('freebsd-x64')} } else if (process.arch === 'arm64') { ${requireTuple('freebsd-arm64')} } else { loadErrors.push(new Error(\`Unsupported architecture on FreeBSD: \${process.arch}\`)) } } else if (process.platform === 'linux') { if (process.arch === 'x64') { if (isMusl()) { ${requireTuple('linux-x64-musl', 10)} } else { ${requireTuple('linux-x64-gnu', 10)} } } else if (process.arch === 'arm64') { if (isMusl()) { ${requireTuple('linux-arm64-musl', 10)} } else { ${requireTuple('linux-arm64-gnu', 10)} } } else if (process.arch === 'arm') { if (isMusl()) { ${requireTuple('linux-arm-musleabihf', 10)} } else { ${requireTuple('linux-arm-gnueabihf', 10)} } } else if (process.arch === 'loong64') { if (isMusl()) { ${requireTuple('linux-loong64-musl', 10)} } else { ${requireTuple('linux-loong64-gnu', 10)} } } else if (process.arch === 'riscv64') { if (isMusl()) { ${requireTuple('linux-riscv64-musl', 10)} } else { ${requireTuple('linux-riscv64-gnu', 10)} } } else if (process.arch === 'ppc64') { ${requireTuple('linux-ppc64-gnu')} } else if (process.arch === 's390x') { ${requireTuple('linux-s390x-gnu')} } else { loadErrors.push(new Error(\`Unsupported architecture on Linux: \${process.arch}\`)) } } else if (process.platform === 'openharmony') { if (process.arch === 'arm64') { ${requireTuple('openharmony-arm64')} } else if (process.arch === 'x64') { ${requireTuple('openharmony-x64')} } else if (process.arch === 'arm') { ${requireTuple('openharmony-arm')} } else { loadErrors.push(new Error(\`Unsupported architecture on OpenHarmony: \${process.arch}\`)) } } else { loadErrors.push(new Error(\`Unsupported OS: \${process.platform}, architecture: \${process.arch}\`)) } } nativeBinding = requireNative() if (!nativeBinding || process.env.NAPI_RS_FORCE_WASI) { let wasiBinding = null let wasiBindingError = null try { wasiBinding = require('./${localName}.wasi.cjs') nativeBinding = wasiBinding } catch (err) { if (process.env.NAPI_RS_FORCE_WASI) { wasiBindingError = err } } if (!nativeBinding || process.env.NAPI_RS_FORCE_WASI) { try { wasiBinding = require('${pkgName}-wasm32-wasi') nativeBinding = wasiBinding } catch (err) { if (process.env.NAPI_RS_FORCE_WASI) { if (!wasiBindingError) { wasiBindingError = err } else { wasiBindingError.cause = err } loadErrors.push(err) } } } if (process.env.NAPI_RS_FORCE_WASI === 'error' && !wasiBinding) { const error = new Error('WASI binding not found and NAPI_RS_FORCE_WASI is set to error') error.cause = wasiBindingError throw error } } if (!nativeBinding) { if (loadErrors.length > 0) { throw new Error( \`Cannot find native binding. \` + \`npm has a bug related to optional dependencies (https://github.com/npm/cli/issues/4828). \` + 'Please try \`npm i\` again after removing both package-lock.json and node_modules directory.', { cause: loadErrors.reduce((err, cur) => { cur.cause = err return cur }), }, ) } throw new Error(\`Failed to load native binding\`) } ` }