node-tpm2 0.0.1 → 0.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -3,11 +3,46 @@
3
3
  Native TPM 2.0 for Node. Zero tooling, no admin.
4
4
 
5
5
  - Windows via TBS, Linux via `/dev/tpmrm0`.
6
- - Ships as prebuilt native binaries: `npm install node-tpm2` pulls one binary, no build step, no tpm2-tools, no PATH edits.
7
- - Subsystem API: `tpm.pcr`, `tpm.keys`, `tpm.random`, `tpm.nv`, `tpm.attest`.
6
+ - Direct TBS command marshalling no tpm2-tss, no tpm2-tools at install or runtime.
7
+ - Ships as prebuilt native binaries via napi-rs platform packages.
8
8
 
9
- > **Status: pre-release placeholder.** The native backend is not published yet.
10
- > `Tpm.open()` throws `NOT_IMPLEMENTED` until the first working release. Watch this repo.
9
+ > **Status: pre-release.** `Tpm.isAvailable()` and `Tpm.info()` work on Windows and Linux.
10
+ > `Tpm.open()` and attestation methods are not implemented yet.
11
+
12
+ ## Install
13
+
14
+ ```bash
15
+ npm install node-tpm2
16
+ ```
17
+
18
+ npm resolves exactly one prebuilt native binary from `optionalDependencies` — no build step,
19
+ no tpm2-tools, no Rust. Requires platform packages published for your OS/arch.
20
+
21
+ ## Development
22
+
23
+ ```bash
24
+ git clone https://github.com/stacks0x/tpm2.git
25
+ cd tpm2
26
+ npm install
27
+ npm run build
28
+ node --input-type=module -e "
29
+ import { Tpm } from './index.js';
30
+ console.log('available', await Tpm.isAvailable());
31
+ console.log('info', await Tpm.info());
32
+ "
33
+ ```
34
+
35
+ On Linux, your user needs read/write on `/dev/tpmrm0` (typically the `tss` group).
36
+
37
+ ## Windows probe (direct TBS validation)
38
+
39
+ Non-elevated PowerShell on Windows 11:
40
+
41
+ ```powershell
42
+ cargo run --bin tbs-probe -- all
43
+ ```
44
+
45
+ See [spike/README.md](./spike/README.md) for probe details and RC discipline.
11
46
 
12
47
  ## License
13
48
 
package/api.js ADDED
@@ -0,0 +1,69 @@
1
+ let native = null;
2
+ let nativeLoadError = null;
3
+
4
+ try {
5
+ const { createRequire } = await import('node:module');
6
+ const require = createRequire(import.meta.url);
7
+ native = require('./native.cjs');
8
+ } catch (err) {
9
+ nativeLoadError = err;
10
+ }
11
+
12
+ export class TpmError extends Error {
13
+ constructor(code, message, suggestion, tpmRc) {
14
+ super(message);
15
+ this.name = 'TpmError';
16
+ this.code = code;
17
+ this.suggestion = suggestion;
18
+ this.tpmRc = tpmRc;
19
+ }
20
+ }
21
+
22
+ export const Tpm = {
23
+ /** Probe for an accessible TPM. False on darwin / no TPM / no access. */
24
+ async isAvailable() {
25
+ if (!native?.isAvailable) {
26
+ return false;
27
+ }
28
+ try {
29
+ return await native.isAvailable();
30
+ } catch {
31
+ return false;
32
+ }
33
+ },
34
+
35
+ /** Open a TPM handle. Not implemented in v0.0.x. */
36
+ async open() {
37
+ if (!native?.isAvailable) {
38
+ throw new TpmError(
39
+ 'TPM_UNAVAILABLE',
40
+ nativeLoadError
41
+ ? `Native backend not loaded: ${nativeLoadError.message}`
42
+ : 'Native backend not built for this platform.',
43
+ 'Run npm run build, or install a published platform package.',
44
+ );
45
+ }
46
+ throw new TpmError(
47
+ 'NOT_IMPLEMENTED',
48
+ 'Tpm.open() is not implemented yet; v0.0.x exposes isAvailable() and info() only.',
49
+ 'See https://github.com/stacks0x/tpm2 for release progress.',
50
+ );
51
+ },
52
+
53
+ /** Manufacturer, firmware, and virtual-TPM hint from GetCapability. */
54
+ async getFixedProperties() {
55
+ if (!native?.getFixedProperties) {
56
+ throw new TpmError(
57
+ 'TPM_UNAVAILABLE',
58
+ 'Native backend not loaded.',
59
+ 'Run npm run build, or install a published platform package.',
60
+ );
61
+ }
62
+ return native.getFixedProperties();
63
+ },
64
+
65
+ /** Alias for getFixedProperties. */
66
+ async info() {
67
+ return this.getFixedProperties();
68
+ },
69
+ };
package/index.d.ts CHANGED
@@ -1,9 +1,23 @@
1
1
  export declare class TpmError extends Error {
2
2
  code: string;
3
3
  suggestion?: string;
4
+ tpmRc?: number;
5
+ constructor(code: string, message: string, suggestion?: string, tpmRc?: number);
4
6
  }
5
7
 
6
8
  export declare const Tpm: {
7
9
  isAvailable(): Promise<boolean>;
8
10
  open(): Promise<never>;
11
+ getFixedProperties(): Promise<{
12
+ manufacturer: string;
13
+ firmwareVersion: string;
14
+ isVirtual: boolean;
15
+ spec: string;
16
+ }>;
17
+ info(): Promise<{
18
+ manufacturer: string;
19
+ firmwareVersion: string;
20
+ isVirtual: boolean;
21
+ spec: string;
22
+ }>;
9
23
  };
package/index.js CHANGED
@@ -1,23 +1 @@
1
- export class TpmError extends Error {
2
- constructor(code, message, suggestion) {
3
- super(message);
4
- this.name = 'TpmError';
5
- this.code = code;
6
- this.suggestion = suggestion;
7
- }
8
- }
9
-
10
- export const Tpm = {
11
- /** Probe for an accessible TPM. Always false in this pre-release placeholder. */
12
- async isAvailable() {
13
- return false;
14
- },
15
- /** Open a TPM handle. Not implemented yet in this pre-release. */
16
- async open() {
17
- throw new TpmError(
18
- 'NOT_IMPLEMENTED',
19
- 'node-tpm2 is a pre-release placeholder; the native backend is not published yet.',
20
- 'Follow https://github.com/stacks0x/tpm2 for the first working release.',
21
- );
22
- },
23
- };
1
+ export { TpmError, Tpm } from './api.js';
package/native.cjs ADDED
@@ -0,0 +1,591 @@
1
+ // prettier-ignore
2
+ /* eslint-disable */
3
+ // @ts-nocheck
4
+ /* auto-generated by NAPI-RS */
5
+
6
+ const { readFileSync } = require('fs')
7
+ let nativeBinding = null
8
+ const loadErrors = []
9
+
10
+ const isMusl = () => {
11
+ let musl = false
12
+ if (process.platform === 'linux') {
13
+ musl = isMuslFromFilesystem()
14
+ if (musl === null) {
15
+ musl = isMuslFromReport()
16
+ }
17
+ if (musl === null) {
18
+ musl = isMuslFromChildProcess()
19
+ }
20
+ }
21
+ return musl
22
+ }
23
+
24
+ const isFileMusl = (f) => f.includes('libc.musl-') || f.includes('ld-musl-')
25
+
26
+ const isMuslFromFilesystem = () => {
27
+ try {
28
+ return readFileSync('/usr/bin/ldd', 'utf-8').includes('musl')
29
+ } catch {
30
+ return null
31
+ }
32
+ }
33
+
34
+ const isMuslFromReport = () => {
35
+ let report = null
36
+ if (process.report && typeof process.report.getReport === 'function') {
37
+ process.report.excludeNetwork = true
38
+ report = process.report.getReport()
39
+ }
40
+ if (!report) {
41
+ return null
42
+ }
43
+ if (report.header && report.header.glibcVersionRuntime) {
44
+ return false
45
+ }
46
+ if (Array.isArray(report.sharedObjects)) {
47
+ if (report.sharedObjects.some(isFileMusl)) {
48
+ return true
49
+ }
50
+ }
51
+ return false
52
+ }
53
+
54
+ const isMuslFromChildProcess = () => {
55
+ try {
56
+ return require('child_process').execSync('ldd --version', { encoding: 'utf8' }).includes('musl')
57
+ } catch (e) {
58
+ // If we reach this case, we don't know if the system is musl or not, so is better to just fallback to false
59
+ return false
60
+ }
61
+ }
62
+
63
+ function requireNative() {
64
+ if (process.env.NAPI_RS_NATIVE_LIBRARY_PATH) {
65
+ try {
66
+ return require(process.env.NAPI_RS_NATIVE_LIBRARY_PATH);
67
+ } catch (err) {
68
+ loadErrors.push(err)
69
+ }
70
+ } else if (process.platform === 'android') {
71
+ if (process.arch === 'arm64') {
72
+ try {
73
+ return require('./node-tpm2.android-arm64.node')
74
+ } catch (e) {
75
+ loadErrors.push(e)
76
+ }
77
+ try {
78
+ const binding = require('node-tpm2-android-arm64')
79
+ const bindingPackageVersion = require('node-tpm2-android-arm64/package.json').version
80
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
81
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
82
+ }
83
+ return binding
84
+ } catch (e) {
85
+ loadErrors.push(e)
86
+ }
87
+ } else if (process.arch === 'arm') {
88
+ try {
89
+ return require('./node-tpm2.android-arm-eabi.node')
90
+ } catch (e) {
91
+ loadErrors.push(e)
92
+ }
93
+ try {
94
+ const binding = require('node-tpm2-android-arm-eabi')
95
+ const bindingPackageVersion = require('node-tpm2-android-arm-eabi/package.json').version
96
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
97
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
98
+ }
99
+ return binding
100
+ } catch (e) {
101
+ loadErrors.push(e)
102
+ }
103
+ } else {
104
+ loadErrors.push(new Error(`Unsupported architecture on Android ${process.arch}`))
105
+ }
106
+ } else if (process.platform === 'win32') {
107
+ if (process.arch === 'x64') {
108
+ if ((process.config && process.config.variables && process.config.variables.shlib_suffix === 'dll.a') || (process.config && process.config.variables && process.config.variables.node_target_type === 'shared_library')) {
109
+ try {
110
+ return require('./node-tpm2.win32-x64-gnu.node')
111
+ } catch (e) {
112
+ loadErrors.push(e)
113
+ }
114
+ try {
115
+ const binding = require('node-tpm2-win32-x64-gnu')
116
+ const bindingPackageVersion = require('node-tpm2-win32-x64-gnu/package.json').version
117
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
118
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
119
+ }
120
+ return binding
121
+ } catch (e) {
122
+ loadErrors.push(e)
123
+ }
124
+ } else {
125
+ try {
126
+ return require('./node-tpm2.win32-x64-msvc.node')
127
+ } catch (e) {
128
+ loadErrors.push(e)
129
+ }
130
+ try {
131
+ const binding = require('node-tpm2-windows-x64-msvc')
132
+ const bindingPackageVersion = require('node-tpm2-windows-x64-msvc/package.json').version
133
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
134
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
135
+ }
136
+ return binding
137
+ } catch (e) {
138
+ loadErrors.push(e)
139
+ }
140
+ }
141
+ } else if (process.arch === 'ia32') {
142
+ try {
143
+ return require('./node-tpm2.win32-ia32-msvc.node')
144
+ } catch (e) {
145
+ loadErrors.push(e)
146
+ }
147
+ try {
148
+ const binding = require('node-tpm2-win32-ia32-msvc')
149
+ const bindingPackageVersion = require('node-tpm2-win32-ia32-msvc/package.json').version
150
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
151
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
152
+ }
153
+ return binding
154
+ } catch (e) {
155
+ loadErrors.push(e)
156
+ }
157
+ } else if (process.arch === 'arm64') {
158
+ try {
159
+ return require('./node-tpm2.win32-arm64-msvc.node')
160
+ } catch (e) {
161
+ loadErrors.push(e)
162
+ }
163
+ try {
164
+ const binding = require('node-tpm2-windows-arm64-msvc')
165
+ const bindingPackageVersion = require('node-tpm2-windows-arm64-msvc/package.json').version
166
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
167
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
168
+ }
169
+ return binding
170
+ } catch (e) {
171
+ loadErrors.push(e)
172
+ }
173
+ } else {
174
+ loadErrors.push(new Error(`Unsupported architecture on Windows: ${process.arch}`))
175
+ }
176
+ } else if (process.platform === 'darwin') {
177
+ try {
178
+ return require('./node-tpm2.darwin-universal.node')
179
+ } catch (e) {
180
+ loadErrors.push(e)
181
+ }
182
+ try {
183
+ const binding = require('node-tpm2-darwin-universal')
184
+ const bindingPackageVersion = require('node-tpm2-darwin-universal/package.json').version
185
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
186
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
187
+ }
188
+ return binding
189
+ } catch (e) {
190
+ loadErrors.push(e)
191
+ }
192
+ if (process.arch === 'x64') {
193
+ try {
194
+ return require('./node-tpm2.darwin-x64.node')
195
+ } catch (e) {
196
+ loadErrors.push(e)
197
+ }
198
+ try {
199
+ const binding = require('node-tpm2-darwin-x64')
200
+ const bindingPackageVersion = require('node-tpm2-darwin-x64/package.json').version
201
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
202
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
203
+ }
204
+ return binding
205
+ } catch (e) {
206
+ loadErrors.push(e)
207
+ }
208
+ } else if (process.arch === 'arm64') {
209
+ try {
210
+ return require('./node-tpm2.darwin-arm64.node')
211
+ } catch (e) {
212
+ loadErrors.push(e)
213
+ }
214
+ try {
215
+ const binding = require('node-tpm2-darwin-arm64')
216
+ const bindingPackageVersion = require('node-tpm2-darwin-arm64/package.json').version
217
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
218
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
219
+ }
220
+ return binding
221
+ } catch (e) {
222
+ loadErrors.push(e)
223
+ }
224
+ } else {
225
+ loadErrors.push(new Error(`Unsupported architecture on macOS: ${process.arch}`))
226
+ }
227
+ } else if (process.platform === 'freebsd') {
228
+ if (process.arch === 'x64') {
229
+ try {
230
+ return require('./node-tpm2.freebsd-x64.node')
231
+ } catch (e) {
232
+ loadErrors.push(e)
233
+ }
234
+ try {
235
+ const binding = require('node-tpm2-freebsd-x64')
236
+ const bindingPackageVersion = require('node-tpm2-freebsd-x64/package.json').version
237
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
238
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
239
+ }
240
+ return binding
241
+ } catch (e) {
242
+ loadErrors.push(e)
243
+ }
244
+ } else if (process.arch === 'arm64') {
245
+ try {
246
+ return require('./node-tpm2.freebsd-arm64.node')
247
+ } catch (e) {
248
+ loadErrors.push(e)
249
+ }
250
+ try {
251
+ const binding = require('node-tpm2-freebsd-arm64')
252
+ const bindingPackageVersion = require('node-tpm2-freebsd-arm64/package.json').version
253
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
254
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
255
+ }
256
+ return binding
257
+ } catch (e) {
258
+ loadErrors.push(e)
259
+ }
260
+ } else {
261
+ loadErrors.push(new Error(`Unsupported architecture on FreeBSD: ${process.arch}`))
262
+ }
263
+ } else if (process.platform === 'linux') {
264
+ if (process.arch === 'x64') {
265
+ if (isMusl()) {
266
+ try {
267
+ return require('./node-tpm2.linux-x64-musl.node')
268
+ } catch (e) {
269
+ loadErrors.push(e)
270
+ }
271
+ try {
272
+ const binding = require('node-tpm2-linux-x64-musl')
273
+ const bindingPackageVersion = require('node-tpm2-linux-x64-musl/package.json').version
274
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
275
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
276
+ }
277
+ return binding
278
+ } catch (e) {
279
+ loadErrors.push(e)
280
+ }
281
+ } else {
282
+ try {
283
+ return require('./node-tpm2.linux-x64-gnu.node')
284
+ } catch (e) {
285
+ loadErrors.push(e)
286
+ }
287
+ try {
288
+ const binding = require('node-tpm2-linux-x64-gnu')
289
+ const bindingPackageVersion = require('node-tpm2-linux-x64-gnu/package.json').version
290
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
291
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
292
+ }
293
+ return binding
294
+ } catch (e) {
295
+ loadErrors.push(e)
296
+ }
297
+ }
298
+ } else if (process.arch === 'arm64') {
299
+ if (isMusl()) {
300
+ try {
301
+ return require('./node-tpm2.linux-arm64-musl.node')
302
+ } catch (e) {
303
+ loadErrors.push(e)
304
+ }
305
+ try {
306
+ const binding = require('node-tpm2-linux-arm64-musl')
307
+ const bindingPackageVersion = require('node-tpm2-linux-arm64-musl/package.json').version
308
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
309
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
310
+ }
311
+ return binding
312
+ } catch (e) {
313
+ loadErrors.push(e)
314
+ }
315
+ } else {
316
+ try {
317
+ return require('./node-tpm2.linux-arm64-gnu.node')
318
+ } catch (e) {
319
+ loadErrors.push(e)
320
+ }
321
+ try {
322
+ const binding = require('node-tpm2-linux-arm64-gnu')
323
+ const bindingPackageVersion = require('node-tpm2-linux-arm64-gnu/package.json').version
324
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
325
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
326
+ }
327
+ return binding
328
+ } catch (e) {
329
+ loadErrors.push(e)
330
+ }
331
+ }
332
+ } else if (process.arch === 'arm') {
333
+ if (isMusl()) {
334
+ try {
335
+ return require('./node-tpm2.linux-arm-musleabihf.node')
336
+ } catch (e) {
337
+ loadErrors.push(e)
338
+ }
339
+ try {
340
+ const binding = require('node-tpm2-linux-arm-musleabihf')
341
+ const bindingPackageVersion = require('node-tpm2-linux-arm-musleabihf/package.json').version
342
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
343
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
344
+ }
345
+ return binding
346
+ } catch (e) {
347
+ loadErrors.push(e)
348
+ }
349
+ } else {
350
+ try {
351
+ return require('./node-tpm2.linux-arm-gnueabihf.node')
352
+ } catch (e) {
353
+ loadErrors.push(e)
354
+ }
355
+ try {
356
+ const binding = require('node-tpm2-linux-arm-gnueabihf')
357
+ const bindingPackageVersion = require('node-tpm2-linux-arm-gnueabihf/package.json').version
358
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
359
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
360
+ }
361
+ return binding
362
+ } catch (e) {
363
+ loadErrors.push(e)
364
+ }
365
+ }
366
+ } else if (process.arch === 'loong64') {
367
+ if (isMusl()) {
368
+ try {
369
+ return require('./node-tpm2.linux-loong64-musl.node')
370
+ } catch (e) {
371
+ loadErrors.push(e)
372
+ }
373
+ try {
374
+ const binding = require('node-tpm2-linux-loong64-musl')
375
+ const bindingPackageVersion = require('node-tpm2-linux-loong64-musl/package.json').version
376
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
377
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
378
+ }
379
+ return binding
380
+ } catch (e) {
381
+ loadErrors.push(e)
382
+ }
383
+ } else {
384
+ try {
385
+ return require('./node-tpm2.linux-loong64-gnu.node')
386
+ } catch (e) {
387
+ loadErrors.push(e)
388
+ }
389
+ try {
390
+ const binding = require('node-tpm2-linux-loong64-gnu')
391
+ const bindingPackageVersion = require('node-tpm2-linux-loong64-gnu/package.json').version
392
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
393
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
394
+ }
395
+ return binding
396
+ } catch (e) {
397
+ loadErrors.push(e)
398
+ }
399
+ }
400
+ } else if (process.arch === 'riscv64') {
401
+ if (isMusl()) {
402
+ try {
403
+ return require('./node-tpm2.linux-riscv64-musl.node')
404
+ } catch (e) {
405
+ loadErrors.push(e)
406
+ }
407
+ try {
408
+ const binding = require('node-tpm2-linux-riscv64-musl')
409
+ const bindingPackageVersion = require('node-tpm2-linux-riscv64-musl/package.json').version
410
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
411
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
412
+ }
413
+ return binding
414
+ } catch (e) {
415
+ loadErrors.push(e)
416
+ }
417
+ } else {
418
+ try {
419
+ return require('./node-tpm2.linux-riscv64-gnu.node')
420
+ } catch (e) {
421
+ loadErrors.push(e)
422
+ }
423
+ try {
424
+ const binding = require('node-tpm2-linux-riscv64-gnu')
425
+ const bindingPackageVersion = require('node-tpm2-linux-riscv64-gnu/package.json').version
426
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
427
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
428
+ }
429
+ return binding
430
+ } catch (e) {
431
+ loadErrors.push(e)
432
+ }
433
+ }
434
+ } else if (process.arch === 'ppc64') {
435
+ try {
436
+ return require('./node-tpm2.linux-ppc64-gnu.node')
437
+ } catch (e) {
438
+ loadErrors.push(e)
439
+ }
440
+ try {
441
+ const binding = require('node-tpm2-linux-ppc64-gnu')
442
+ const bindingPackageVersion = require('node-tpm2-linux-ppc64-gnu/package.json').version
443
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
444
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
445
+ }
446
+ return binding
447
+ } catch (e) {
448
+ loadErrors.push(e)
449
+ }
450
+ } else if (process.arch === 's390x') {
451
+ try {
452
+ return require('./node-tpm2.linux-s390x-gnu.node')
453
+ } catch (e) {
454
+ loadErrors.push(e)
455
+ }
456
+ try {
457
+ const binding = require('node-tpm2-linux-s390x-gnu')
458
+ const bindingPackageVersion = require('node-tpm2-linux-s390x-gnu/package.json').version
459
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
460
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
461
+ }
462
+ return binding
463
+ } catch (e) {
464
+ loadErrors.push(e)
465
+ }
466
+ } else {
467
+ loadErrors.push(new Error(`Unsupported architecture on Linux: ${process.arch}`))
468
+ }
469
+ } else if (process.platform === 'openharmony') {
470
+ if (process.arch === 'arm64') {
471
+ try {
472
+ return require('./node-tpm2.openharmony-arm64.node')
473
+ } catch (e) {
474
+ loadErrors.push(e)
475
+ }
476
+ try {
477
+ const binding = require('node-tpm2-openharmony-arm64')
478
+ const bindingPackageVersion = require('node-tpm2-openharmony-arm64/package.json').version
479
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
480
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
481
+ }
482
+ return binding
483
+ } catch (e) {
484
+ loadErrors.push(e)
485
+ }
486
+ } else if (process.arch === 'x64') {
487
+ try {
488
+ return require('./node-tpm2.openharmony-x64.node')
489
+ } catch (e) {
490
+ loadErrors.push(e)
491
+ }
492
+ try {
493
+ const binding = require('node-tpm2-openharmony-x64')
494
+ const bindingPackageVersion = require('node-tpm2-openharmony-x64/package.json').version
495
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
496
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
497
+ }
498
+ return binding
499
+ } catch (e) {
500
+ loadErrors.push(e)
501
+ }
502
+ } else if (process.arch === 'arm') {
503
+ try {
504
+ return require('./node-tpm2.openharmony-arm.node')
505
+ } catch (e) {
506
+ loadErrors.push(e)
507
+ }
508
+ try {
509
+ const binding = require('node-tpm2-openharmony-arm')
510
+ const bindingPackageVersion = require('node-tpm2-openharmony-arm/package.json').version
511
+ if (bindingPackageVersion !== '0.0.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
512
+ throw new Error(`Native binding package version mismatch, expected 0.0.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
513
+ }
514
+ return binding
515
+ } catch (e) {
516
+ loadErrors.push(e)
517
+ }
518
+ } else {
519
+ loadErrors.push(new Error(`Unsupported architecture on OpenHarmony: ${process.arch}`))
520
+ }
521
+ } else {
522
+ loadErrors.push(new Error(`Unsupported OS: ${process.platform}, architecture: ${process.arch}`))
523
+ }
524
+ }
525
+
526
+ nativeBinding = requireNative()
527
+
528
+ // NAPI_RS_FORCE_WASI is a tri-state flag:
529
+ // unset / any other value → native binding preferred, WASI is only a fallback
530
+ // 'true' → force WASI fallback even if native loaded
531
+ // 'error' → force WASI and throw if no WASI binding is found
532
+ // Treating any non-empty string as truthy (the historical behavior) meant
533
+ // NAPI_RS_FORCE_WASI=false, NAPI_RS_FORCE_WASI=0, etc. inadvertently triggered
534
+ // the WASI path, causing ENOENT for packages shipped without a .wasi.cjs file.
535
+ const forceWasi =
536
+ process.env.NAPI_RS_FORCE_WASI === 'true' || process.env.NAPI_RS_FORCE_WASI === 'error'
537
+
538
+ if (!nativeBinding || forceWasi) {
539
+ let wasiBinding = null
540
+ let wasiBindingError = null
541
+ try {
542
+ wasiBinding = require('./node-tpm2.wasi.cjs')
543
+ nativeBinding = wasiBinding
544
+ } catch (err) {
545
+ if (forceWasi) {
546
+ wasiBindingError = err
547
+ }
548
+ }
549
+ if (!nativeBinding || forceWasi) {
550
+ try {
551
+ wasiBinding = require('node-tpm2-wasm32-wasi')
552
+ nativeBinding = wasiBinding
553
+ } catch (err) {
554
+ if (forceWasi) {
555
+ if (!wasiBindingError) {
556
+ wasiBindingError = err
557
+ } else {
558
+ wasiBindingError.cause = err
559
+ }
560
+ loadErrors.push(err)
561
+ }
562
+ }
563
+ }
564
+ if (process.env.NAPI_RS_FORCE_WASI === 'error' && !wasiBinding) {
565
+ const error = new Error('WASI binding not found and NAPI_RS_FORCE_WASI is set to error')
566
+ error.cause = wasiBindingError
567
+ throw error
568
+ }
569
+ }
570
+
571
+ if (!nativeBinding) {
572
+ if (loadErrors.length > 0) {
573
+ const error = new Error(
574
+ `Cannot find native binding. ` +
575
+ `npm has a bug related to optional dependencies (https://github.com/npm/cli/issues/4828). ` +
576
+ 'Please try `npm i` again after removing both package-lock.json and node_modules directory.',
577
+ )
578
+ // assign instead of the `new Error(message, { cause })` options form,
579
+ // which Node < 16.9 silently ignores
580
+ error.cause = loadErrors.reduce((err, cur) => {
581
+ cur.cause = err
582
+ return cur
583
+ })
584
+ throw error
585
+ }
586
+ throw new Error(`Failed to load native binding`)
587
+ }
588
+
589
+ module.exports = nativeBinding
590
+ module.exports.getFixedProperties = nativeBinding.getFixedProperties
591
+ module.exports.isAvailable = nativeBinding.isAvailable
package/native.d.ts ADDED
@@ -0,0 +1,12 @@
1
+ /* auto-generated by NAPI-RS */
2
+ /* eslint-disable */
3
+ export interface FixedPropertiesJs {
4
+ manufacturer: string
5
+ firmwareVersion: string
6
+ isVirtual: boolean
7
+ spec: string
8
+ }
9
+
10
+ export declare function getFixedProperties(): Promise<FixedPropertiesJs>
11
+
12
+ export declare function isAvailable(): Promise<boolean>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "node-tpm2",
3
- "version": "0.0.1",
3
+ "version": "0.0.3",
4
4
  "description": "Native TPM 2.0 for Node. Zero tooling, no admin. Windows (TBS) and Linux (/dev/tpmrm0). Pre-release.",
5
5
  "type": "module",
6
6
  "license": "Apache-2.0",
@@ -21,14 +21,31 @@
21
21
  "types": "./index.d.ts"
22
22
  }
23
23
  },
24
- "files": ["index.js", "index.d.ts"],
24
+ "napi": {
25
+ "binaryName": "node-tpm2",
26
+ "targets": [
27
+ "x86_64-pc-windows-msvc",
28
+ "aarch64-pc-windows-msvc",
29
+ "x86_64-unknown-linux-gnu",
30
+ "aarch64-unknown-linux-gnu",
31
+ "x86_64-unknown-linux-musl",
32
+ "aarch64-unknown-linux-musl",
33
+ "aarch64-apple-darwin"
34
+ ]
35
+ },
36
+ "files": [
37
+ "index.js",
38
+ "index.d.ts",
39
+ "api.js",
40
+ "native.cjs",
41
+ "native.d.ts"
42
+ ],
25
43
  "engines": {
26
44
  "node": ">=20"
27
45
  },
28
46
  "keywords": [
29
47
  "tpm",
30
48
  "tpm2",
31
- "node-tpm2",
32
49
  "attestation",
33
50
  "tbs",
34
51
  "trusted-platform-module",
@@ -37,5 +54,30 @@
37
54
  ],
38
55
  "publishConfig": {
39
56
  "access": "public"
57
+ },
58
+ "scripts": {
59
+ "build": "napi build --platform --release --js native.cjs --dts native.d.ts",
60
+ "build:debug": "napi build --platform --js native.cjs --dts native.d.ts",
61
+ "build:esapi": "napi build --platform --release --features esapi --js native.cjs --dts native.d.ts",
62
+ "tbs-probe": "cargo run --bin tbs-probe",
63
+ "spike": "cargo run --features esapi --bin spike --",
64
+ "spike:all": "cargo run --features esapi --bin spike -- all",
65
+ "artifacts": "napi artifacts",
66
+ "patch-windows-npm": "node scripts/patch-windows-npm-packages.mjs",
67
+ "prepublishOnly": "napi prepublish -t npm --skip-optional-publish"
68
+ },
69
+ "devDependencies": {
70
+ "@napi-rs/cli": "^3.7.2"
71
+ },
72
+ "optionalDependencies": {
73
+ "node-tpm2-windows-x64-msvc": "0.0.3",
74
+ "node-tpm2-windows-arm64-msvc": "0.0.3",
75
+ "node-tpm2-linux-x64-gnu": "0.0.3",
76
+ "node-tpm2-linux-arm64-gnu": "0.0.3",
77
+ "node-tpm2-linux-x64-musl": "0.0.3",
78
+ "node-tpm2-linux-arm64-musl": "0.0.3",
79
+ "node-tpm2-darwin-arm64": "0.0.3",
80
+ "node-tpm2-win32-x64-msvc": "0.0.3",
81
+ "node-tpm2-win32-arm64-msvc": "0.0.3"
40
82
  }
41
- }
83
+ }