Compare commits

..

2 Commits

6 changed files with 33 additions and 3 deletions

View File

@@ -1,5 +1,13 @@
# Changelog
## 2025-12-03 - 3.1.11 - fix(virtualstream)
Expose transport localData to handlers via TypedTools; improve VirtualStream payload encode/decode to preserve built-ins and handle nested arrays/objects
- TypedHandler: pass typedRequest.localData into a TypedTools instance so handlers can access transport-layer context (e.g. websocket peer).
- TypedTools: add a public localData property to store transport-specific context available to handlers.
- VirtualStream.decodePayloadFromNetwork: preserve built-in objects (Set, Map, Date, RegExp, Error, Promise or thenable) to avoid incorrect transformation.
- VirtualStream.encodePayloadForNetwork / decodePayloadFromNetwork: added proper recursion for arrays and objects to correctly handle nested payloads and virtual streams, with path tracking to support deduplication logic.
## 2024-10-16 - 3.1.10 - fix(VirtualStream)
Fix stream closure logic in `writeToWebstream` method

View File

@@ -1,6 +1,6 @@
{
"name": "@api.global/typedrequest",
"version": "3.1.10",
"version": "3.1.11",
"private": false,
"description": "A TypeScript library for making typed requests towards APIs, including facilities for handling requests, routing, and virtual stream handling.",
"main": "dist_ts/index.js",

View File

@@ -3,6 +3,6 @@
*/
export const commitinfo = {
name: '@api.global/typedrequest',
version: '3.1.10',
version: '3.1.11',
description: 'A TypeScript library for making typed requests towards APIs, including facilities for handling requests, routing, and virtual stream handling.'
}

View File

@@ -31,6 +31,10 @@ export class TypedHandler<T extends plugins.typedRequestInterfaces.ITypedRequest
}
let typedResponseError: TypedResponseError;
const typedtoolsInstance = new TypedTools();
// Pass localData from the request to TypedTools so handlers can access transport-layer context
if (typedRequestArg.localData) {
typedtoolsInstance.localData = typedRequestArg.localData;
}
const response = await this.handlerFunction(typedRequestArg.request, typedtoolsInstance).catch((e) => {
if (e instanceof TypedResponseError) {
typedResponseError = e;

View File

@@ -2,6 +2,12 @@ import { TypedResponseError } from './classes.typedresponseerror.js';
import * as plugins from './plugins.js';
export class TypedTools {
/**
* Local data passed from the transport layer.
* This can contain connection-specific context like the WebSocket peer.
*/
public localData: Record<string, any> = {};
public async passGuards<T = any>(guardsArg: plugins.smartguard.Guard<T>[], dataArg: T) {
const guardSet = new plugins.smartguard.GuardSet<T>(guardsArg);
const guardResult = await guardSet.allGuardsPass(dataArg);

View File

@@ -82,7 +82,7 @@ export class VirtualStream<T = Uint8Array> implements plugins.typedRequestInterf
}
public static decodePayloadFromNetwork(objectPayload: any, commFunctions: ICommFunctions): any {
if (
plugins.smartbuffer.isBufferLike(objectPayload)
|| objectPayload instanceof TypedRouter
@@ -90,6 +90,18 @@ export class VirtualStream<T = Uint8Array> implements plugins.typedRequestInterf
return objectPayload;
}
if (objectPayload !== null && typeof objectPayload === 'object') {
// Preserve built-in objects that shouldn't be transformed
if (
objectPayload instanceof Set ||
objectPayload instanceof Map ||
objectPayload instanceof Date ||
objectPayload instanceof RegExp ||
objectPayload instanceof Error ||
objectPayload instanceof Promise ||
typeof objectPayload.then === 'function'
) {
return objectPayload;
}
if (objectPayload._isVirtualStream) {
const virtualStream = new VirtualStream();
virtualStream.streamId = objectPayload.streamId;