@labacacia/nps-sdk 1.0.0-alpha.1 → 1.0.0-alpha.13
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/CHANGELOG.cn.md +181 -0
- package/CHANGELOG.md +234 -0
- package/LICENSE +0 -0
- package/NOTICE +0 -0
- package/README.cn.md +163 -0
- package/README.md +16 -6
- package/dist/core/anchor-cache.d.ts +0 -0
- package/dist/core/anchor-cache.d.ts.map +0 -0
- package/dist/core/anchor-cache.js +0 -0
- package/dist/core/anchor-cache.js.map +0 -0
- package/dist/core/cache.d.ts +0 -0
- package/dist/core/cache.d.ts.map +0 -0
- package/dist/core/cache.js +0 -0
- package/dist/core/cache.js.map +0 -0
- package/dist/core/canonical-json.d.ts +0 -0
- package/dist/core/canonical-json.d.ts.map +0 -0
- package/dist/core/canonical-json.js +0 -0
- package/dist/core/canonical-json.js.map +0 -0
- package/dist/core/codec.d.ts +0 -0
- package/dist/core/codec.d.ts.map +0 -0
- package/dist/core/codec.js +0 -0
- package/dist/core/codec.js.map +0 -0
- package/dist/core/codecs/index.d.ts +0 -0
- package/dist/core/codecs/index.d.ts.map +0 -0
- package/dist/core/codecs/index.js +0 -0
- package/dist/core/codecs/index.js.map +0 -0
- package/dist/core/codecs/ncp-codec.d.ts +0 -0
- package/dist/core/codecs/ncp-codec.d.ts.map +0 -0
- package/dist/core/codecs/ncp-codec.js +0 -0
- package/dist/core/codecs/ncp-codec.js.map +0 -0
- package/dist/core/codecs/tier1-json-codec.d.ts +0 -0
- package/dist/core/codecs/tier1-json-codec.d.ts.map +0 -0
- package/dist/core/codecs/tier1-json-codec.js +0 -0
- package/dist/core/codecs/tier1-json-codec.js.map +0 -0
- package/dist/core/codecs/tier2-msgpack-codec.d.ts +0 -0
- package/dist/core/codecs/tier2-msgpack-codec.d.ts.map +0 -0
- package/dist/core/codecs/tier2-msgpack-codec.js +0 -0
- package/dist/core/codecs/tier2-msgpack-codec.js.map +0 -0
- package/dist/core/crypto-provider.d.ts +0 -0
- package/dist/core/crypto-provider.d.ts.map +0 -0
- package/dist/core/crypto-provider.js +0 -0
- package/dist/core/crypto-provider.js.map +0 -0
- package/dist/core/exceptions.d.ts +0 -0
- package/dist/core/exceptions.d.ts.map +0 -0
- package/dist/core/exceptions.js +0 -0
- package/dist/core/exceptions.js.map +0 -0
- package/dist/core/frame-header.d.ts +1 -0
- package/dist/core/frame-header.d.ts.map +1 -1
- package/dist/core/frame-header.js +1 -0
- package/dist/core/frame-header.js.map +1 -1
- package/dist/core/frame-registry.d.ts +0 -0
- package/dist/core/frame-registry.d.ts.map +1 -1
- package/dist/core/frame-registry.js +1 -0
- package/dist/core/frame-registry.js.map +1 -1
- package/dist/core/frames.d.ts +3 -0
- package/dist/core/frames.d.ts.map +1 -1
- package/dist/core/frames.js +3 -0
- package/dist/core/frames.js.map +1 -1
- package/dist/core/index.d.ts +6 -4
- package/dist/core/index.d.ts.map +1 -1
- package/dist/core/index.js +17 -5
- package/dist/core/index.js.map +1 -1
- package/dist/core/registry.d.ts +0 -0
- package/dist/core/registry.d.ts.map +0 -0
- package/dist/core/registry.js +0 -0
- package/dist/core/registry.js.map +0 -0
- package/dist/core/status-codes.d.ts +20 -0
- package/dist/core/status-codes.d.ts.map +1 -1
- package/dist/core/status-codes.js +51 -0
- package/dist/core/status-codes.js.map +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/ncp/frames/anchor-frame.d.ts +0 -0
- package/dist/ncp/frames/anchor-frame.d.ts.map +0 -0
- package/dist/ncp/frames/anchor-frame.js +0 -0
- package/dist/ncp/frames/anchor-frame.js.map +0 -0
- package/dist/ncp/frames/caps-frame.d.ts +0 -0
- package/dist/ncp/frames/caps-frame.d.ts.map +0 -0
- package/dist/ncp/frames/caps-frame.js +0 -0
- package/dist/ncp/frames/caps-frame.js.map +0 -0
- package/dist/ncp/frames/diff-frame.d.ts +0 -0
- package/dist/ncp/frames/diff-frame.d.ts.map +0 -0
- package/dist/ncp/frames/diff-frame.js +0 -0
- package/dist/ncp/frames/diff-frame.js.map +0 -0
- package/dist/ncp/frames/error-frame.d.ts +0 -0
- package/dist/ncp/frames/error-frame.d.ts.map +0 -0
- package/dist/ncp/frames/error-frame.js +0 -0
- package/dist/ncp/frames/error-frame.js.map +0 -0
- package/dist/ncp/frames/hello-frame.d.ts +0 -0
- package/dist/ncp/frames/hello-frame.d.ts.map +0 -0
- package/dist/ncp/frames/hello-frame.js +0 -0
- package/dist/ncp/frames/hello-frame.js.map +0 -0
- package/dist/ncp/frames/stream-frame.d.ts +0 -0
- package/dist/ncp/frames/stream-frame.d.ts.map +0 -0
- package/dist/ncp/frames/stream-frame.js +0 -0
- package/dist/ncp/frames/stream-frame.js.map +0 -0
- package/dist/ncp/frames.d.ts +25 -0
- package/dist/ncp/frames.d.ts.map +1 -1
- package/dist/ncp/frames.js +61 -0
- package/dist/ncp/frames.js.map +1 -1
- package/dist/ncp/handshake.d.ts +0 -0
- package/dist/ncp/handshake.d.ts.map +0 -0
- package/dist/ncp/handshake.js +0 -0
- package/dist/ncp/handshake.js.map +0 -0
- package/dist/ncp/index.d.ts +1 -0
- package/dist/ncp/index.d.ts.map +1 -1
- package/dist/ncp/index.js +1 -0
- package/dist/ncp/index.js.map +1 -1
- package/dist/ncp/ncp-error-codes.d.ts +5 -0
- package/dist/ncp/ncp-error-codes.d.ts.map +1 -1
- package/dist/ncp/ncp-error-codes.js +27 -0
- package/dist/ncp/ncp-error-codes.js.map +1 -1
- package/dist/ncp/ncp-patch-format.d.ts +0 -0
- package/dist/ncp/ncp-patch-format.d.ts.map +0 -0
- package/dist/ncp/ncp-patch-format.js +0 -0
- package/dist/ncp/ncp-patch-format.js.map +0 -0
- package/dist/ncp/preamble.d.ts +47 -0
- package/dist/ncp/preamble.d.ts.map +1 -0
- package/dist/ncp/preamble.js +74 -0
- package/dist/ncp/preamble.js.map +1 -0
- package/dist/ncp/registry.d.ts +0 -0
- package/dist/ncp/registry.d.ts.map +1 -1
- package/dist/ncp/registry.js +2 -1
- package/dist/ncp/registry.js.map +1 -1
- package/dist/ncp/stream-manager.d.ts +0 -0
- package/dist/ncp/stream-manager.d.ts.map +0 -0
- package/dist/ncp/stream-manager.js +0 -0
- package/dist/ncp/stream-manager.js.map +0 -0
- package/dist/ndp/dns-txt.d.ts +35 -0
- package/dist/ndp/dns-txt.d.ts.map +1 -0
- package/dist/ndp/dns-txt.js +67 -0
- package/dist/ndp/dns-txt.js.map +1 -0
- package/dist/ndp/frames.d.ts +34 -9
- package/dist/ndp/frames.d.ts.map +1 -1
- package/dist/ndp/frames.js +54 -15
- package/dist/ndp/frames.js.map +1 -1
- package/dist/ndp/index.d.ts +3 -0
- package/dist/ndp/index.d.ts.map +1 -1
- package/dist/ndp/index.js +3 -0
- package/dist/ndp/index.js.map +1 -1
- package/dist/ndp/ndp-error-codes.d.ts +25 -0
- package/dist/ndp/ndp-error-codes.d.ts.map +1 -0
- package/dist/ndp/ndp-error-codes.js +48 -0
- package/dist/ndp/ndp-error-codes.js.map +1 -0
- package/dist/ndp/ndp-registry.d.ts +2 -0
- package/dist/ndp/ndp-registry.d.ts.map +1 -1
- package/dist/ndp/ndp-registry.js +25 -0
- package/dist/ndp/ndp-registry.js.map +1 -1
- package/dist/ndp/registry.d.ts +0 -0
- package/dist/ndp/registry.d.ts.map +0 -0
- package/dist/ndp/registry.js +0 -0
- package/dist/ndp/registry.js.map +0 -0
- package/dist/ndp/security.d.ts +8 -0
- package/dist/ndp/security.d.ts.map +1 -0
- package/dist/ndp/security.js +9 -0
- package/dist/ndp/security.js.map +1 -0
- package/dist/ndp/validator.d.ts +0 -0
- package/dist/ndp/validator.d.ts.map +0 -0
- package/dist/ndp/validator.js +0 -0
- package/dist/ndp/validator.js.map +0 -0
- package/dist/nip/acme/client.d.ts +31 -0
- package/dist/nip/acme/client.d.ts.map +1 -0
- package/dist/nip/acme/client.js +136 -0
- package/dist/nip/acme/client.js.map +1 -0
- package/dist/nip/acme/index.d.ts +6 -0
- package/dist/nip/acme/index.d.ts.map +1 -0
- package/dist/nip/acme/index.js +8 -0
- package/dist/nip/acme/index.js.map +1 -0
- package/dist/nip/acme/jws.d.ts +31 -0
- package/dist/nip/acme/jws.d.ts.map +1 -0
- package/dist/nip/acme/jws.js +76 -0
- package/dist/nip/acme/jws.js.map +1 -0
- package/dist/nip/acme/messages.d.ts +71 -0
- package/dist/nip/acme/messages.d.ts.map +1 -0
- package/dist/nip/acme/messages.js +4 -0
- package/dist/nip/acme/messages.js.map +1 -0
- package/dist/nip/acme/server.d.ts +41 -0
- package/dist/nip/acme/server.d.ts.map +1 -0
- package/dist/nip/acme/server.js +458 -0
- package/dist/nip/acme/server.js.map +1 -0
- package/dist/nip/acme/wire.d.ts +19 -0
- package/dist/nip/acme/wire.d.ts.map +1 -0
- package/dist/nip/acme/wire.js +21 -0
- package/dist/nip/acme/wire.js.map +1 -0
- package/dist/nip/assurance-level.d.ts +19 -0
- package/dist/nip/assurance-level.d.ts.map +1 -0
- package/dist/nip/assurance-level.js +38 -0
- package/dist/nip/assurance-level.js.map +1 -0
- package/dist/nip/cert-format.d.ts +5 -0
- package/dist/nip/cert-format.d.ts.map +1 -0
- package/dist/nip/cert-format.js +6 -0
- package/dist/nip/cert-format.js.map +1 -0
- package/dist/nip/error-codes.d.ts +44 -0
- package/dist/nip/error-codes.d.ts.map +1 -0
- package/dist/nip/error-codes.js +97 -0
- package/dist/nip/error-codes.js.map +1 -0
- package/dist/nip/frames.d.ts +19 -1
- package/dist/nip/frames.d.ts.map +1 -1
- package/dist/nip/frames.js +39 -4
- package/dist/nip/frames.js.map +1 -1
- package/dist/nip/identity.d.ts +0 -0
- package/dist/nip/identity.d.ts.map +0 -0
- package/dist/nip/identity.js +0 -0
- package/dist/nip/identity.js.map +0 -0
- package/dist/nip/index.d.ts +7 -0
- package/dist/nip/index.d.ts.map +1 -1
- package/dist/nip/index.js +9 -0
- package/dist/nip/index.js.map +1 -1
- package/dist/nip/registry.d.ts +0 -0
- package/dist/nip/registry.d.ts.map +0 -0
- package/dist/nip/registry.js +0 -0
- package/dist/nip/registry.js.map +0 -0
- package/dist/nip/reputation-client.d.ts +116 -0
- package/dist/nip/reputation-client.d.ts.map +1 -0
- package/dist/nip/reputation-client.js +261 -0
- package/dist/nip/reputation-client.js.map +1 -0
- package/dist/nip/verifier.d.ts +23 -0
- package/dist/nip/verifier.d.ts.map +1 -0
- package/dist/nip/verifier.js +90 -0
- package/dist/nip/verifier.js.map +1 -0
- package/dist/nip/x509/builder.d.ts +35 -0
- package/dist/nip/x509/builder.d.ts.map +1 -0
- package/dist/nip/x509/builder.js +59 -0
- package/dist/nip/x509/builder.js.map +1 -0
- package/dist/nip/x509/index.d.ts +4 -0
- package/dist/nip/x509/index.d.ts.map +1 -0
- package/dist/nip/x509/index.js +6 -0
- package/dist/nip/x509/index.js.map +1 -0
- package/dist/nip/x509/oids.d.ts +16 -0
- package/dist/nip/x509/oids.d.ts.map +1 -0
- package/dist/nip/x509/oids.js +22 -0
- package/dist/nip/x509/oids.js.map +1 -0
- package/dist/nip/x509/verifier.d.ts +26 -0
- package/dist/nip/x509/verifier.d.ts.map +1 -0
- package/dist/nip/x509/verifier.js +171 -0
- package/dist/nip/x509/verifier.js.map +1 -0
- package/dist/nop/client.d.ts +0 -0
- package/dist/nop/client.d.ts.map +0 -0
- package/dist/nop/client.js +0 -0
- package/dist/nop/client.js.map +1 -1
- package/dist/nop/dag-validator.d.ts +15 -0
- package/dist/nop/dag-validator.d.ts.map +1 -0
- package/dist/nop/dag-validator.js +91 -0
- package/dist/nop/dag-validator.js.map +1 -0
- package/dist/nop/frames.d.ts +8 -3
- package/dist/nop/frames.d.ts.map +1 -1
- package/dist/nop/frames.js +21 -6
- package/dist/nop/frames.js.map +1 -1
- package/dist/nop/index.d.ts +3 -0
- package/dist/nop/index.d.ts.map +1 -1
- package/dist/nop/index.js +3 -0
- package/dist/nop/index.js.map +1 -1
- package/dist/nop/models.d.ts +13 -2
- package/dist/nop/models.d.ts.map +1 -1
- package/dist/nop/models.js +9 -0
- package/dist/nop/models.js.map +1 -1
- package/dist/nop/nop-error-codes.d.ts +31 -0
- package/dist/nop/nop-error-codes.d.ts.map +1 -0
- package/dist/nop/nop-error-codes.js +61 -0
- package/dist/nop/nop-error-codes.js.map +1 -0
- package/dist/nop/nop-types.d.ts +0 -0
- package/dist/nop/nop-types.d.ts.map +0 -0
- package/dist/nop/nop-types.js +0 -0
- package/dist/nop/nop-types.js.map +0 -0
- package/dist/nop/orchestrator.d.ts +66 -0
- package/dist/nop/orchestrator.d.ts.map +1 -0
- package/dist/nop/orchestrator.js +229 -0
- package/dist/nop/orchestrator.js.map +1 -0
- package/dist/nop/registry.d.ts +0 -0
- package/dist/nop/registry.d.ts.map +0 -0
- package/dist/nop/registry.js +0 -0
- package/dist/nop/registry.js.map +0 -0
- package/dist/nwp/anchor-client.d.ts +109 -0
- package/dist/nwp/anchor-client.d.ts.map +1 -0
- package/dist/nwp/anchor-client.js +279 -0
- package/dist/nwp/anchor-client.js.map +1 -0
- package/dist/nwp/anchor-server.d.ts +127 -0
- package/dist/nwp/anchor-server.d.ts.map +1 -0
- package/dist/nwp/anchor-server.js +649 -0
- package/dist/nwp/anchor-server.js.map +1 -0
- package/dist/nwp/bridge.d.ts +24 -0
- package/dist/nwp/bridge.d.ts.map +1 -0
- package/dist/nwp/bridge.js +26 -0
- package/dist/nwp/bridge.js.map +1 -0
- package/dist/nwp/cgn.d.ts +19 -0
- package/dist/nwp/cgn.d.ts.map +1 -0
- package/dist/nwp/cgn.js +29 -0
- package/dist/nwp/cgn.js.map +1 -0
- package/dist/nwp/client.d.ts +10 -3
- package/dist/nwp/client.d.ts.map +1 -1
- package/dist/nwp/client.js +58 -7
- package/dist/nwp/client.js.map +1 -1
- package/dist/nwp/frames.d.ts +56 -2
- package/dist/nwp/frames.d.ts.map +1 -1
- package/dist/nwp/frames.js +86 -4
- package/dist/nwp/frames.js.map +1 -1
- package/dist/nwp/http-headers.d.ts +24 -0
- package/dist/nwp/http-headers.d.ts.map +1 -0
- package/dist/nwp/http-headers.js +29 -0
- package/dist/nwp/http-headers.js.map +1 -0
- package/dist/nwp/index.d.ts +9 -0
- package/dist/nwp/index.d.ts.map +1 -1
- package/dist/nwp/index.js +9 -0
- package/dist/nwp/index.js.map +1 -1
- package/dist/nwp/manifest.d.ts +75 -0
- package/dist/nwp/manifest.d.ts.map +1 -0
- package/dist/nwp/manifest.js +5 -0
- package/dist/nwp/manifest.js.map +1 -0
- package/dist/nwp/memory-node-server.d.ts +70 -0
- package/dist/nwp/memory-node-server.d.ts.map +1 -0
- package/dist/nwp/memory-node-server.js +315 -0
- package/dist/nwp/memory-node-server.js.map +1 -0
- package/dist/nwp/nwp-error-codes.d.ts +48 -0
- package/dist/nwp/nwp-error-codes.d.ts.map +1 -0
- package/dist/nwp/nwp-error-codes.js +108 -0
- package/dist/nwp/nwp-error-codes.js.map +1 -0
- package/dist/nwp/registry.d.ts +0 -0
- package/dist/nwp/registry.d.ts.map +0 -0
- package/dist/nwp/registry.js +0 -0
- package/dist/nwp/registry.js.map +0 -0
- package/dist/nwp/reputation.d.ts +37 -0
- package/dist/nwp/reputation.d.ts.map +1 -0
- package/dist/nwp/reputation.js +118 -0
- package/dist/nwp/reputation.js.map +1 -0
- package/dist/setup.d.ts +0 -0
- package/dist/setup.d.ts.map +0 -0
- package/dist/setup.js +0 -0
- package/dist/setup.js.map +0 -0
- package/doc/nps-sdk.core.cn.md +321 -0
- package/doc/nps-sdk.core.md +326 -0
- package/doc/nps-sdk.ncp.cn.md +270 -0
- package/doc/nps-sdk.ncp.md +276 -0
- package/doc/nps-sdk.ndp.cn.md +267 -0
- package/doc/nps-sdk.ndp.md +273 -0
- package/doc/nps-sdk.nip.cn.md +265 -0
- package/doc/nps-sdk.nip.md +272 -0
- package/doc/nps-sdk.nop.cn.md +329 -0
- package/doc/nps-sdk.nop.md +332 -0
- package/doc/nps-sdk.nwp.cn.md +288 -0
- package/doc/nps-sdk.nwp.md +295 -0
- package/doc/overview.cn.md +149 -0
- package/doc/overview.md +153 -0
- package/package.json +33 -4
- package/CONTRIBUTING.md +0 -33
- package/dist/codec-CmHeovTV.d.cts +0 -120
- package/dist/codec-CmHeovTV.d.ts +0 -120
- package/dist/core/index.cjs +0 -371
- package/dist/core/index.cjs.map +0 -1
- package/dist/core/index.d.cts +0 -41
- package/dist/frames-B3qLdl_g.d.cts +0 -77
- package/dist/frames-Ff7-ZPUl.d.ts +0 -77
- package/dist/index.cjs +0 -1556
- package/dist/index.cjs.map +0 -1
- package/dist/index.d.cts +0 -21
- package/dist/ncp/index.cjs +0 -188
- package/dist/ncp/index.cjs.map +0 -1
- package/dist/ncp/index.d.cts +0 -6
- package/dist/ndp/index.cjs +0 -252
- package/dist/ndp/index.cjs.map +0 -1
- package/dist/ndp/index.d.cts +0 -86
- package/dist/nip/index.cjs +0 -214
- package/dist/nip/index.cjs.map +0 -1
- package/dist/nip/index.d.cts +0 -65
- package/dist/nop/index.cjs +0 -762
- package/dist/nop/index.cjs.map +0 -1
- package/dist/nop/index.d.cts +0 -155
- package/dist/nwp/index.cjs +0 -658
- package/dist/nwp/index.cjs.map +0 -1
- package/dist/nwp/index.d.cts +0 -65
- package/nip-ca-server/Dockerfile +0 -27
- package/nip-ca-server/README.md +0 -45
- package/nip-ca-server/db/001_init.sql +0 -25
- package/nip-ca-server/docker-compose.yml +0 -29
- package/nip-ca-server/package.json +0 -23
- package/nip-ca-server/src/ca.ts +0 -155
- package/nip-ca-server/src/db.ts +0 -104
- package/nip-ca-server/src/index.ts +0 -157
- package/nip-ca-server/tsconfig.json +0 -13
- package/src/core/anchor-cache.ts +0 -129
- package/src/core/cache.ts +0 -93
- package/src/core/canonical-json.ts +0 -50
- package/src/core/codec.ts +0 -158
- package/src/core/codecs/index.ts +0 -5
- package/src/core/codecs/ncp-codec.ts +0 -170
- package/src/core/codecs/tier1-json-codec.ts +0 -33
- package/src/core/codecs/tier2-msgpack-codec.ts +0 -30
- package/src/core/crypto-provider.ts +0 -47
- package/src/core/exceptions.ts +0 -57
- package/src/core/frame-header.ts +0 -282
- package/src/core/frame-registry.ts +0 -91
- package/src/core/frames.ts +0 -183
- package/src/core/index.ts +0 -10
- package/src/core/registry.ts +0 -28
- package/src/core/status-codes.ts +0 -46
- package/src/index.ts +0 -10
- package/src/ncp/frames/anchor-frame.ts +0 -87
- package/src/ncp/frames/caps-frame.ts +0 -59
- package/src/ncp/frames/diff-frame.ts +0 -69
- package/src/ncp/frames/error-frame.ts +0 -26
- package/src/ncp/frames/hello-frame.ts +0 -50
- package/src/ncp/frames/stream-frame.ts +0 -35
- package/src/ncp/frames.ts +0 -199
- package/src/ncp/handshake.ts +0 -95
- package/src/ncp/index.ts +0 -12
- package/src/ncp/ncp-error-codes.ts +0 -34
- package/src/ncp/ncp-patch-format.ts +0 -16
- package/src/ncp/registry.ts +0 -14
- package/src/ncp/stream-manager.ts +0 -212
- package/src/ndp/frames.ts +0 -124
- package/src/ndp/index.ts +0 -7
- package/src/ndp/ndp-registry.ts +0 -82
- package/src/ndp/registry.ts +0 -12
- package/src/ndp/validator.ts +0 -64
- package/src/nip/frames.ts +0 -106
- package/src/nip/identity.ts +0 -113
- package/src/nip/index.ts +0 -6
- package/src/nip/registry.ts +0 -12
- package/src/nop/client.ts +0 -103
- package/src/nop/frames.ts +0 -181
- package/src/nop/index.ts +0 -7
- package/src/nop/models.ts +0 -79
- package/src/nop/nop-types.ts +0 -208
- package/src/nop/registry.ts +0 -13
- package/src/nwp/client.ts +0 -114
- package/src/nwp/frames.ts +0 -116
- package/src/nwp/index.ts +0 -6
- package/src/nwp/registry.ts +0 -11
- package/src/setup.ts +0 -32
- package/tests/core/anchor-cache.test.ts +0 -242
- package/tests/core/codec.test.ts +0 -205
- package/tests/core/frame-registry.test.ts +0 -46
- package/tests/core.test.ts +0 -327
- package/tests/ncp/diff-binary-bitset.test.ts +0 -107
- package/tests/ncp/e2e-enc-reject.test.ts +0 -93
- package/tests/ncp/err-error-frame.test.ts +0 -152
- package/tests/ncp/frames.test.ts +0 -359
- package/tests/ncp/framing.test.ts +0 -233
- package/tests/ncp/hello-frame.test.ts +0 -122
- package/tests/ncp/inline-anchor.test.ts +0 -88
- package/tests/ncp/security.test.ts +0 -184
- package/tests/ncp/stream-window.test.ts +0 -167
- package/tests/ncp/stream.test.ts +0 -242
- package/tests/ncp/version-negotiation.test.ts +0 -123
- package/tests/ndp.test.ts +0 -271
- package/tests/nip.test.ts +0 -184
- package/tests/nop.test.ts +0 -344
- package/tests/nwp.test.ts +0 -237
- package/tsconfig.json +0 -20
- package/tsup.config.ts +0 -20
- package/vitest.config.ts +0 -10
|
@@ -1,35 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
-
// Copyright (c) 2026 LabAcacia / INNO LOTUS PTY LTD
|
|
3
|
-
//
|
|
4
|
-
// StreamFrame (0x03) — Streaming data chunks with flow control
|
|
5
|
-
// NPS-1 §4.3
|
|
6
|
-
|
|
7
|
-
import { NcpError } from "../../core/frame-header.js";
|
|
8
|
-
|
|
9
|
-
export interface StreamFrame {
|
|
10
|
-
frame: string;
|
|
11
|
-
stream_id: string;
|
|
12
|
-
seq: number;
|
|
13
|
-
is_last: boolean;
|
|
14
|
-
anchor_ref?: string;
|
|
15
|
-
data: unknown[];
|
|
16
|
-
window_size?: number;
|
|
17
|
-
error_code?: string;
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
// UUID v4 format: xxxxxxxx-xxxx-4xxx-[89ab]xxx-xxxxxxxxxxxx
|
|
21
|
-
const UUID_V4_RE =
|
|
22
|
-
/^[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;
|
|
23
|
-
|
|
24
|
-
/**
|
|
25
|
-
* Validate stream_id is a valid UUID v4.
|
|
26
|
-
* @throws {NcpError} NPS-CLIENT-BAD-FRAME if stream_id is not a valid UUID v4.
|
|
27
|
-
*/
|
|
28
|
-
export function validateStreamFrame(frame: StreamFrame): void {
|
|
29
|
-
if (!UUID_V4_RE.test(frame.stream_id)) {
|
|
30
|
-
throw new NcpError(
|
|
31
|
-
"NPS-CLIENT-BAD-FRAME",
|
|
32
|
-
`stream_id "${frame.stream_id}" is not a valid UUID v4`,
|
|
33
|
-
);
|
|
34
|
-
}
|
|
35
|
-
}
|
package/src/ncp/frames.ts
DELETED
|
@@ -1,199 +0,0 @@
|
|
|
1
|
-
// Copyright 2026 INNO LOTUS PTY LTD
|
|
2
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
3
|
-
|
|
4
|
-
import { EncodingTier, FrameType } from "../core/frames.js";
|
|
5
|
-
import type { NpsFrame } from "../core/codec.js";
|
|
6
|
-
|
|
7
|
-
// ── FrameSchema ───────────────────────────────────────────────────────────────
|
|
8
|
-
|
|
9
|
-
export interface SchemaField {
|
|
10
|
-
name: string;
|
|
11
|
-
type: string;
|
|
12
|
-
semantic?: string;
|
|
13
|
-
nullable?: boolean;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
export interface FrameSchema {
|
|
17
|
-
fields: readonly SchemaField[];
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
// ── AnchorFrame ───────────────────────────────────────────────────────────────
|
|
21
|
-
|
|
22
|
-
export class AnchorFrame implements NpsFrame {
|
|
23
|
-
readonly frameType = FrameType.ANCHOR;
|
|
24
|
-
readonly preferredTier = EncodingTier.MSGPACK;
|
|
25
|
-
|
|
26
|
-
constructor(
|
|
27
|
-
public readonly anchorId: string,
|
|
28
|
-
public readonly schema: FrameSchema,
|
|
29
|
-
public readonly ttl: number = 3600,
|
|
30
|
-
) {}
|
|
31
|
-
|
|
32
|
-
toDict(): Record<string, unknown> {
|
|
33
|
-
return {
|
|
34
|
-
anchor_id: this.anchorId,
|
|
35
|
-
schema: { fields: this.schema.fields.map((f) => ({ ...f })) },
|
|
36
|
-
ttl: this.ttl,
|
|
37
|
-
};
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
static fromDict(data: Record<string, unknown>): AnchorFrame {
|
|
41
|
-
const schemaRaw = data["schema"] as { fields: SchemaField[] };
|
|
42
|
-
return new AnchorFrame(
|
|
43
|
-
data["anchor_id"] as string,
|
|
44
|
-
{ fields: schemaRaw.fields },
|
|
45
|
-
(data["ttl"] as number | undefined) ?? 3600,
|
|
46
|
-
);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
// ── JsonPatchOperation ────────────────────────────────────────────────────────
|
|
51
|
-
|
|
52
|
-
export interface JsonPatchOperation {
|
|
53
|
-
op: string;
|
|
54
|
-
path: string;
|
|
55
|
-
value?: unknown;
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
// ── DiffFrame ─────────────────────────────────────────────────────────────────
|
|
59
|
-
|
|
60
|
-
export class DiffFrame implements NpsFrame {
|
|
61
|
-
readonly frameType = FrameType.DIFF;
|
|
62
|
-
readonly preferredTier = EncodingTier.MSGPACK;
|
|
63
|
-
|
|
64
|
-
constructor(
|
|
65
|
-
public readonly anchorRef: string,
|
|
66
|
-
public readonly baseSeq: number,
|
|
67
|
-
public readonly patch: readonly JsonPatchOperation[],
|
|
68
|
-
public readonly entityId?: string,
|
|
69
|
-
) {}
|
|
70
|
-
|
|
71
|
-
toDict(): Record<string, unknown> {
|
|
72
|
-
return {
|
|
73
|
-
anchor_ref: this.anchorRef,
|
|
74
|
-
base_seq: this.baseSeq,
|
|
75
|
-
patch: this.patch.map((p) => ({ ...p })),
|
|
76
|
-
entity_id: this.entityId ?? null,
|
|
77
|
-
};
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
static fromDict(data: Record<string, unknown>): DiffFrame {
|
|
81
|
-
return new DiffFrame(
|
|
82
|
-
data["anchor_ref"] as string,
|
|
83
|
-
data["base_seq"] as number,
|
|
84
|
-
data["patch"] as JsonPatchOperation[],
|
|
85
|
-
(data["entity_id"] as string | null) ?? undefined,
|
|
86
|
-
);
|
|
87
|
-
}
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
// ── StreamFrame ───────────────────────────────────────────────────────────────
|
|
91
|
-
|
|
92
|
-
export class StreamFrame implements NpsFrame {
|
|
93
|
-
readonly frameType = FrameType.STREAM;
|
|
94
|
-
readonly preferredTier = EncodingTier.MSGPACK;
|
|
95
|
-
|
|
96
|
-
constructor(
|
|
97
|
-
public readonly streamId: string,
|
|
98
|
-
public readonly seq: number,
|
|
99
|
-
public readonly isLast: boolean,
|
|
100
|
-
public readonly data: readonly Record<string, unknown>[],
|
|
101
|
-
public readonly anchorRef?: string,
|
|
102
|
-
public readonly windowSize?: number,
|
|
103
|
-
) {}
|
|
104
|
-
|
|
105
|
-
toDict(): Record<string, unknown> {
|
|
106
|
-
return {
|
|
107
|
-
stream_id: this.streamId,
|
|
108
|
-
seq: this.seq,
|
|
109
|
-
is_last: this.isLast,
|
|
110
|
-
data: this.data,
|
|
111
|
-
anchor_ref: this.anchorRef ?? null,
|
|
112
|
-
window_size: this.windowSize ?? null,
|
|
113
|
-
};
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
static fromDict(data: Record<string, unknown>): StreamFrame {
|
|
117
|
-
return new StreamFrame(
|
|
118
|
-
data["stream_id"] as string,
|
|
119
|
-
data["seq"] as number,
|
|
120
|
-
data["is_last"] as boolean,
|
|
121
|
-
data["data"] as Record<string, unknown>[],
|
|
122
|
-
(data["anchor_ref"] as string | null) ?? undefined,
|
|
123
|
-
(data["window_size"] as number | null) ?? undefined,
|
|
124
|
-
);
|
|
125
|
-
}
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
// ── CapsFrame ─────────────────────────────────────────────────────────────────
|
|
129
|
-
|
|
130
|
-
export class CapsFrame implements NpsFrame {
|
|
131
|
-
readonly frameType = FrameType.CAPS;
|
|
132
|
-
readonly preferredTier = EncodingTier.MSGPACK;
|
|
133
|
-
|
|
134
|
-
constructor(
|
|
135
|
-
public readonly anchorRef: string,
|
|
136
|
-
public readonly count: number,
|
|
137
|
-
public readonly data: readonly Record<string, unknown>[],
|
|
138
|
-
public readonly nextCursor?: string,
|
|
139
|
-
public readonly tokenEst?: number,
|
|
140
|
-
public readonly cached?: boolean,
|
|
141
|
-
public readonly tokenizerUsed?: string,
|
|
142
|
-
) {}
|
|
143
|
-
|
|
144
|
-
toDict(): Record<string, unknown> {
|
|
145
|
-
return {
|
|
146
|
-
anchor_ref: this.anchorRef,
|
|
147
|
-
count: this.count,
|
|
148
|
-
data: this.data,
|
|
149
|
-
next_cursor: this.nextCursor ?? null,
|
|
150
|
-
token_est: this.tokenEst ?? null,
|
|
151
|
-
cached: this.cached ?? null,
|
|
152
|
-
tokenizer_used: this.tokenizerUsed ?? null,
|
|
153
|
-
};
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
static fromDict(data: Record<string, unknown>): CapsFrame {
|
|
157
|
-
return new CapsFrame(
|
|
158
|
-
data["anchor_ref"] as string,
|
|
159
|
-
data["count"] as number,
|
|
160
|
-
data["data"] as Record<string, unknown>[],
|
|
161
|
-
(data["next_cursor"] as string | null) ?? undefined,
|
|
162
|
-
(data["token_est"] as number | null) ?? undefined,
|
|
163
|
-
(data["cached"] as boolean | null) ?? undefined,
|
|
164
|
-
(data["tokenizer_used"] as string | null) ?? undefined,
|
|
165
|
-
);
|
|
166
|
-
}
|
|
167
|
-
}
|
|
168
|
-
|
|
169
|
-
// ── ErrorFrame ────────────────────────────────────────────────────────────────
|
|
170
|
-
|
|
171
|
-
export class ErrorFrame implements NpsFrame {
|
|
172
|
-
readonly frameType = FrameType.ERROR;
|
|
173
|
-
readonly preferredTier = EncodingTier.MSGPACK;
|
|
174
|
-
|
|
175
|
-
constructor(
|
|
176
|
-
public readonly status: string,
|
|
177
|
-
public readonly error: string,
|
|
178
|
-
public readonly message?: string,
|
|
179
|
-
public readonly details?: Record<string, unknown>,
|
|
180
|
-
) {}
|
|
181
|
-
|
|
182
|
-
toDict(): Record<string, unknown> {
|
|
183
|
-
return {
|
|
184
|
-
status: this.status,
|
|
185
|
-
error: this.error,
|
|
186
|
-
message: this.message ?? null,
|
|
187
|
-
details: this.details ?? null,
|
|
188
|
-
};
|
|
189
|
-
}
|
|
190
|
-
|
|
191
|
-
static fromDict(data: Record<string, unknown>): ErrorFrame {
|
|
192
|
-
return new ErrorFrame(
|
|
193
|
-
data["status"] as string,
|
|
194
|
-
data["error"] as string,
|
|
195
|
-
(data["message"] as string | null) ?? undefined,
|
|
196
|
-
(data["details"] as Record<string, unknown> | null) ?? undefined,
|
|
197
|
-
);
|
|
198
|
-
}
|
|
199
|
-
}
|
package/src/ncp/handshake.ts
DELETED
|
@@ -1,95 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
-
// Copyright (c) 2026 LabAcacia / INNO LOTUS PTY LTD
|
|
3
|
-
//
|
|
4
|
-
// Handshake — Version negotiation and encoding negotiation
|
|
5
|
-
// NPS-1 §2.6
|
|
6
|
-
|
|
7
|
-
/**
|
|
8
|
-
* Parse a "major.minor" (optionally "major.minor.patch") version string into a
|
|
9
|
-
* tuple of numeric components. Invalid parts become NaN which makes subsequent
|
|
10
|
-
* comparisons return false in both directions (safe failure).
|
|
11
|
-
*/
|
|
12
|
-
function parseVersion(v: string): number[] {
|
|
13
|
-
return v.split(".").map((p) => Number.parseInt(p, 10));
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
/**
|
|
17
|
-
* Numeric component-wise comparison of two version strings.
|
|
18
|
-
* Returns negative if a < b, zero if equal, positive if a > b.
|
|
19
|
-
* Avoids the lexicographic pitfall where "0.9" > "0.10".
|
|
20
|
-
*/
|
|
21
|
-
function compareVersions(a: string, b: string): number {
|
|
22
|
-
const partsA = parseVersion(a);
|
|
23
|
-
const partsB = parseVersion(b);
|
|
24
|
-
const len = Math.max(partsA.length, partsB.length);
|
|
25
|
-
for (let i = 0; i < len; i += 1) {
|
|
26
|
-
const x = partsA[i] ?? 0;
|
|
27
|
-
const y = partsB[i] ?? 0;
|
|
28
|
-
if (x !== y) return x - y;
|
|
29
|
-
}
|
|
30
|
-
return 0;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
/**
|
|
34
|
-
* Negotiate the session NPS version between client and server.
|
|
35
|
-
*
|
|
36
|
-
* Session version = numeric min of client.nps_version and server.nps_version
|
|
37
|
-
* (component-wise — "0.9" < "0.10" < "1.0").
|
|
38
|
-
* If the effective client minimum (min_version ?? nps_version) > server.nps_version,
|
|
39
|
-
* the versions are incompatible.
|
|
40
|
-
*
|
|
41
|
-
* Spec: NPS-1 §2.6
|
|
42
|
-
*/
|
|
43
|
-
export function negotiateVersion(
|
|
44
|
-
client: { nps_version: string; min_version?: string },
|
|
45
|
-
server: { nps_version: string },
|
|
46
|
-
): { session_version: string; compatible: boolean; error_code?: string } {
|
|
47
|
-
const clientMin = client.min_version ?? client.nps_version;
|
|
48
|
-
const serverVersion = server.nps_version;
|
|
49
|
-
|
|
50
|
-
if (compareVersions(clientMin, serverVersion) > 0) {
|
|
51
|
-
return {
|
|
52
|
-
session_version: serverVersion,
|
|
53
|
-
compatible: false,
|
|
54
|
-
error_code: "NCP-VERSION-INCOMPATIBLE",
|
|
55
|
-
};
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
// Session version = component-wise min of client.nps_version and server.nps_version
|
|
59
|
-
const sessionVersion =
|
|
60
|
-
compareVersions(client.nps_version, serverVersion) <= 0
|
|
61
|
-
? client.nps_version
|
|
62
|
-
: serverVersion;
|
|
63
|
-
|
|
64
|
-
return { session_version: sessionVersion, compatible: true };
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* Negotiate the encoding between client and server preferred lists.
|
|
69
|
-
*
|
|
70
|
-
* Returns the first mutually supported encoding, preferring "msgpack" over "json".
|
|
71
|
-
* Returns null if there is no intersection.
|
|
72
|
-
*/
|
|
73
|
-
export function negotiateEncoding(
|
|
74
|
-
client: string[],
|
|
75
|
-
server: string[],
|
|
76
|
-
): { encoding: string | null } {
|
|
77
|
-
const serverSet = new Set(server);
|
|
78
|
-
|
|
79
|
-
// Prefer msgpack over json (and over any other encoding)
|
|
80
|
-
if (client.includes("msgpack") && serverSet.has("msgpack")) {
|
|
81
|
-
return { encoding: "msgpack" };
|
|
82
|
-
}
|
|
83
|
-
if (client.includes("json") && serverSet.has("json")) {
|
|
84
|
-
return { encoding: "json" };
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
// Fall back to first intersection in client-preference order
|
|
88
|
-
for (const enc of client) {
|
|
89
|
-
if (serverSet.has(enc)) {
|
|
90
|
-
return { encoding: enc };
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
return { encoding: null };
|
|
95
|
-
}
|
package/src/ncp/index.ts
DELETED
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
-
// Copyright (c) 2026 LabAcacia / INNO LOTUS PTY LTD
|
|
3
|
-
export * from "./frames/anchor-frame.js";
|
|
4
|
-
export * from "./frames/caps-frame.js";
|
|
5
|
-
export * from "./frames/diff-frame.js";
|
|
6
|
-
export * from "./frames/error-frame.js";
|
|
7
|
-
export * from "./frames/hello-frame.js";
|
|
8
|
-
export * from "./frames/stream-frame.js";
|
|
9
|
-
export * from "./ncp-error-codes.js";
|
|
10
|
-
export * from "./ncp-patch-format.js";
|
|
11
|
-
export * from "./handshake.js";
|
|
12
|
-
export * from "./stream-manager.js";
|
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
-
// Copyright (c) 2026 LabAcacia / INNO LOTUS PTY LTD
|
|
3
|
-
//
|
|
4
|
-
// NCP Error Codes — All v0.4 protocol error codes
|
|
5
|
-
// NPS-1 §6 + §7.4
|
|
6
|
-
//
|
|
7
|
-
// Implementation-only codes (NCP_FRAME_PARSE_ERROR, NCP_FRAME_INCOMPLETE) cover
|
|
8
|
-
// wire-layer parse failures not in spec §6. See test/ncp_test_results.md spec
|
|
9
|
-
// question 2 for the proposal to register them upstream.
|
|
10
|
-
|
|
11
|
-
export const NCP_ERROR_CODES = {
|
|
12
|
-
// Implementation-only codes (not in spec §6 — see test_results.md spec question 2)
|
|
13
|
-
NCP_FRAME_PARSE_ERROR: "NCP-FRAME-PARSE-ERROR",
|
|
14
|
-
NCP_FRAME_INCOMPLETE: "NCP-FRAME-INCOMPLETE",
|
|
15
|
-
// Spec-defined codes
|
|
16
|
-
NCP_FRAME_UNKNOWN_TYPE: "NCP-FRAME-UNKNOWN-TYPE",
|
|
17
|
-
NCP_FRAME_PAYLOAD_TOO_LARGE: "NCP-FRAME-PAYLOAD-TOO-LARGE",
|
|
18
|
-
NCP_FRAME_FLAGS_INVALID: "NCP-FRAME-FLAGS-INVALID",
|
|
19
|
-
NCP_ANCHOR_NOT_FOUND: "NCP-ANCHOR-NOT-FOUND",
|
|
20
|
-
NCP_ANCHOR_SCHEMA_INVALID: "NCP-ANCHOR-SCHEMA-INVALID",
|
|
21
|
-
NCP_ANCHOR_ID_MISMATCH: "NCP-ANCHOR-ID-MISMATCH",
|
|
22
|
-
NCP_ANCHOR_STALE: "NCP-ANCHOR-STALE",
|
|
23
|
-
NCP_STREAM_SEQ_GAP: "NCP-STREAM-SEQ-GAP",
|
|
24
|
-
NCP_STREAM_NOT_FOUND: "NCP-STREAM-NOT-FOUND",
|
|
25
|
-
NCP_STREAM_LIMIT_EXCEEDED: "NCP-STREAM-LIMIT-EXCEEDED",
|
|
26
|
-
NCP_STREAM_WINDOW_OVERFLOW: "NCP-STREAM-WINDOW-OVERFLOW",
|
|
27
|
-
NCP_ENCODING_UNSUPPORTED: "NCP-ENCODING-UNSUPPORTED",
|
|
28
|
-
NCP_DIFF_FORMAT_UNSUPPORTED: "NCP-DIFF-FORMAT-UNSUPPORTED",
|
|
29
|
-
NCP_VERSION_INCOMPATIBLE: "NCP-VERSION-INCOMPATIBLE",
|
|
30
|
-
NCP_ENC_NOT_NEGOTIATED: "NCP-ENC-NOT-NEGOTIATED",
|
|
31
|
-
NCP_ENC_AUTH_FAILED: "NCP-ENC-AUTH-FAILED",
|
|
32
|
-
} as const;
|
|
33
|
-
|
|
34
|
-
export type NcpErrorCode = typeof NCP_ERROR_CODES[keyof typeof NCP_ERROR_CODES];
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
-
// Copyright (c) 2026 LabAcacia / INNO LOTUS PTY LTD
|
|
3
|
-
//
|
|
4
|
-
// NCP Patch Format — DiffFrame patch encoding types
|
|
5
|
-
// NPS-1 §4.2
|
|
6
|
-
|
|
7
|
-
export const PATCH_FORMAT = {
|
|
8
|
-
JSON_PATCH: "json_patch",
|
|
9
|
-
BINARY_BITSET: "binary_bitset",
|
|
10
|
-
} as const;
|
|
11
|
-
|
|
12
|
-
export type PatchFormat = typeof PATCH_FORMAT[keyof typeof PATCH_FORMAT];
|
|
13
|
-
|
|
14
|
-
export function isValidPatchFormat(v: unknown): v is PatchFormat {
|
|
15
|
-
return v === PATCH_FORMAT.JSON_PATCH || v === PATCH_FORMAT.BINARY_BITSET;
|
|
16
|
-
}
|
package/src/ncp/registry.ts
DELETED
|
@@ -1,14 +0,0 @@
|
|
|
1
|
-
// Copyright 2026 INNO LOTUS PTY LTD
|
|
2
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
3
|
-
|
|
4
|
-
import { FrameRegistry } from "../core/registry.js";
|
|
5
|
-
import { FrameType } from "../core/frames.js";
|
|
6
|
-
import { AnchorFrame, CapsFrame, DiffFrame, ErrorFrame, StreamFrame } from "./frames.js";
|
|
7
|
-
|
|
8
|
-
export function registerNcpFrames(registry: FrameRegistry): void {
|
|
9
|
-
registry.register(FrameType.ANCHOR, AnchorFrame);
|
|
10
|
-
registry.register(FrameType.DIFF, DiffFrame);
|
|
11
|
-
registry.register(FrameType.STREAM, StreamFrame);
|
|
12
|
-
registry.register(FrameType.CAPS, CapsFrame);
|
|
13
|
-
registry.register(FrameType.ERROR, ErrorFrame);
|
|
14
|
-
}
|
|
@@ -1,212 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: Apache-2.0
|
|
2
|
-
// Copyright (c) 2026 LabAcacia / INNO LOTUS PTY LTD
|
|
3
|
-
//
|
|
4
|
-
// StreamManager — Concurrent stream tracking, sequence validation, flow control
|
|
5
|
-
// NPS-1 §4.3, §7.3
|
|
6
|
-
|
|
7
|
-
import { NcpError } from "../core/frame-header.js";
|
|
8
|
-
import { NCP_ERROR_CODES } from "./ncp-error-codes.js";
|
|
9
|
-
import type { StreamFrame } from "./frames/stream-frame.js";
|
|
10
|
-
|
|
11
|
-
interface ActiveStream {
|
|
12
|
-
streamId: string;
|
|
13
|
-
expectedSeq: number;
|
|
14
|
-
chunks: unknown[][];
|
|
15
|
-
completed: boolean;
|
|
16
|
-
errorCode?: string;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
interface OutgoingStream {
|
|
20
|
-
streamId: string;
|
|
21
|
-
remainingWindow: number | undefined; // undefined = no flow control
|
|
22
|
-
paused: boolean;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
/**
|
|
26
|
-
* Manages concurrent StreamFrame streams.
|
|
27
|
-
*
|
|
28
|
-
* - Tracks active streams by stream_id
|
|
29
|
-
* - Validates sequential seq numbers
|
|
30
|
-
* - Enforces max concurrent stream limit (NPS-1 §7.3, default 32)
|
|
31
|
-
* - Detects early termination via error_code
|
|
32
|
-
* - Enforces window-based flow control on outgoing streams (NCP-S-07–11)
|
|
33
|
-
*/
|
|
34
|
-
export class StreamManager {
|
|
35
|
-
private readonly streams = new Map<string, ActiveStream>();
|
|
36
|
-
private readonly outgoing = new Map<string, OutgoingStream>();
|
|
37
|
-
private readonly maxConcurrent: number;
|
|
38
|
-
|
|
39
|
-
constructor(options?: { maxConcurrent?: number }) {
|
|
40
|
-
this.maxConcurrent = options?.maxConcurrent ?? 32;
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
/**
|
|
44
|
-
* Receive a StreamFrame chunk.
|
|
45
|
-
*
|
|
46
|
-
* @returns true if stream is complete (is_last=true or error_code set).
|
|
47
|
-
* @throws {NcpError} NCP-STREAM-LIMIT-EXCEEDED if too many concurrent streams.
|
|
48
|
-
* @throws {NcpError} NCP-STREAM-NOT-FOUND if frame.seq > 0 for a stream that was never opened.
|
|
49
|
-
* @throws {NcpError} NPS-CLIENT-CONFLICT if the stream_id was already completed (stream-id reuse; see test_cases NCP-S-04).
|
|
50
|
-
* @throws {NcpError} NCP-STREAM-SEQ-GAP if sequence number is not expected.
|
|
51
|
-
*/
|
|
52
|
-
receive(frame: StreamFrame): boolean {
|
|
53
|
-
let stream = this.streams.get(frame.stream_id);
|
|
54
|
-
|
|
55
|
-
if (!stream) {
|
|
56
|
-
// A stream is opened only by seq=0. Any other seq on an unknown stream_id
|
|
57
|
-
// means the opener was never seen (NCP-S-13: unknown stream_id).
|
|
58
|
-
if (frame.seq !== 0) {
|
|
59
|
-
throw new NcpError(
|
|
60
|
-
NCP_ERROR_CODES.NCP_STREAM_NOT_FOUND,
|
|
61
|
-
`Unknown stream_id ${frame.stream_id} — first frame must have seq=0`,
|
|
62
|
-
);
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
// New stream — check concurrent limit
|
|
66
|
-
if (this.streams.size >= this.maxConcurrent) {
|
|
67
|
-
throw new NcpError(
|
|
68
|
-
NCP_ERROR_CODES.NCP_STREAM_LIMIT_EXCEEDED,
|
|
69
|
-
`Max concurrent streams (${this.maxConcurrent}) exceeded`,
|
|
70
|
-
);
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
stream = {
|
|
74
|
-
streamId: frame.stream_id,
|
|
75
|
-
expectedSeq: 0,
|
|
76
|
-
chunks: [],
|
|
77
|
-
completed: false,
|
|
78
|
-
};
|
|
79
|
-
this.streams.set(frame.stream_id, stream);
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
// Reject writes to completed streams. Per test_cases NCP-S-04, the spec does
|
|
83
|
-
// not assign a dedicated NCP code for stream-id reuse; interim mapping uses
|
|
84
|
-
// the NPS-level NPS-CLIENT-CONFLICT until a spec-side code is added.
|
|
85
|
-
if (stream.completed) {
|
|
86
|
-
throw new NcpError(
|
|
87
|
-
"NPS-CLIENT-CONFLICT",
|
|
88
|
-
`Stream ${frame.stream_id} is already completed — cannot reuse stream_id`,
|
|
89
|
-
);
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
// Sequence validation
|
|
93
|
-
if (frame.seq !== stream.expectedSeq) {
|
|
94
|
-
// Duplicate detection — same seq as last accepted
|
|
95
|
-
if (frame.seq < stream.expectedSeq) {
|
|
96
|
-
// Ignore duplicate (idempotent)
|
|
97
|
-
return false;
|
|
98
|
-
}
|
|
99
|
-
throw new NcpError(
|
|
100
|
-
NCP_ERROR_CODES.NCP_STREAM_SEQ_GAP,
|
|
101
|
-
`Expected seq ${stream.expectedSeq}, got ${frame.seq} on stream ${frame.stream_id}`,
|
|
102
|
-
);
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
stream.chunks.push(frame.data);
|
|
106
|
-
stream.expectedSeq = frame.seq + 1;
|
|
107
|
-
|
|
108
|
-
// Early termination via error_code
|
|
109
|
-
if (frame.error_code) {
|
|
110
|
-
stream.completed = true;
|
|
111
|
-
stream.errorCode = frame.error_code;
|
|
112
|
-
return true;
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
// Normal completion
|
|
116
|
-
if (frame.is_last) {
|
|
117
|
-
stream.completed = true;
|
|
118
|
-
return true;
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
return false;
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
/**
|
|
125
|
-
* Send a StreamFrame on an outgoing stream, enforcing window-based flow control.
|
|
126
|
-
*
|
|
127
|
-
* - seq=0 with window_size initialises remainingWindow (no decrement for opening frame).
|
|
128
|
-
* - Subsequent sends decrement remainingWindow when flow control is active.
|
|
129
|
-
* - Throws NCP-STREAM-WINDOW-OVERFLOW when remainingWindow === 0.
|
|
130
|
-
*
|
|
131
|
-
* @throws {NcpError} NCP-STREAM-WINDOW-OVERFLOW if window is exhausted.
|
|
132
|
-
*/
|
|
133
|
-
send(frame: StreamFrame): void {
|
|
134
|
-
let out = this.outgoing.get(frame.stream_id);
|
|
135
|
-
|
|
136
|
-
if (!out) {
|
|
137
|
-
out = {
|
|
138
|
-
streamId: frame.stream_id,
|
|
139
|
-
remainingWindow: undefined,
|
|
140
|
-
paused: false,
|
|
141
|
-
};
|
|
142
|
-
this.outgoing.set(frame.stream_id, out);
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
// Opening frame: initialise window from window_size if provided.
|
|
146
|
-
if (frame.seq === 0 && frame.window_size !== undefined) {
|
|
147
|
-
out.remainingWindow = frame.window_size;
|
|
148
|
-
return; // opening frame does not consume a window slot
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
// Flow control check for subsequent frames.
|
|
152
|
-
if (out.remainingWindow !== undefined) {
|
|
153
|
-
if (out.remainingWindow === 0) {
|
|
154
|
-
throw new NcpError(
|
|
155
|
-
NCP_ERROR_CODES.NCP_STREAM_WINDOW_OVERFLOW,
|
|
156
|
-
`Window exhausted on stream ${frame.stream_id}`,
|
|
157
|
-
);
|
|
158
|
-
}
|
|
159
|
-
out.remainingWindow -= 1;
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
/**
|
|
164
|
-
* Update the send window for a stream.
|
|
165
|
-
*
|
|
166
|
-
* Called when a reverse-direction StreamFrame arrives with data=[] and window_size set.
|
|
167
|
-
* Replaces remainingWindow with new_size. Sets paused=true when new_size === 0.
|
|
168
|
-
*/
|
|
169
|
-
updateWindow(streamId: string, newSize: number): void {
|
|
170
|
-
let out = this.outgoing.get(streamId);
|
|
171
|
-
if (!out) {
|
|
172
|
-
out = {
|
|
173
|
-
streamId,
|
|
174
|
-
remainingWindow: newSize,
|
|
175
|
-
paused: newSize === 0,
|
|
176
|
-
};
|
|
177
|
-
this.outgoing.set(streamId, out);
|
|
178
|
-
return;
|
|
179
|
-
}
|
|
180
|
-
out.remainingWindow = newSize;
|
|
181
|
-
out.paused = newSize === 0;
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
/**
|
|
185
|
-
* Returns true when the outgoing stream is paused (window=0 was received).
|
|
186
|
-
* Resumes (returns false) once a non-zero window update arrives.
|
|
187
|
-
*/
|
|
188
|
-
isPaused(streamId: string): boolean {
|
|
189
|
-
return this.outgoing.get(streamId)?.paused ?? false;
|
|
190
|
-
}
|
|
191
|
-
|
|
192
|
-
/** Get reassembled data for a completed stream. */
|
|
193
|
-
getData(streamId: string): unknown[] | null {
|
|
194
|
-
const stream = this.streams.get(streamId);
|
|
195
|
-
if (!stream || !stream.completed) return null;
|
|
196
|
-
return stream.chunks.flat();
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
/** Get error code if stream terminated with error. */
|
|
200
|
-
getError(streamId: string): string | undefined {
|
|
201
|
-
return this.streams.get(streamId)?.errorCode;
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
/** Number of active (non-completed) streams. */
|
|
205
|
-
get activeCount(): number {
|
|
206
|
-
let count = 0;
|
|
207
|
-
for (const s of this.streams.values()) {
|
|
208
|
-
if (!s.completed) count++;
|
|
209
|
-
}
|
|
210
|
-
return count;
|
|
211
|
-
}
|
|
212
|
-
}
|