@candypoets/nipworker 0.0.24 → 0.0.26

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/dist/hooks.d.ts CHANGED
@@ -1,8 +1,8 @@
1
1
  import { NostrEvent } from 'nostr-tools';
2
- import { SubscriptionOptions } from '.';
2
+ import { SubscriptionOptions, NostrManager } from '.';
3
3
  import { Request } from './types';
4
4
 
5
- export declare function useSubscription(subId: string, requests: Request[], callback?: any, options?: SubscriptionOptions): () => void;
5
+ export declare function useSubscription(subId: string, requests: Request[], callback?: any, options?: SubscriptionOptions, manager?: NostrManager): () => void;
6
6
  export declare function usePublish(pubId: string, event: NostrEvent, callback?: any, options?: {
7
7
  trackStatus?: boolean;
8
8
  }): () => void;
@@ -1 +1 @@
1
- {"version":3,"file":"hooks.d.ts","sourceRoot":"","sources":["../src/hooks.ts"],"names":[],"mappings":"AAAA,OAAO,EAAiB,UAAU,EAAE,MAAM,aAAa,CAAC;AACxD,OAAO,EAAgB,mBAAmB,EAAE,MAAM,GAAG,CAAC;AAEtD,OAAO,KAAK,EAAuB,OAAO,EAAE,MAAM,WAAW,CAAC;AAE9D,wBAAgB,eAAe,CAC7B,KAAK,EAAE,MAAM,EACb,QAAQ,EAAE,OAAO,EAAE,EACnB,QAAQ,GAAE,GAAc,EACxB,OAAO,GAAE,mBAA4C,GACpD,MAAM,IAAI,CAsFZ;AAGD,wBAAgB,UAAU,CACxB,KAAK,EAAE,MAAM,EACb,KAAK,EAAE,UAAU,EACjB,QAAQ,GAAE,GAAc,EACxB,OAAO,GAAE;IAAE,WAAW,CAAC,EAAE,OAAO,CAAA;CAA0B,GACzD,MAAM,IAAI,CA2CZ"}
1
+ {"version":3,"file":"hooks.d.ts","sourceRoot":"","sources":["../src/hooks.ts"],"names":[],"mappings":"AAAA,OAAO,EAAG,UAAU,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAgB,mBAAmB,EAAE,KAAK,YAAY,EAAE,MAAM,GAAG,CAAC;AAEzE,OAAO,KAAK,EAAuB,OAAO,EAAE,MAAM,WAAW,CAAC;AAE9D,wBAAgB,eAAe,CAC7B,KAAK,EAAE,MAAM,EACb,QAAQ,EAAE,OAAO,EAAE,EACnB,QAAQ,GAAE,GAAc,EACxB,OAAO,GAAE,mBAA4C,EACrD,OAAO,GAAE,YAA2B,GACnC,MAAM,IAAI,CAsFZ;AAGD,wBAAgB,UAAU,CACxB,KAAK,EAAE,MAAM,EACb,KAAK,EAAE,UAAU,EACjB,QAAQ,GAAE,GAAc,EACxB,OAAO,GAAE;IAAE,WAAW,CAAC,EAAE,OAAO,CAAA;CAA0B,GACzD,MAAM,IAAI,CA2CZ"}
package/dist/hooks.js CHANGED
@@ -1 +1 @@
1
- import{n as T,S as m}from"./index3.js";function C(i,S,o=()=>{},l={closeOnEose:!1}){if(!i)return console.warn("useSharedSubscription: No subscription ID provided"),()=>{};let s=null,r=4,e=null,t=15;const E=4e3;let a=!0,u=!1,c=!1;const b=()=>{a=!1,e!==null&&clearTimeout(e),c&&!u&&(T.unsubscribe(i),u=!0)};s=T.subscribe(i,S,l),c=!0;const d=()=>{if(!a||!s){e!==null&&clearTimeout(e);return}const p=m.readMessages(s,r);p.hasNewData?(t=32,p.messages.forEach(n=>{"SubscriptionEvent"in n?n.SubscriptionEvent.event_type==="BUFFER_FULL"?(o([],"BUFFER_FULL"),b()):n.SubscriptionEvent.event_data.forEach(w=>{o(w,n.SubscriptionEvent.event_type)}):"ConnectionStatus"in n?(l.closeOnEose&&n.ConnectionStatus.status=="EOSE"&&b(),o(n.ConnectionStatus,"CONNECTION_STATUS")):"Eoce"in n?o([],"EOCE"):"Proofs"in n?o(n.Proofs):"Count"in n&&o([n.Count],"Count")}),r=p.newReadPosition):t=Math.min(t*2,E),e!==null&&clearTimeout(e),e=window.setTimeout(d,t)};return e=window.setTimeout(d,0),b}function f(i,S,o=()=>{},l={trackStatus:!0}){if(!i)return console.warn("usePublish: No publish ID provided"),()=>{};let s=null,r=4,e=null,t=!0;const E=()=>{t=!1,e!==null&&clearTimeout(e)};if(s=T.publish(i,S),l.trackStatus&&s){const a=()=>{if(!t||!s){e!==null&&clearTimeout(e);return}const u=m.readMessages(s,r);u.hasNewData&&(u.messages.forEach(c=>{"ConnectionStatus"in c&&o(c.ConnectionStatus,"CONNECTION_STATUS")}),r=u.newReadPosition),e=window.setTimeout(a,50)};e=window.setTimeout(a,0)}return E}export{f as usePublish,C as useSubscription};
1
+ import{n as m,S as w}from"./index3.js";function C(i,E,t=()=>{},l={closeOnEose:!1},o=m){if(!i)return console.warn("useSharedSubscription: No subscription ID provided"),()=>{};let u=null,s=4,e=null,r=15;const S=4e3;let a=!0,c=!1,T=!1;const b=()=>{a=!1,e!==null&&clearTimeout(e),T&&!c&&(o.unsubscribe(i),c=!0)};u=o.subscribe(i,E,l),T=!0;const d=()=>{if(!a||!u){e!==null&&clearTimeout(e);return}const p=w.readMessages(u,s);p.hasNewData?(r=32,p.messages.forEach(n=>{"SubscriptionEvent"in n?n.SubscriptionEvent.event_type==="BUFFER_FULL"?(t([],"BUFFER_FULL"),b()):n.SubscriptionEvent.event_data.forEach(f=>{t(f,n.SubscriptionEvent.event_type)}):"ConnectionStatus"in n?(l.closeOnEose&&n.ConnectionStatus.status=="EOSE"&&b(),t(n.ConnectionStatus,"CONNECTION_STATUS")):"Eoce"in n?t([],"EOCE"):"Proofs"in n?t(n.Proofs):"Count"in n&&t([n.Count],"Count")}),s=p.newReadPosition):r=Math.min(r*2,S),e!==null&&clearTimeout(e),e=window.setTimeout(d,r)};return e=window.setTimeout(d,0),b}function h(i,E,t=()=>{},l={trackStatus:!0}){if(!i)return console.warn("usePublish: No publish ID provided"),()=>{};let o=null,u=4,s=null,e=!0;const r=()=>{e=!1,s!==null&&clearTimeout(s)};if(o=m.publish(i,E),l.trackStatus&&o){const S=()=>{if(!e||!o){s!==null&&clearTimeout(s);return}const a=w.readMessages(o,u);a.hasNewData&&(a.messages.forEach(c=>{"ConnectionStatus"in c&&t(c.ConnectionStatus,"CONNECTION_STATUS")}),u=a.newReadPosition),s=window.setTimeout(S,50)};s=window.setTimeout(S,0)}return r}export{h as usePublish,C as useSubscription};
package/dist/hooks.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"hooks.js","sources":["../src/hooks.ts"],"sourcesContent":["import { EventTemplate, NostrEvent } from \"nostr-tools\";\nimport { nostrManager, SubscriptionOptions } from \".\";\nimport { SharedBufferReader } from \"src/lib/sharedBuffer\";\nimport type { WorkerToMainMessage, Request } from \"src/types\";\n\nexport function useSubscription(\n subId: string,\n requests: Request[],\n callback: any = () => {},\n options: SubscriptionOptions = { closeOnEose: false },\n): () => void {\n if (!subId) {\n console.warn(\"useSharedSubscription: No subscription ID provided\");\n return () => {};\n }\n let buffer: SharedArrayBuffer | null = null;\n let lastReadPos: number = 4;\n let timeoutId: number | null = null;\n let pollInterval: number = 15; // Start at 5ms - very aggressive\n const maxInterval: number = 4000; // Max 4 seconds\n let running: boolean = true;\n\n let hasUnsubscribed = false;\n let hasSubscribed = false;\n\n const unsubscribe = (): void => {\n running = false;\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n if (hasSubscribed && !hasUnsubscribed) {\n nostrManager.unsubscribe(subId);\n hasUnsubscribed = true;\n }\n };\n\n buffer = nostrManager.subscribe(subId, requests, options);\n\n hasSubscribed = true;\n\n const processEvents = (): void => {\n if (!running || !buffer) {\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n return;\n }\n\n const result = SharedBufferReader.readMessages(buffer, lastReadPos);\n\n if (result.hasNewData) {\n // Found new data - reset to aggressive polling\n pollInterval = 32;\n\n result.messages.forEach((message: WorkerToMainMessage) => {\n if (\"SubscriptionEvent\" in message) {\n if (message.SubscriptionEvent.event_type === \"BUFFER_FULL\" as any) {\n callback([], \"BUFFER_FULL\");\n unsubscribe()\n } else {\n message.SubscriptionEvent.event_data.forEach((event) => {\n callback(event, message.SubscriptionEvent.event_type);\n });\n }\n } else if (\"ConnectionStatus\" in message) {\n if (options.closeOnEose && message.ConnectionStatus.status == \"EOSE\") {\n unsubscribe();\n }\n callback(message.ConnectionStatus, \"CONNECTION_STATUS\");\n } else if (\"Eoce\" in message) {\n callback([], \"EOCE\");\n } else if (\"Proofs\" in message) {\n callback(message.Proofs);\n } else if (\"Count\" in message) {\n callback([message.Count], \"Count\")\n }\n });\n lastReadPos = result.newReadPosition;\n } else {\n // No new data - back off exponentially (faster backoff)\n pollInterval = Math.min(pollInterval * 2, maxInterval);\n }\n\n // Clear any existing timeout before scheduling a new one\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n\n // Schedule next poll\n timeoutId = window.setTimeout(processEvents, pollInterval);\n };\n\n // Start after a minimal delay to ensure the return function is available\n timeoutId = window.setTimeout(processEvents, 0);\n\n return unsubscribe\n}\n\n\nexport function usePublish(\n pubId: string,\n event: NostrEvent,\n callback: any = () => {},\n options: { trackStatus?: boolean } = { trackStatus: true }\n): () => void {\n if (!pubId) {\n console.warn(\"usePublish: No publish ID provided\");\n return () => {};\n }\n\n let buffer: SharedArrayBuffer | null = null;\n let lastReadPos: number = 4;\n let timeoutId: number | null = null;\n let running = true;\n\n const unsubscribe = (): void => {\n running = false;\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n };\n\n buffer = nostrManager.publish(pubId, event);\n\n if (options.trackStatus && buffer) {\n const poll = (): void => {\n if (!running || !buffer) {\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n return;\n }\n const result = SharedBufferReader.readMessages(buffer, lastReadPos);\n if (result.hasNewData) {\n result.messages.forEach((message: WorkerToMainMessage) => {\n if (\"ConnectionStatus\" in message) {\n callback(message.ConnectionStatus, \"CONNECTION_STATUS\");\n }\n });\n lastReadPos = result.newReadPosition;\n }\n timeoutId = window.setTimeout(poll, 50);\n };\n timeoutId = window.setTimeout(poll, 0);\n }\n\n return unsubscribe;\n}\n"],"names":["useSubscription","subId","requests","callback","options","buffer","lastReadPos","timeoutId","pollInterval","maxInterval","running","hasUnsubscribed","hasSubscribed","unsubscribe","nostrManager","processEvents","result","SharedBufferReader","message","event","usePublish","pubId","poll"],"mappings":";AAKO,SAASA,EACdC,GACAC,GACAC,IAAgB,MAAM;AAAC,GACvBC,IAA+B,EAAE,aAAa,MAClC;AACZ,MAAI,CAACH;AACH,mBAAQ,KAAK,oDAAoD,GAC1D,MAAM;AAAA,IAAC;AAEhB,MAAII,IAAmC,MACnCC,IAAsB,GACtBC,IAA2B,MAC3BC,IAAuB;AAC3B,QAAMC,IAAsB;AAC5B,MAAIC,IAAmB,IAEnBC,IAAkB,IAClBC,IAAgB;AAEpB,QAAMC,IAAc,MAAY;AAC9B,IAAAH,IAAU,IACNH,MAAc,QAChB,aAAaA,CAAS,GAEpBK,KAAiB,CAACD,MACpBG,EAAa,YAAYb,CAAK,GAC9BU,IAAkB;AAAA,EAEtB;AAEA,EAAAN,IAASS,EAAa,UAAUb,GAAOC,GAAUE,CAAO,GAExDQ,IAAgB;AAEhB,QAAMG,IAAgB,MAAY;AAChC,QAAI,CAACL,KAAW,CAACL,GAAQ;AACvB,MAAIE,MAAc,QAChB,aAAaA,CAAS;AAExB;AAAA,IACF;AAEA,UAAMS,IAASC,EAAmB,aAAaZ,GAAQC,CAAW;AAElE,IAAIU,EAAO,cAETR,IAAe,IAEfQ,EAAO,SAAS,QAAQ,CAACE,MAAiC;AACxD,MAAI,uBAAuBA,IACrBA,EAAQ,kBAAkB,eAAe,iBAC3Cf,EAAS,CAAA,GAAI,aAAa,GAC1BU,EAAA,KAEAK,EAAQ,kBAAkB,WAAW,QAAQ,CAACC,MAAU;AACtD,QAAAhB,EAASgB,GAAOD,EAAQ,kBAAkB,UAAU;AAAA,MACtD,CAAC,IAEM,sBAAsBA,KAC3Bd,EAAQ,eAAec,EAAQ,iBAAiB,UAAU,UAC5DL,EAAA,GAEFV,EAASe,EAAQ,kBAAkB,mBAAmB,KAC7C,UAAUA,IACnBf,EAAS,CAAA,GAAI,MAAM,IACV,YAAYe,IACrBf,EAASe,EAAQ,MAAM,IACd,WAAWA,KACpBf,EAAS,CAACe,EAAQ,KAAK,GAAG,OAAO;AAAA,IAErC,CAAC,GACDZ,IAAcU,EAAO,mBAGrBR,IAAe,KAAK,IAAIA,IAAe,GAAGC,CAAW,GAInDF,MAAc,QAChB,aAAaA,CAAS,GAIxBA,IAAY,OAAO,WAAWQ,GAAeP,CAAY;AAAA,EAC3D;AAGA,SAAAD,IAAY,OAAO,WAAWQ,GAAe,CAAC,GAEvCF;AACT;AAGO,SAASO,EACdC,GACAF,GACAhB,IAAgB,MAAM;AAAC,GACvBC,IAAqC,EAAE,aAAa,MACxC;AACZ,MAAI,CAACiB;AACH,mBAAQ,KAAK,oCAAoC,GAC1C,MAAM;AAAA,IAAC;AAGhB,MAAIhB,IAAmC,MACnCC,IAAsB,GACtBC,IAA2B,MAC3BG,IAAU;AAEd,QAAMG,IAAc,MAAY;AAC9B,IAAAH,IAAU,IACNH,MAAc,QAChB,aAAaA,CAAS;AAAA,EAE1B;AAIA,MAFAF,IAASS,EAAa,QAAQO,GAAOF,CAAK,GAEtCf,EAAQ,eAAeC,GAAQ;AACjC,UAAMiB,IAAO,MAAY;AACvB,UAAI,CAACZ,KAAW,CAACL,GAAQ;AACvB,QAAIE,MAAc,QAChB,aAAaA,CAAS;AAExB;AAAA,MACF;AACA,YAAMS,IAASC,EAAmB,aAAaZ,GAAQC,CAAW;AAClE,MAAIU,EAAO,eACTA,EAAO,SAAS,QAAQ,CAACE,MAAiC;AACxD,QAAI,sBAAsBA,KACxBf,EAASe,EAAQ,kBAAkB,mBAAmB;AAAA,MAE1D,CAAC,GACDZ,IAAcU,EAAO,kBAEvBT,IAAY,OAAO,WAAWe,GAAM,EAAE;AAAA,IACxC;AACA,IAAAf,IAAY,OAAO,WAAWe,GAAM,CAAC;AAAA,EACvC;AAEA,SAAOT;AACT;"}
1
+ {"version":3,"file":"hooks.js","sources":["../src/hooks.ts"],"sourcesContent":["import { NostrEvent } from \"nostr-tools\";\nimport { nostrManager, SubscriptionOptions, type NostrManager } from \".\";\nimport { SharedBufferReader } from \"src/lib/sharedBuffer\";\nimport type { WorkerToMainMessage, Request } from \"src/types\";\n\nexport function useSubscription(\n subId: string,\n requests: Request[],\n callback: any = () => {},\n options: SubscriptionOptions = { closeOnEose: false },\n manager: NostrManager = nostrManager\n): () => void {\n if (!subId) {\n console.warn(\"useSharedSubscription: No subscription ID provided\");\n return () => {};\n }\n let buffer: SharedArrayBuffer | null = null;\n let lastReadPos: number = 4;\n let timeoutId: number | null = null;\n let pollInterval: number = 15; // Start at 5ms - very aggressive\n const maxInterval: number = 4000; // Max 4 seconds\n let running: boolean = true;\n\n let hasUnsubscribed = false;\n let hasSubscribed = false;\n\n const unsubscribe = (): void => {\n running = false;\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n if (hasSubscribed && !hasUnsubscribed) {\n manager.unsubscribe(subId);\n hasUnsubscribed = true;\n }\n };\n\n buffer = manager.subscribe(subId, requests, options);\n\n hasSubscribed = true;\n\n const processEvents = (): void => {\n if (!running || !buffer) {\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n return;\n }\n\n const result = SharedBufferReader.readMessages(buffer, lastReadPos);\n\n if (result.hasNewData) {\n // Found new data - reset to aggressive polling\n pollInterval = 32;\n\n result.messages.forEach((message: WorkerToMainMessage) => {\n if (\"SubscriptionEvent\" in message) {\n if (message.SubscriptionEvent.event_type === \"BUFFER_FULL\" as any) {\n callback([], \"BUFFER_FULL\");\n unsubscribe()\n } else {\n message.SubscriptionEvent.event_data.forEach((event) => {\n callback(event, message.SubscriptionEvent.event_type);\n });\n }\n } else if (\"ConnectionStatus\" in message) {\n if (options.closeOnEose && message.ConnectionStatus.status == \"EOSE\") {\n unsubscribe();\n }\n callback(message.ConnectionStatus, \"CONNECTION_STATUS\");\n } else if (\"Eoce\" in message) {\n callback([], \"EOCE\");\n } else if (\"Proofs\" in message) {\n callback(message.Proofs);\n } else if (\"Count\" in message) {\n callback([message.Count], \"Count\")\n }\n });\n lastReadPos = result.newReadPosition;\n } else {\n // No new data - back off exponentially (faster backoff)\n pollInterval = Math.min(pollInterval * 2, maxInterval);\n }\n\n // Clear any existing timeout before scheduling a new one\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n\n // Schedule next poll\n timeoutId = window.setTimeout(processEvents, pollInterval);\n };\n\n // Start after a minimal delay to ensure the return function is available\n timeoutId = window.setTimeout(processEvents, 0);\n\n return unsubscribe\n}\n\n\nexport function usePublish(\n pubId: string,\n event: NostrEvent,\n callback: any = () => {},\n options: { trackStatus?: boolean } = { trackStatus: true }\n): () => void {\n if (!pubId) {\n console.warn(\"usePublish: No publish ID provided\");\n return () => {};\n }\n\n let buffer: SharedArrayBuffer | null = null;\n let lastReadPos: number = 4;\n let timeoutId: number | null = null;\n let running = true;\n\n const unsubscribe = (): void => {\n running = false;\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n };\n\n buffer = nostrManager.publish(pubId, event);\n\n if (options.trackStatus && buffer) {\n const poll = (): void => {\n if (!running || !buffer) {\n if (timeoutId !== null) {\n clearTimeout(timeoutId);\n }\n return;\n }\n const result = SharedBufferReader.readMessages(buffer, lastReadPos);\n if (result.hasNewData) {\n result.messages.forEach((message: WorkerToMainMessage) => {\n if (\"ConnectionStatus\" in message) {\n callback(message.ConnectionStatus, \"CONNECTION_STATUS\");\n }\n });\n lastReadPos = result.newReadPosition;\n }\n timeoutId = window.setTimeout(poll, 50);\n };\n timeoutId = window.setTimeout(poll, 0);\n }\n\n return unsubscribe;\n}\n"],"names":["useSubscription","subId","requests","callback","options","manager","nostrManager","buffer","lastReadPos","timeoutId","pollInterval","maxInterval","running","hasUnsubscribed","hasSubscribed","unsubscribe","processEvents","result","SharedBufferReader","message","event","usePublish","pubId","poll"],"mappings":";AAKO,SAASA,EACdC,GACAC,GACAC,IAAgB,MAAM;AAAC,GACvBC,IAA+B,EAAE,aAAa,GAAA,GAC9CC,IAAwBC,GACZ;AACZ,MAAI,CAACL;AACH,mBAAQ,KAAK,oDAAoD,GAC1D,MAAM;AAAA,IAAC;AAEhB,MAAIM,IAAmC,MACnCC,IAAsB,GACtBC,IAA2B,MAC3BC,IAAuB;AAC3B,QAAMC,IAAsB;AAC5B,MAAIC,IAAmB,IAEnBC,IAAkB,IAClBC,IAAgB;AAEpB,QAAMC,IAAc,MAAY;AAC9B,IAAAH,IAAU,IACNH,MAAc,QAChB,aAAaA,CAAS,GAEpBK,KAAiB,CAACD,MACpBR,EAAQ,YAAYJ,CAAK,GACzBY,IAAkB;AAAA,EAEtB;AAEA,EAAAN,IAASF,EAAQ,UAAUJ,GAAOC,GAAUE,CAAO,GAEnDU,IAAgB;AAEhB,QAAME,IAAgB,MAAY;AAChC,QAAI,CAACJ,KAAW,CAACL,GAAQ;AACvB,MAAIE,MAAc,QAChB,aAAaA,CAAS;AAExB;AAAA,IACF;AAEA,UAAMQ,IAASC,EAAmB,aAAaX,GAAQC,CAAW;AAElE,IAAIS,EAAO,cAETP,IAAe,IAEfO,EAAO,SAAS,QAAQ,CAACE,MAAiC;AACxD,MAAI,uBAAuBA,IACrBA,EAAQ,kBAAkB,eAAe,iBAC3ChB,EAAS,CAAA,GAAI,aAAa,GAC1BY,EAAA,KAEAI,EAAQ,kBAAkB,WAAW,QAAQ,CAACC,MAAU;AACtD,QAAAjB,EAASiB,GAAOD,EAAQ,kBAAkB,UAAU;AAAA,MACtD,CAAC,IAEM,sBAAsBA,KAC3Bf,EAAQ,eAAee,EAAQ,iBAAiB,UAAU,UAC5DJ,EAAA,GAEFZ,EAASgB,EAAQ,kBAAkB,mBAAmB,KAC7C,UAAUA,IACnBhB,EAAS,CAAA,GAAI,MAAM,IACV,YAAYgB,IACrBhB,EAASgB,EAAQ,MAAM,IACd,WAAWA,KACpBhB,EAAS,CAACgB,EAAQ,KAAK,GAAG,OAAO;AAAA,IAErC,CAAC,GACDX,IAAcS,EAAO,mBAGrBP,IAAe,KAAK,IAAIA,IAAe,GAAGC,CAAW,GAInDF,MAAc,QAChB,aAAaA,CAAS,GAIxBA,IAAY,OAAO,WAAWO,GAAeN,CAAY;AAAA,EAC3D;AAGA,SAAAD,IAAY,OAAO,WAAWO,GAAe,CAAC,GAEvCD;AACT;AAGO,SAASM,EACdC,GACAF,GACAjB,IAAgB,MAAM;AAAC,GACvBC,IAAqC,EAAE,aAAa,MACxC;AACZ,MAAI,CAACkB;AACH,mBAAQ,KAAK,oCAAoC,GAC1C,MAAM;AAAA,IAAC;AAGhB,MAAIf,IAAmC,MACnCC,IAAsB,GACtBC,IAA2B,MAC3BG,IAAU;AAEd,QAAMG,IAAc,MAAY;AAC9B,IAAAH,IAAU,IACNH,MAAc,QAChB,aAAaA,CAAS;AAAA,EAE1B;AAIA,MAFAF,IAASD,EAAa,QAAQgB,GAAOF,CAAK,GAEtChB,EAAQ,eAAeG,GAAQ;AACjC,UAAMgB,IAAO,MAAY;AACvB,UAAI,CAACX,KAAW,CAACL,GAAQ;AACvB,QAAIE,MAAc,QAChB,aAAaA,CAAS;AAExB;AAAA,MACF;AACA,YAAMQ,IAASC,EAAmB,aAAaX,GAAQC,CAAW;AAClE,MAAIS,EAAO,eACTA,EAAO,SAAS,QAAQ,CAACE,MAAiC;AACxD,QAAI,sBAAsBA,KACxBhB,EAASgB,EAAQ,kBAAkB,mBAAmB;AAAA,MAE1D,CAAC,GACDX,IAAcS,EAAO,kBAEvBR,IAAY,OAAO,WAAWc,GAAM,EAAE;AAAA,IACxC;AACA,IAAAd,IAAY,OAAO,WAAWc,GAAM,CAAC;AAAA,EACvC;AAEA,SAAOR;AACT;"}
package/dist/index.d.ts CHANGED
@@ -33,7 +33,7 @@ export interface NostrManagerConfig {
33
33
  * Pure TypeScript NostrClient that manages worker communication and state.
34
34
  * Uses WASM utilities for heavy lifting (encoding, decoding, crypto).
35
35
  */
36
- declare class NostrManager {
36
+ export declare class NostrManager {
37
37
  private worker;
38
38
  private subscriptions;
39
39
  private publishes;
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAEL,KAAK,cAAc,EACnB,KAAK,OAAO,EAEZ,KAAK,mBAAmB,EACxB,KAAK,gBAAgB,EACtB,MAAM,uBAAuB,CAAC;AAE/B,OAAO,KAAK,EAAE,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAGtD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAK/C,YAAY,EAAE,OAAO,EAAE,gBAAgB,EAAE,CAAC;AAI1C,MAAM,MAAM,oBAAoB,GAAG,CACjC,IAAI,EAAE,WAAW,CAAC,OAAO,CAAC,EAAE,GAAG,MAAM,EACrC,IAAI,EAAE,aAAa,KAChB,IAAI,CAAC;AAGV,MAAM,WAAW,mBAAmB;IAClC,QAAQ,CAAC,EAAE,cAAc,CAAC;IAC1B,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAC7B,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,cAAc,CAAC,EAAE,mBAAmB,CAAC;CACtC;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC;;OAEG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB;;OAEG;IACH,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAID;;;GAGG;AACH,cAAM,YAAY;IAChB,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,aAAa,CAOjB;IACJ,OAAO,CAAC,SAAS,CAAkD;IACnE,OAAO,CAAC,OAAO,CAA6B;IAErC,uBAAuB,WAAoC;gBAEtD,MAAM,GAAE,kBAAuB;IAK3C,OAAO,CAAC,YAAY;IAIpB,OAAO,CAAC,mBAAmB;IA2B3B,OAAO,CAAC,mBAAmB;IAe3B,OAAO,CAAC,iBAAiB;IAIzB,OAAO,CAAC,eAAe;IAIvB,OAAO,CAAC,aAAa;IAWrB,SAAS,CACP,cAAc,EAAE,MAAM,EACtB,QAAQ,EAAE,OAAO,EAAE,EACnB,OAAO,GAAE,mBAAwB,GAChC,iBAAiB;IA4FpB,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,iBAAiB,GAAG,SAAS;IASvD,WAAW,CAAC,cAAc,EAAE,MAAM,GAAG,IAAI;IAWzC,OAAO,CAAC,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,UAAU,GAAG,iBAAiB;IAkCjE,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,IAAI;IAanD,SAAS,CAAC,KAAK,EAAE,UAAU;IAgB3B,YAAY;IAQZ,OAAO,IAAI,IAAI;CA0BhB;AAED;;;;GAIG;AACH,wBAAgB,kBAAkB,CAChC,MAAM,GAAE,kBAAuB,GAC9B,YAAY,CAEd;AAED;;;GAGG;AACH,eAAO,MAAM,YAAY,cAAqB,CAAC;AAE/C,wBAAgB,OAAO,IAAI,IAAI,CAE9B;AAED,cAAc,SAAS,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,OAAO,EAEL,KAAK,cAAc,EACnB,KAAK,OAAO,EAEZ,KAAK,mBAAmB,EACxB,KAAK,gBAAgB,EACtB,MAAM,uBAAuB,CAAC;AAE/B,OAAO,KAAK,EAAE,OAAO,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAGtD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAK/C,YAAY,EAAE,OAAO,EAAE,gBAAgB,EAAE,CAAC;AAI1C,MAAM,MAAM,oBAAoB,GAAG,CACjC,IAAI,EAAE,WAAW,CAAC,OAAO,CAAC,EAAE,GAAG,MAAM,EACrC,IAAI,EAAE,aAAa,KAChB,IAAI,CAAC;AAGV,MAAM,WAAW,mBAAmB;IAClC,QAAQ,CAAC,EAAE,cAAc,CAAC;IAC1B,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAC7B,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,cAAc,CAAC,EAAE,mBAAmB,CAAC;CACtC;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC;;OAEG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB;;OAEG;IACH,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAID;;;GAGG;AACH,qBAAa,YAAY;IACvB,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,aAAa,CAOjB;IACJ,OAAO,CAAC,SAAS,CAAkD;IACnE,OAAO,CAAC,OAAO,CAA6B;IAErC,uBAAuB,WAAoC;gBAEtD,MAAM,GAAE,kBAAuB;IAK3C,OAAO,CAAC,YAAY;IAIpB,OAAO,CAAC,mBAAmB;IA2B3B,OAAO,CAAC,mBAAmB;IAe3B,OAAO,CAAC,iBAAiB;IAIzB,OAAO,CAAC,eAAe;IAIvB,OAAO,CAAC,aAAa;IAWrB,SAAS,CACP,cAAc,EAAE,MAAM,EACtB,QAAQ,EAAE,OAAO,EAAE,EACnB,OAAO,GAAE,mBAAwB,GAChC,iBAAiB;IA4FpB,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,iBAAiB,GAAG,SAAS;IASvD,WAAW,CAAC,cAAc,EAAE,MAAM,GAAG,IAAI;IAWzC,OAAO,CAAC,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,UAAU,GAAG,iBAAiB;IAkCjE,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,IAAI;IAanD,SAAS,CAAC,KAAK,EAAE,UAAU;IAgB3B,YAAY;IAQZ,OAAO,IAAI,IAAI;CA0BhB;AAED;;;;GAIG;AACH,wBAAgB,kBAAkB,CAChC,MAAM,GAAE,kBAAuB,GAC9B,YAAY,CAEd;AAED;;;GAGG;AACH,eAAO,MAAM,YAAY,cAAqB,CAAC;AAE/C,wBAAgB,OAAO,IAAI,IAAI,CAE9B;AAED,cAAc,SAAS,CAAC"}
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- import{R as a,a as s,c as r,i as o,n as e}from"./index3.js";import"@msgpack/msgpack";import"@candypoets/rust-worker/worker.js?worker";import{S as t}from"./index2.js";export{a as ReactionType,t as SignerTypes,s as cleanup,r as createNostrManager,o as isKind39089,e as nostrManager};
1
+ import{N as a,R as s,a as r,c as o,i as e,n as t}from"./index3.js";import"@msgpack/msgpack";import"@candypoets/rust-worker/worker.js?worker";import{S as i}from"./index2.js";export{a as NostrManager,s as ReactionType,i as SignerTypes,r as cleanup,o as createNostrManager,e as isKind39089,t as nostrManager};
package/dist/index3.js CHANGED
@@ -1 +1 @@
1
- import{encode as h,decode as k}from"@msgpack/msgpack";import{unpack as M}from"msgpackr";import S from"@candypoets/rust-worker/worker.js?worker";import"./index2.js";const v=Object.freeze({SubscriptionEvent:{subscription_id:"",event_type:"BUFFER_FULL",event_data:[]}});class p{static initializeBuffer(e){new DataView(e).setUint32(0,4,!0)}static writeMessage(e,t){const s=new DataView(e),r=new Uint8Array(e),o=s.getUint32(0,!0),n=4+t.length;if(o+n>e.byteLength)return console.warn("Buffer full, cannot write message"),!1;s.setUint32(o,t.length,!0),r.set(t,o+4);const i=o+n;return s.setUint32(0,i,!0),!0}static writeMessageObject(e,t){try{const s=h(t);return this.writeMessage(e,new Uint8Array(s))}catch(s){return console.error("Failed to encode message:",s),!1}}static readMessages(e,t=0){const s=new DataView(e),r=new Uint8Array(e),o=s.getUint32(0,!0),n=4;let i=t<n?n:t;if(o<=i)return{messages:[],newReadPosition:i,hasNewData:!1};const d=256,l=new Array(d);let a=0,f=new Uint8Array(65536);try{for(;i<o&&!(i+4>o);){const u=s.getUint32(i,!0);if(i+=4,u===1&&i<o&&r[i]===255){a<d&&(l[a++]=v),i+=1;continue}if(i+u>o)break;f.length<u&&(f=new Uint8Array(u)),f.set(r.subarray(i,i+u),0);const g=M(f.subarray(0,u));a<d&&(l[a++]=g),i+=u}return l.length=a,{messages:l,newReadPosition:i,hasNewData:a>0}}catch(u){return console.error("Failed to decode length-prefixed msgpack data from SharedArrayBuffer:",u),l.length=a,{messages:l,newReadPosition:t<n?n:t,hasNewData:!1}}}static readAllMessages(e){const t=this.readMessages(e,0);return{messages:t.messages,totalMessages:t.messages.length}}static getCurrentWritePosition(e){return new DataView(e).getUint32(0,!0)}static hasNewData(e,t){const s=this.getCurrentWritePosition(e),r=Math.max(t,4);return s>r}static calculateBufferSize(e=100,t=3072){const s=e*t,r=Math.floor(s*.25);return 4+s+r}}function E(c){return c.kind===39089}var b=(c=>(c.LIKE="+",c.DISLIKE="-",c.EMOJI="emoji",c.CUSTOM="custom",c))(b||{});class w{constructor(e={}){this.subscriptions=new Map,this.publishes=new Map,this.signers=new Map,this.PERPETUAL_SUBSCRIPTIONS=["notifications","starterpack"],this.worker=this.createWorker(e),this.setupWorkerListener()}createWorker(e){return new S}setupWorkerListener(){this.worker.onmessage=async e=>{if(e.data instanceof Uint8Array){let t=e.data;try{const s=k(t);this.handleWorkerMessage(s)}catch(s){console.error("Failed to decode worker message:",s)}finally{t&&(t.fill(0),t=null)}}else console.log("Received non-arrayBuffer message:",e.data)},this.worker.onerror=e=>{console.error("Worker error:",e)}}handleWorkerMessage(e){"Count"in e||("SignedEvent"in e?this.handleSignedEvent(e.SignedEvent.content,e.SignedEvent.signed_event):"PublicKey"in e?this.handlePublicKey(e.PublicKey.public_key):console.warn("Unknown message type from worker:",e))}handleSignedEvent(e,t){console.log("Signed event received:",e,t)}handlePublicKey(e){console.log("Public key received:",e)}createShortId(e){let t=0;for(let s=0;s<e.length;s++){const r=e.charCodeAt(s);t=(t<<5)-t+r,t=t&t}return Math.abs(t).toString(36).substring(0,63)}subscribe(e,t,s={}){const r=e.length<64?e:this.createShortId(e),o=this.subscriptions.get(r);if(o)return o.refCount++,o.buffer;const n={closeOnEose:!1,cacheFirst:!0,skipCache:!1,force:!1,enableOptimization:!0,...s},i=t.reduce((g,m)=>g+(m.limit||100),0),d=p.calculateBufferSize(i,s.bytesPerEvent);let l=new Uint8Array;s.initialMessage&&(l=h(s.initialMessage));const a=new SharedArrayBuffer(d+l.length);p.initializeBuffer(a),l.length>0&&(p.writeMessage(a,l)||console.error("Failed to write initial message to buffer")),this.subscriptions.set(r,{buffer:a,options:n,refCount:1});const f={pipeline:n.pipeline,closeOnEose:n.closeOnEose,cacheFirst:n.cacheFirst,timeoutMs:n.timeoutMs,maxEvents:n.maxEvents,enableOptimization:n.enableOptimization,skipCache:n.skipCache,force:n.force,bytesPerEvent:n.bytesPerEvent},u={Subscribe:{subscription_id:r,requests:t,config:f}};try{const g=h(u);return this.worker.postMessage({serializedMessage:g,sharedBuffer:a}),a}catch(g){throw this.subscriptions.delete(r),g}}getBuffer(e){const t=this.subscriptions.get(e);if(t)return t.refCount++,t.buffer}unsubscribe(e){const t=e.length<64?e:this.createShortId(e),s=this.subscriptions.get(t);s&&s.refCount--}publish(e,t){const s=new SharedArrayBuffer(3072);p.initializeBuffer(s);try{const r={kind:t.kind,content:t.content,tags:t.tags||[]},o=h({Publish:{publish_id:e,template:r}});return this.worker.postMessage({serializedMessage:o,sharedBuffer:s}),this.publishes.set(e,{buffer:s}),s}catch(r){throw console.error("Failed to publish event:",r),r}}setSigner(e,t){const s=h({SetSigner:{signer_type:e,private_key:t}});this.worker.postMessage(s),this.signers.set(e,t)}signEvent(e){const t={SignEvent:{template:{kind:e.kind,content:e.content,tags:e.tags}}},s=h(t);this.worker.postMessage(s)}getPublicKey(){const e=h({GetPublicKey:{}});this.worker.postMessage(e)}cleanup(){const e=[];for(const[t,s]of this.subscriptions.entries())s.refCount<=0&&!this.PERPETUAL_SUBSCRIPTIONS.includes(t)&&e.push(t);for(const t of e)if(this.subscriptions.get(t)){const s=h({Unsubscribe:{subscription_id:t}});this.worker.postMessage(s),this.subscriptions.delete(t)}}}function U(c={}){return new w(c)}const y=new w;function P(){y.cleanup()}export{b as R,p as S,P as a,U as c,E as i,y as n};
1
+ import{encode as h,decode as k}from"@msgpack/msgpack";import{unpack as M}from"msgpackr";import S from"@candypoets/rust-worker/worker.js?worker";import"./index2.js";const v=Object.freeze({SubscriptionEvent:{subscription_id:"",event_type:"BUFFER_FULL",event_data:[]}});class p{static initializeBuffer(e){new DataView(e).setUint32(0,4,!0)}static writeMessage(e,t){const s=new DataView(e),r=new Uint8Array(e),o=s.getUint32(0,!0),n=4+t.length;if(o+n>e.byteLength)return console.warn("Buffer full, cannot write message"),!1;s.setUint32(o,t.length,!0),r.set(t,o+4);const i=o+n;return s.setUint32(0,i,!0),!0}static writeMessageObject(e,t){try{const s=h(t);return this.writeMessage(e,new Uint8Array(s))}catch(s){return console.error("Failed to encode message:",s),!1}}static readMessages(e,t=0){const s=new DataView(e),r=new Uint8Array(e),o=s.getUint32(0,!0),n=4;let i=t<n?n:t;if(o<=i)return{messages:[],newReadPosition:i,hasNewData:!1};const d=256,l=new Array(d);let a=0,f=new Uint8Array(65536);try{for(;i<o&&!(i+4>o);){const u=s.getUint32(i,!0);if(i+=4,u===1&&i<o&&r[i]===255){a<d&&(l[a++]=v),i+=1;continue}if(i+u>o)break;f.length<u&&(f=new Uint8Array(u)),f.set(r.subarray(i,i+u),0);const g=M(f.subarray(0,u));a<d&&(l[a++]=g),i+=u}return l.length=a,{messages:l,newReadPosition:i,hasNewData:a>0}}catch(u){return console.error("Failed to decode length-prefixed msgpack data from SharedArrayBuffer:",u),l.length=a,{messages:l,newReadPosition:t<n?n:t,hasNewData:!1}}}static readAllMessages(e){const t=this.readMessages(e,0);return{messages:t.messages,totalMessages:t.messages.length}}static getCurrentWritePosition(e){return new DataView(e).getUint32(0,!0)}static hasNewData(e,t){const s=this.getCurrentWritePosition(e),r=Math.max(t,4);return s>r}static calculateBufferSize(e=100,t=3072){const s=e*t,r=Math.floor(s*.25);return 4+s+r}}function E(c){return c.kind===39089}var w=(c=>(c.LIKE="+",c.DISLIKE="-",c.EMOJI="emoji",c.CUSTOM="custom",c))(w||{});class b{constructor(e={}){this.subscriptions=new Map,this.publishes=new Map,this.signers=new Map,this.PERPETUAL_SUBSCRIPTIONS=["notifications","starterpack"],this.worker=this.createWorker(e),this.setupWorkerListener()}createWorker(e){return new S}setupWorkerListener(){this.worker.onmessage=async e=>{if(e.data instanceof Uint8Array){let t=e.data;try{const s=k(t);this.handleWorkerMessage(s)}catch(s){console.error("Failed to decode worker message:",s)}finally{t&&(t.fill(0),t=null)}}else console.log("Received non-arrayBuffer message:",e.data)},this.worker.onerror=e=>{console.error("Worker error:",e)}}handleWorkerMessage(e){"Count"in e||("SignedEvent"in e?this.handleSignedEvent(e.SignedEvent.content,e.SignedEvent.signed_event):"PublicKey"in e?this.handlePublicKey(e.PublicKey.public_key):console.warn("Unknown message type from worker:",e))}handleSignedEvent(e,t){console.log("Signed event received:",e,t)}handlePublicKey(e){console.log("Public key received:",e)}createShortId(e){let t=0;for(let s=0;s<e.length;s++){const r=e.charCodeAt(s);t=(t<<5)-t+r,t=t&t}return Math.abs(t).toString(36).substring(0,63)}subscribe(e,t,s={}){const r=e.length<64?e:this.createShortId(e),o=this.subscriptions.get(r);if(o)return o.refCount++,o.buffer;const n={closeOnEose:!1,cacheFirst:!0,skipCache:!1,force:!1,enableOptimization:!0,...s},i=t.reduce((g,m)=>g+(m.limit||100),0),d=p.calculateBufferSize(i,s.bytesPerEvent);let l=new Uint8Array;s.initialMessage&&(l=h(s.initialMessage));const a=new SharedArrayBuffer(d+l.length);p.initializeBuffer(a),l.length>0&&(p.writeMessage(a,l)||console.error("Failed to write initial message to buffer")),this.subscriptions.set(r,{buffer:a,options:n,refCount:1});const f={pipeline:n.pipeline,closeOnEose:n.closeOnEose,cacheFirst:n.cacheFirst,timeoutMs:n.timeoutMs,maxEvents:n.maxEvents,enableOptimization:n.enableOptimization,skipCache:n.skipCache,force:n.force,bytesPerEvent:n.bytesPerEvent},u={Subscribe:{subscription_id:r,requests:t,config:f}};try{const g=h(u);return this.worker.postMessage({serializedMessage:g,sharedBuffer:a}),a}catch(g){throw this.subscriptions.delete(r),g}}getBuffer(e){const t=this.subscriptions.get(e);if(t)return t.refCount++,t.buffer}unsubscribe(e){const t=e.length<64?e:this.createShortId(e),s=this.subscriptions.get(t);s&&s.refCount--}publish(e,t){const s=new SharedArrayBuffer(3072);p.initializeBuffer(s);try{const r={kind:t.kind,content:t.content,tags:t.tags||[]},o=h({Publish:{publish_id:e,template:r}});return this.worker.postMessage({serializedMessage:o,sharedBuffer:s}),this.publishes.set(e,{buffer:s}),s}catch(r){throw console.error("Failed to publish event:",r),r}}setSigner(e,t){const s=h({SetSigner:{signer_type:e,private_key:t}});this.worker.postMessage(s),this.signers.set(e,t)}signEvent(e){const t={SignEvent:{template:{kind:e.kind,content:e.content,tags:e.tags}}},s=h(t);this.worker.postMessage(s)}getPublicKey(){const e=h({GetPublicKey:{}});this.worker.postMessage(e)}cleanup(){const e=[];for(const[t,s]of this.subscriptions.entries())s.refCount<=0&&!this.PERPETUAL_SUBSCRIPTIONS.includes(t)&&e.push(t);for(const t of e)if(this.subscriptions.get(t)){const s=h({Unsubscribe:{subscription_id:t}});this.worker.postMessage(s),this.subscriptions.delete(t)}}}function U(c={}){return new b(c)}const y=new b;function P(){y.cleanup()}export{b as N,w as R,p as S,P as a,U as c,E as i,y as n};
@@ -1 +1 @@
1
- {"version":3,"file":"index3.js","sources":["../src/lib/sharedBuffer.ts","../src/types/kind39089.ts","../src/types/kind7.ts","../src/index.ts"],"sourcesContent":["import { encode } from \"@msgpack/msgpack\";\nimport { unpack } from \"msgpackr\";\nimport type { WorkerToMainMessage } from \"src/types\";\n\nconst WorkerToMainMessageBufferFull = Object.freeze({\n SubscriptionEvent: {\n subscription_id: \"\",\n event_type: \"BUFFER_FULL\",\n event_data: []\n }\n});\n\n/**\n * Utility library for reading from SharedArrayBuffer with 4-byte header approach\n * Header format: [0-3]: Write position (4 bytes, little endian)\n * Data format: [4+]: [4-byte length][msgpack event][4-byte length][msgpack event]...\n */\nexport class SharedBufferReader {\n /**\n * Initialize a buffer for writing - sets the write position header to 4\n * @param buffer The SharedArrayBuffer to initialize\n */\n static initializeBuffer(buffer: SharedArrayBuffer): void {\n const view = new DataView(buffer);\n // Set initial write position to 4 (right after the header)\n view.setUint32(0, 4, true);\n }\n\n /**\n * Write a message to the SharedArrayBuffer\n * @param buffer The SharedArrayBuffer to write to\n * @param data The data to write (already encoded as Uint8Array)\n * @returns True if written successfully, false if buffer is full\n */\n static writeMessage(buffer: SharedArrayBuffer, data: Uint8Array): boolean {\n const view = new DataView(buffer);\n const uint8View = new Uint8Array(buffer);\n\n // Get current write position\n const currentWritePosition = view.getUint32(0, true);\n\n // Check if there's enough space (4 bytes for length + data length)\n const requiredSpace = 4 + data.length;\n if (currentWritePosition + requiredSpace > buffer.byteLength) {\n console.warn(\"Buffer full, cannot write message\");\n return false;\n }\n\n // Write the length prefix (4 bytes, little endian)\n view.setUint32(currentWritePosition, data.length, true);\n\n // Write the actual data\n uint8View.set(data, currentWritePosition + 4);\n\n // Update the write position header\n const newWritePosition = currentWritePosition + requiredSpace;\n view.setUint32(0, newWritePosition, true);\n\n return true;\n }\n\n /**\n * Write a message object to the SharedArrayBuffer (handles encoding)\n * @param buffer The SharedArrayBuffer to write to\n * @param message The message object to write\n * @returns True if written successfully, false if buffer is full\n */\n static writeMessageObject(buffer: SharedArrayBuffer, message: any): boolean {\n try {\n const encoded = encode(message);\n return this.writeMessage(buffer, new Uint8Array(encoded));\n } catch (error) {\n console.error(\"Failed to encode message:\", error);\n return false;\n }\n }\n\n\n /**\n * Read new messages from SharedArrayBuffer since last read position\n * @param buffer The SharedArrayBuffer to read from\n * @param lastReadPosition Last position read (default: 0, meaning read from beginning)\n * @returns Object containing new messages and updated read position\n */\n static readMessages(buffer: SharedArrayBuffer, lastReadPosition: number = 0) {\n const view = new DataView(buffer);\n const uint8View = new Uint8Array(buffer);\n\n const currentWritePosition = view.getUint32(0, true);\n\n const dataStartOffset = 4;\n let currentPos = lastReadPosition < dataStartOffset\n ? dataStartOffset\n : lastReadPosition;\n\n if (currentWritePosition <= currentPos) {\n return { messages: [], newReadPosition: currentPos, hasNewData: false };\n }\n\n const maxMessages = 256;\n const messages = new Array(maxMessages);\n let msgCount = 0;\n\n // Reusable decoding buffer (non-shared!)\n let scratch = new Uint8Array(65536); // initial size — grows if needed\n\n try {\n while (currentPos < currentWritePosition) {\n if (currentPos + 4 > currentWritePosition) break;\n\n const eventLength = view.getUint32(currentPos, true);\n currentPos += 4;\n\n if (eventLength === 1) {\n if (currentPos < currentWritePosition && uint8View[currentPos] === 0xFF) {\n if (msgCount < maxMessages) {\n messages[msgCount++] = WorkerToMainMessageBufferFull;\n }\n currentPos += 1;\n continue;\n }\n }\n\n if (currentPos + eventLength > currentWritePosition) break;\n\n // Ensure scratch buffer is large enough\n if (scratch.length < eventLength) {\n scratch = new Uint8Array(eventLength);\n }\n\n // Copy into non-shared buffer\n scratch.set(\n uint8View.subarray(currentPos, currentPos + eventLength),\n 0\n );\n\n const message = unpack(scratch.subarray(0, eventLength));\n if (msgCount < maxMessages) {\n messages[msgCount++] = message;\n }\n\n currentPos += eventLength;\n }\n\n messages.length = msgCount;\n return {\n messages,\n newReadPosition: currentPos,\n hasNewData: msgCount > 0,\n };\n\n } catch (error) {\n console.error('Failed to decode length-prefixed msgpack data from SharedArrayBuffer:', error);\n messages.length = msgCount;\n return {\n messages,\n newReadPosition: lastReadPosition < dataStartOffset ? dataStartOffset : lastReadPosition,\n hasNewData: false,\n };\n }\n }\n\n\n\n /**\n * Read all messages from SharedArrayBuffer from the beginning (ignores lastReadPosition)\n * @param buffer The SharedArrayBuffer to read from\n * @returns Object containing all messages in the buffer\n */\n static readAllMessages(buffer: SharedArrayBuffer): {\n messages: WorkerToMainMessage[];\n totalMessages: number;\n } {\n const result = this.readMessages(buffer, 0); // Always start from beginning\n return {\n messages: result.messages,\n totalMessages: result.messages.length,\n };\n }\n\n /**\n * Get current write position from buffer header\n * @param buffer The SharedArrayBuffer to read from\n * @returns Current write position\n */\n static getCurrentWritePosition(buffer: SharedArrayBuffer): number {\n const view = new DataView(buffer);\n return view.getUint32(0, true);\n }\n\n /**\n * Check if buffer has new data since last read\n * @param buffer The SharedArrayBuffer to check\n * @param lastReadPosition Last position read\n * @returns True if there's new data to read\n */\n static hasNewData(\n buffer: SharedArrayBuffer,\n lastReadPosition: number,\n ): boolean {\n const currentWritePosition = this.getCurrentWritePosition(buffer);\n const dataStartOffset = 4;\n const actualLastReadPosition = Math.max(lastReadPosition, dataStartOffset);\n return currentWritePosition > actualLastReadPosition;\n }\n\n /**\n * Calculate recommended buffer size based on request limits\n * @param totalEventLimit Total expected events across all requests\n * @param bytesPerEvent Estimated bytes per event (default: 3072)\n * @returns Recommended buffer size in bytes\n */\n static calculateBufferSize(\n totalEventLimit: number = 100,\n bytesPerEvent: number = 3072,\n ): number {\n const headerSize = 4;\n const dataSize = totalEventLimit * bytesPerEvent;\n const overhead = Math.floor(dataSize * 0.25); // 25% overhead\n return headerSize + dataSize + overhead;\n }\n}\n","import type { ParsedEvent } from 'src/types';\n\nexport interface Kind39089Parsed {\n\tlist_identifier: string;\n\tpeople: string[];\n\ttitle?: string;\n\tdescription?: string;\n\timage?: string;\n}\n\nexport function isKind39089(event: ParsedEvent<unknown>): event is ParsedEvent<Kind39089Parsed> {\n\treturn event.kind === 39089;\n}\n","export enum ReactionType {\n\tLIKE = '+',\n\tDISLIKE = '-',\n\tEMOJI = 'emoji',\n\tCUSTOM = 'custom'\n}\n\nexport type Kind7Parsed = {\n\ttype: ReactionType;\n\teventId: string; // The id of the event being reacted to\n\tpubkey: string; // The pubkey of the author of the reacted event\n\teventKind?: number; // The kind of the event being reacted to (from k tag)\n\temoji?: {\n\t\tshortcode: string;\n\t\turl: string;\n\t};\n\ttargetCoordinates?: string; // For addressable events (from a tag)\n};\n","import { SharedBufferReader } from \"src/lib/sharedBuffer\";\nimport {\n type MainToWorkerMessage,\n type PipelineConfig,\n type Request,\n type SubscriptionConfig,\n type WorkerToMainMessage,\n type ConnectionStatus\n} from \"@candypoets/rust-main\";\n\nimport type { AnyKind, ParsedEvent } from \"src/types\";\n\nimport { decode, encode } from \"@msgpack/msgpack\";\nimport type { NostrEvent } from \"nostr-tools\";\nimport type { SubscribeKind } from \"src/types\";\n\nimport RustWorker from \"@candypoets/rust-worker/worker.js?worker\";\n\n// Re-export types for external use\nexport type { Request, ConnectionStatus };\n\n\n// Callback for subscription events\nexport type SubscriptionCallback = (\n data: ParsedEvent<AnyKind>[] | number,\n type: SubscribeKind,\n) => void;\n\n\nexport interface SubscriptionOptions {\n pipeline?: PipelineConfig;\n closeOnEose?: boolean;\n cacheFirst?: boolean;\n timeoutMs?: number;\n maxEvents?: number;\n enableOptimization?: boolean;\n skipCache?: boolean;\n force?: boolean;\n bytesPerEvent?: number;\n initialMessage?: WorkerToMainMessage;\n}\n\n/**\n * Configuration for the Nostr Manager\n */\nexport interface NostrManagerConfig {\n /**\n * Custom worker URL. If not provided, uses the bundled worker.\n */\n workerUrl?: string;\n /**\n * Custom worker instance. If provided, workerUrl is ignored.\n */\n worker?: Worker;\n}\n\n// const wasmReady = init(mainWasmUrl);\n\n/**\n * Pure TypeScript NostrClient that manages worker communication and state.\n * Uses WASM utilities for heavy lifting (encoding, decoding, crypto).\n */\nclass NostrManager {\n private worker: Worker;\n private subscriptions = new Map<\n string,\n {\n buffer: SharedArrayBuffer;\n options: SubscriptionOptions;\n refCount: number;\n }\n >();\n private publishes = new Map<string, {buffer: SharedArrayBuffer}>();\n private signers = new Map<string, string>(); // name -> secret key hex\n\n public PERPETUAL_SUBSCRIPTIONS = [\"notifications\", \"starterpack\"];\n\n constructor(config: NostrManagerConfig = {}) {\n this.worker = this.createWorker(config);\n this.setupWorkerListener();\n }\n\n private createWorker(config: NostrManagerConfig): Worker {\n return new RustWorker();\n }\n\n private setupWorkerListener() {\n this.worker.onmessage = async (event) => {\n // await wasmReady;\n if (event.data instanceof Uint8Array) {\n let uint8Array = event.data;\n try {\n const message: any = decode(uint8Array);\n this.handleWorkerMessage(message);\n } catch (error) {\n console.error(\"Failed to decode worker message:\", error);\n } finally {\n // Aggressively clear memory references\n if (uint8Array) {\n uint8Array.fill(0);\n (uint8Array as any) = null;\n }\n }\n } else {\n console.log(\"Received non-arrayBuffer message:\", event.data);\n }\n };\n\n this.worker.onerror = (error) => {\n console.error(\"Worker error:\", error);\n };\n }\n\n private handleWorkerMessage(message: WorkerToMainMessage) {\n if (\"Count\" in message) {\n // this.handleSubscriptionCount(message.Count.subscription_id, message.Count.count);\n } else if (\"SignedEvent\" in message) {\n this.handleSignedEvent(\n message.SignedEvent.content,\n message.SignedEvent.signed_event,\n );\n } else if (\"PublicKey\" in message) {\n this.handlePublicKey(message.PublicKey.public_key);\n } else {\n console.warn(\"Unknown message type from worker:\", message);\n }\n }\n\n private handleSignedEvent(content: string, signedEvent: any) {\n console.log(\"Signed event received:\", content, signedEvent);\n }\n\n private handlePublicKey(publicKey: string) {\n console.log(\"Public key received:\", publicKey);\n }\n\n private createShortId(input: string): string {\n let hash = 0;\n for (let i = 0; i < input.length; i++) {\n const char = input.charCodeAt(i);\n hash = (hash << 5) - hash + char;\n hash = hash & hash;\n }\n const shortId = Math.abs(hash).toString(36);\n return shortId.substring(0, 63);\n }\n\n subscribe(\n subscriptionId: string,\n requests: Request[],\n options: SubscriptionOptions = {},\n ): SharedArrayBuffer {\n const subId =\n subscriptionId.length < 64\n ? subscriptionId\n : this.createShortId(subscriptionId);\n\n const existingSubscription = this.subscriptions.get(subId);\n\n if (existingSubscription) {\n existingSubscription.refCount++;\n return existingSubscription.buffer;\n }\n\n const defaultOptions: SubscriptionOptions = {\n closeOnEose: false,\n cacheFirst: true,\n skipCache: false,\n force: false,\n enableOptimization: true,\n ...options,\n };\n\n const totalLimit = requests.reduce(\n (sum, req) => sum + (req.limit || 100),\n 0,\n );\n\n const bufferSize = SharedBufferReader.calculateBufferSize(\n totalLimit,\n options.bytesPerEvent,\n );\n\n let initialMessage: Uint8Array<ArrayBufferLike> = new Uint8Array();\n\n if(options.initialMessage) {\n initialMessage = encode(options.initialMessage);\n }\n\n const buffer = new SharedArrayBuffer(bufferSize + initialMessage.length);\n\n // Initialize the buffer (sets write position to 4)\n SharedBufferReader.initializeBuffer(buffer);\n\n // Write the initial message if provided\n if(initialMessage.length > 0) {\n const success = SharedBufferReader.writeMessage(buffer, initialMessage);\n if (!success) {\n console.error(\"Failed to write initial message to buffer\");\n }\n }\n\n this.subscriptions.set(subId, {\n buffer,\n options: defaultOptions,\n refCount: 1,\n });\n\n // Convert SubscriptionOptions to SubscriptionConfig for the worker\n const config: SubscriptionConfig = {\n pipeline: defaultOptions.pipeline,\n closeOnEose: defaultOptions.closeOnEose,\n cacheFirst: defaultOptions.cacheFirst,\n timeoutMs: defaultOptions.timeoutMs,\n maxEvents: defaultOptions.maxEvents,\n enableOptimization: defaultOptions.enableOptimization,\n skipCache: defaultOptions.skipCache,\n force: defaultOptions.force,\n bytesPerEvent: defaultOptions.bytesPerEvent,\n };\n\n const message: MainToWorkerMessage = {\n Subscribe: {\n subscription_id: subId,\n requests: requests,\n config: config,\n },\n };\n\n try {\n const pack = encode(message);\n this.worker.postMessage({\n serializedMessage: pack,\n sharedBuffer: buffer,\n });\n\n return buffer;\n } catch (error) {\n this.subscriptions.delete(subId);\n throw error;\n }\n }\n\n getBuffer(subId: string): SharedArrayBuffer | undefined {\n const existingSubscription = this.subscriptions.get(subId);\n if (existingSubscription) {\n existingSubscription.refCount++;\n return existingSubscription.buffer;\n }\n return undefined;\n }\n\n unsubscribe(subscriptionId: string): void {\n const subId =\n subscriptionId.length < 64\n ? subscriptionId\n : this.createShortId(subscriptionId);\n const subscription = this.subscriptions.get(subId);\n if (subscription) {\n subscription.refCount--;\n }\n }\n\n publish(publish_id: string, event: NostrEvent): SharedArrayBuffer {\n\n // a publish buffer fit in 3kb\n const buffer = new SharedArrayBuffer(3072);\n\n // Initialize the buffer (sets write position to 4)\n SharedBufferReader.initializeBuffer(buffer);\n\n try {\n const template = {\n kind: event.kind,\n content: event.content,\n tags: event.tags || [],\n };\n\n const message: MainToWorkerMessage = {\n Publish: {\n publish_id: publish_id,\n template,\n },\n };\n\n const p = encode(message);\n\n this.worker.postMessage({ serializedMessage: p, sharedBuffer: buffer });\n\n this.publishes.set(publish_id, {buffer});\n return buffer;\n } catch (error) {\n console.error(\"Failed to publish event:\", error);\n throw error;\n }\n }\n\n setSigner(name: string, secretKeyHex: string): void {\n const message: MainToWorkerMessage = {\n SetSigner: {\n signer_type: name,\n private_key: secretKeyHex,\n },\n };\n\n const pack = encode(message);\n this.worker.postMessage(pack);\n this.signers.set(name, secretKeyHex);\n }\n\n signEvent(event: NostrEvent) {\n const template = {\n kind: event.kind,\n content: event.content,\n tags: event.tags,\n };\n\n const message: MainToWorkerMessage = {\n SignEvent: {\n template: template,\n },\n };\n const pack = encode(message);\n this.worker.postMessage(pack);\n }\n\n getPublicKey() {\n const message: MainToWorkerMessage = {\n GetPublicKey: {},\n };\n const pack = encode(message);\n this.worker.postMessage(pack);\n }\n\n cleanup(): void {\n const subscriptionsToDelete: string[] = [];\n\n for (const [subId, subscription] of this.subscriptions.entries()) {\n if (\n subscription.refCount <= 0 &&\n !this.PERPETUAL_SUBSCRIPTIONS.includes(subId)\n ) {\n subscriptionsToDelete.push(subId);\n }\n }\n\n for (const subId of subscriptionsToDelete) {\n const subscription = this.subscriptions.get(subId);\n if (subscription) {\n const message: MainToWorkerMessage = {\n Unsubscribe: {\n subscription_id: subId,\n },\n };\n const pack = encode(message);\n this.worker.postMessage(pack);\n this.subscriptions.delete(subId);\n }\n }\n }\n}\n\n/**\n * Factory function to create a new NostrManager instance.\n * @param config - Configuration for the NostrManager.\n * @returns A new instance of NostrManager.\n */\nexport function createNostrManager(\n config: NostrManagerConfig = {},\n): NostrManager {\n return new NostrManager(config);\n}\n\n/**\n * Default singleton instance of the NostrManager.\n * Useful for applications that only need one instance.\n */\nexport const nostrManager = new NostrManager();\n\nexport function cleanup(): void {\n nostrManager.cleanup();\n}\n\nexport * from \"./types\";\n"],"names":["WorkerToMainMessageBufferFull","SharedBufferReader","buffer","data","view","uint8View","currentWritePosition","requiredSpace","newWritePosition","message","encoded","encode","error","lastReadPosition","dataStartOffset","currentPos","maxMessages","messages","msgCount","scratch","eventLength","unpack","result","actualLastReadPosition","totalEventLimit","bytesPerEvent","dataSize","overhead","isKind39089","event","ReactionType","NostrManager","config","RustWorker","uint8Array","decode","content","signedEvent","publicKey","input","hash","i","char","subscriptionId","requests","options","subId","existingSubscription","defaultOptions","totalLimit","sum","req","bufferSize","initialMessage","pack","subscription","publish_id","template","p","name","secretKeyHex","subscriptionsToDelete","createNostrManager","nostrManager","cleanup"],"mappings":"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"}
1
+ {"version":3,"file":"index3.js","sources":["../src/lib/sharedBuffer.ts","../src/types/kind39089.ts","../src/types/kind7.ts","../src/index.ts"],"sourcesContent":["import { encode } from \"@msgpack/msgpack\";\nimport { unpack } from \"msgpackr\";\nimport type { WorkerToMainMessage } from \"src/types\";\n\nconst WorkerToMainMessageBufferFull = Object.freeze({\n SubscriptionEvent: {\n subscription_id: \"\",\n event_type: \"BUFFER_FULL\",\n event_data: []\n }\n});\n\n/**\n * Utility library for reading from SharedArrayBuffer with 4-byte header approach\n * Header format: [0-3]: Write position (4 bytes, little endian)\n * Data format: [4+]: [4-byte length][msgpack event][4-byte length][msgpack event]...\n */\nexport class SharedBufferReader {\n /**\n * Initialize a buffer for writing - sets the write position header to 4\n * @param buffer The SharedArrayBuffer to initialize\n */\n static initializeBuffer(buffer: SharedArrayBuffer): void {\n const view = new DataView(buffer);\n // Set initial write position to 4 (right after the header)\n view.setUint32(0, 4, true);\n }\n\n /**\n * Write a message to the SharedArrayBuffer\n * @param buffer The SharedArrayBuffer to write to\n * @param data The data to write (already encoded as Uint8Array)\n * @returns True if written successfully, false if buffer is full\n */\n static writeMessage(buffer: SharedArrayBuffer, data: Uint8Array): boolean {\n const view = new DataView(buffer);\n const uint8View = new Uint8Array(buffer);\n\n // Get current write position\n const currentWritePosition = view.getUint32(0, true);\n\n // Check if there's enough space (4 bytes for length + data length)\n const requiredSpace = 4 + data.length;\n if (currentWritePosition + requiredSpace > buffer.byteLength) {\n console.warn(\"Buffer full, cannot write message\");\n return false;\n }\n\n // Write the length prefix (4 bytes, little endian)\n view.setUint32(currentWritePosition, data.length, true);\n\n // Write the actual data\n uint8View.set(data, currentWritePosition + 4);\n\n // Update the write position header\n const newWritePosition = currentWritePosition + requiredSpace;\n view.setUint32(0, newWritePosition, true);\n\n return true;\n }\n\n /**\n * Write a message object to the SharedArrayBuffer (handles encoding)\n * @param buffer The SharedArrayBuffer to write to\n * @param message The message object to write\n * @returns True if written successfully, false if buffer is full\n */\n static writeMessageObject(buffer: SharedArrayBuffer, message: any): boolean {\n try {\n const encoded = encode(message);\n return this.writeMessage(buffer, new Uint8Array(encoded));\n } catch (error) {\n console.error(\"Failed to encode message:\", error);\n return false;\n }\n }\n\n\n /**\n * Read new messages from SharedArrayBuffer since last read position\n * @param buffer The SharedArrayBuffer to read from\n * @param lastReadPosition Last position read (default: 0, meaning read from beginning)\n * @returns Object containing new messages and updated read position\n */\n static readMessages(buffer: SharedArrayBuffer, lastReadPosition: number = 0) {\n const view = new DataView(buffer);\n const uint8View = new Uint8Array(buffer);\n\n const currentWritePosition = view.getUint32(0, true);\n\n const dataStartOffset = 4;\n let currentPos = lastReadPosition < dataStartOffset\n ? dataStartOffset\n : lastReadPosition;\n\n if (currentWritePosition <= currentPos) {\n return { messages: [], newReadPosition: currentPos, hasNewData: false };\n }\n\n const maxMessages = 256;\n const messages = new Array(maxMessages);\n let msgCount = 0;\n\n // Reusable decoding buffer (non-shared!)\n let scratch = new Uint8Array(65536); // initial size — grows if needed\n\n try {\n while (currentPos < currentWritePosition) {\n if (currentPos + 4 > currentWritePosition) break;\n\n const eventLength = view.getUint32(currentPos, true);\n currentPos += 4;\n\n if (eventLength === 1) {\n if (currentPos < currentWritePosition && uint8View[currentPos] === 0xFF) {\n if (msgCount < maxMessages) {\n messages[msgCount++] = WorkerToMainMessageBufferFull;\n }\n currentPos += 1;\n continue;\n }\n }\n\n if (currentPos + eventLength > currentWritePosition) break;\n\n // Ensure scratch buffer is large enough\n if (scratch.length < eventLength) {\n scratch = new Uint8Array(eventLength);\n }\n\n // Copy into non-shared buffer\n scratch.set(\n uint8View.subarray(currentPos, currentPos + eventLength),\n 0\n );\n\n const message = unpack(scratch.subarray(0, eventLength));\n if (msgCount < maxMessages) {\n messages[msgCount++] = message;\n }\n\n currentPos += eventLength;\n }\n\n messages.length = msgCount;\n return {\n messages,\n newReadPosition: currentPos,\n hasNewData: msgCount > 0,\n };\n\n } catch (error) {\n console.error('Failed to decode length-prefixed msgpack data from SharedArrayBuffer:', error);\n messages.length = msgCount;\n return {\n messages,\n newReadPosition: lastReadPosition < dataStartOffset ? dataStartOffset : lastReadPosition,\n hasNewData: false,\n };\n }\n }\n\n\n\n /**\n * Read all messages from SharedArrayBuffer from the beginning (ignores lastReadPosition)\n * @param buffer The SharedArrayBuffer to read from\n * @returns Object containing all messages in the buffer\n */\n static readAllMessages(buffer: SharedArrayBuffer): {\n messages: WorkerToMainMessage[];\n totalMessages: number;\n } {\n const result = this.readMessages(buffer, 0); // Always start from beginning\n return {\n messages: result.messages,\n totalMessages: result.messages.length,\n };\n }\n\n /**\n * Get current write position from buffer header\n * @param buffer The SharedArrayBuffer to read from\n * @returns Current write position\n */\n static getCurrentWritePosition(buffer: SharedArrayBuffer): number {\n const view = new DataView(buffer);\n return view.getUint32(0, true);\n }\n\n /**\n * Check if buffer has new data since last read\n * @param buffer The SharedArrayBuffer to check\n * @param lastReadPosition Last position read\n * @returns True if there's new data to read\n */\n static hasNewData(\n buffer: SharedArrayBuffer,\n lastReadPosition: number,\n ): boolean {\n const currentWritePosition = this.getCurrentWritePosition(buffer);\n const dataStartOffset = 4;\n const actualLastReadPosition = Math.max(lastReadPosition, dataStartOffset);\n return currentWritePosition > actualLastReadPosition;\n }\n\n /**\n * Calculate recommended buffer size based on request limits\n * @param totalEventLimit Total expected events across all requests\n * @param bytesPerEvent Estimated bytes per event (default: 3072)\n * @returns Recommended buffer size in bytes\n */\n static calculateBufferSize(\n totalEventLimit: number = 100,\n bytesPerEvent: number = 3072,\n ): number {\n const headerSize = 4;\n const dataSize = totalEventLimit * bytesPerEvent;\n const overhead = Math.floor(dataSize * 0.25); // 25% overhead\n return headerSize + dataSize + overhead;\n }\n}\n","import type { ParsedEvent } from 'src/types';\n\nexport interface Kind39089Parsed {\n\tlist_identifier: string;\n\tpeople: string[];\n\ttitle?: string;\n\tdescription?: string;\n\timage?: string;\n}\n\nexport function isKind39089(event: ParsedEvent<unknown>): event is ParsedEvent<Kind39089Parsed> {\n\treturn event.kind === 39089;\n}\n","export enum ReactionType {\n\tLIKE = '+',\n\tDISLIKE = '-',\n\tEMOJI = 'emoji',\n\tCUSTOM = 'custom'\n}\n\nexport type Kind7Parsed = {\n\ttype: ReactionType;\n\teventId: string; // The id of the event being reacted to\n\tpubkey: string; // The pubkey of the author of the reacted event\n\teventKind?: number; // The kind of the event being reacted to (from k tag)\n\temoji?: {\n\t\tshortcode: string;\n\t\turl: string;\n\t};\n\ttargetCoordinates?: string; // For addressable events (from a tag)\n};\n","import { SharedBufferReader } from \"src/lib/sharedBuffer\";\nimport {\n type MainToWorkerMessage,\n type PipelineConfig,\n type Request,\n type SubscriptionConfig,\n type WorkerToMainMessage,\n type ConnectionStatus\n} from \"@candypoets/rust-main\";\n\nimport type { AnyKind, ParsedEvent } from \"src/types\";\n\nimport { decode, encode } from \"@msgpack/msgpack\";\nimport type { NostrEvent } from \"nostr-tools\";\nimport type { SubscribeKind } from \"src/types\";\n\nimport RustWorker from \"@candypoets/rust-worker/worker.js?worker\";\n\n// Re-export types for external use\nexport type { Request, ConnectionStatus };\n\n\n// Callback for subscription events\nexport type SubscriptionCallback = (\n data: ParsedEvent<AnyKind>[] | number,\n type: SubscribeKind,\n) => void;\n\n\nexport interface SubscriptionOptions {\n pipeline?: PipelineConfig;\n closeOnEose?: boolean;\n cacheFirst?: boolean;\n timeoutMs?: number;\n maxEvents?: number;\n enableOptimization?: boolean;\n skipCache?: boolean;\n force?: boolean;\n bytesPerEvent?: number;\n initialMessage?: WorkerToMainMessage;\n}\n\n/**\n * Configuration for the Nostr Manager\n */\nexport interface NostrManagerConfig {\n /**\n * Custom worker URL. If not provided, uses the bundled worker.\n */\n workerUrl?: string;\n /**\n * Custom worker instance. If provided, workerUrl is ignored.\n */\n worker?: Worker;\n}\n\n// const wasmReady = init(mainWasmUrl);\n\n/**\n * Pure TypeScript NostrClient that manages worker communication and state.\n * Uses WASM utilities for heavy lifting (encoding, decoding, crypto).\n */\nexport class NostrManager {\n private worker: Worker;\n private subscriptions = new Map<\n string,\n {\n buffer: SharedArrayBuffer;\n options: SubscriptionOptions;\n refCount: number;\n }\n >();\n private publishes = new Map<string, {buffer: SharedArrayBuffer}>();\n private signers = new Map<string, string>(); // name -> secret key hex\n\n public PERPETUAL_SUBSCRIPTIONS = [\"notifications\", \"starterpack\"];\n\n constructor(config: NostrManagerConfig = {}) {\n this.worker = this.createWorker(config);\n this.setupWorkerListener();\n }\n\n private createWorker(config: NostrManagerConfig): Worker {\n return new RustWorker();\n }\n\n private setupWorkerListener() {\n this.worker.onmessage = async (event) => {\n // await wasmReady;\n if (event.data instanceof Uint8Array) {\n let uint8Array = event.data;\n try {\n const message: any = decode(uint8Array);\n this.handleWorkerMessage(message);\n } catch (error) {\n console.error(\"Failed to decode worker message:\", error);\n } finally {\n // Aggressively clear memory references\n if (uint8Array) {\n uint8Array.fill(0);\n (uint8Array as any) = null;\n }\n }\n } else {\n console.log(\"Received non-arrayBuffer message:\", event.data);\n }\n };\n\n this.worker.onerror = (error) => {\n console.error(\"Worker error:\", error);\n };\n }\n\n private handleWorkerMessage(message: WorkerToMainMessage) {\n if (\"Count\" in message) {\n // this.handleSubscriptionCount(message.Count.subscription_id, message.Count.count);\n } else if (\"SignedEvent\" in message) {\n this.handleSignedEvent(\n message.SignedEvent.content,\n message.SignedEvent.signed_event,\n );\n } else if (\"PublicKey\" in message) {\n this.handlePublicKey(message.PublicKey.public_key);\n } else {\n console.warn(\"Unknown message type from worker:\", message);\n }\n }\n\n private handleSignedEvent(content: string, signedEvent: any) {\n console.log(\"Signed event received:\", content, signedEvent);\n }\n\n private handlePublicKey(publicKey: string) {\n console.log(\"Public key received:\", publicKey);\n }\n\n private createShortId(input: string): string {\n let hash = 0;\n for (let i = 0; i < input.length; i++) {\n const char = input.charCodeAt(i);\n hash = (hash << 5) - hash + char;\n hash = hash & hash;\n }\n const shortId = Math.abs(hash).toString(36);\n return shortId.substring(0, 63);\n }\n\n subscribe(\n subscriptionId: string,\n requests: Request[],\n options: SubscriptionOptions = {},\n ): SharedArrayBuffer {\n const subId =\n subscriptionId.length < 64\n ? subscriptionId\n : this.createShortId(subscriptionId);\n\n const existingSubscription = this.subscriptions.get(subId);\n\n if (existingSubscription) {\n existingSubscription.refCount++;\n return existingSubscription.buffer;\n }\n\n const defaultOptions: SubscriptionOptions = {\n closeOnEose: false,\n cacheFirst: true,\n skipCache: false,\n force: false,\n enableOptimization: true,\n ...options,\n };\n\n const totalLimit = requests.reduce(\n (sum, req) => sum + (req.limit || 100),\n 0,\n );\n\n const bufferSize = SharedBufferReader.calculateBufferSize(\n totalLimit,\n options.bytesPerEvent,\n );\n\n let initialMessage: Uint8Array<ArrayBufferLike> = new Uint8Array();\n\n if(options.initialMessage) {\n initialMessage = encode(options.initialMessage);\n }\n\n const buffer = new SharedArrayBuffer(bufferSize + initialMessage.length);\n\n // Initialize the buffer (sets write position to 4)\n SharedBufferReader.initializeBuffer(buffer);\n\n // Write the initial message if provided\n if(initialMessage.length > 0) {\n const success = SharedBufferReader.writeMessage(buffer, initialMessage);\n if (!success) {\n console.error(\"Failed to write initial message to buffer\");\n }\n }\n\n this.subscriptions.set(subId, {\n buffer,\n options: defaultOptions,\n refCount: 1,\n });\n\n // Convert SubscriptionOptions to SubscriptionConfig for the worker\n const config: SubscriptionConfig = {\n pipeline: defaultOptions.pipeline,\n closeOnEose: defaultOptions.closeOnEose,\n cacheFirst: defaultOptions.cacheFirst,\n timeoutMs: defaultOptions.timeoutMs,\n maxEvents: defaultOptions.maxEvents,\n enableOptimization: defaultOptions.enableOptimization,\n skipCache: defaultOptions.skipCache,\n force: defaultOptions.force,\n bytesPerEvent: defaultOptions.bytesPerEvent,\n };\n\n const message: MainToWorkerMessage = {\n Subscribe: {\n subscription_id: subId,\n requests: requests,\n config: config,\n },\n };\n\n try {\n const pack = encode(message);\n this.worker.postMessage({\n serializedMessage: pack,\n sharedBuffer: buffer,\n });\n\n return buffer;\n } catch (error) {\n this.subscriptions.delete(subId);\n throw error;\n }\n }\n\n getBuffer(subId: string): SharedArrayBuffer | undefined {\n const existingSubscription = this.subscriptions.get(subId);\n if (existingSubscription) {\n existingSubscription.refCount++;\n return existingSubscription.buffer;\n }\n return undefined;\n }\n\n unsubscribe(subscriptionId: string): void {\n const subId =\n subscriptionId.length < 64\n ? subscriptionId\n : this.createShortId(subscriptionId);\n const subscription = this.subscriptions.get(subId);\n if (subscription) {\n subscription.refCount--;\n }\n }\n\n publish(publish_id: string, event: NostrEvent): SharedArrayBuffer {\n\n // a publish buffer fit in 3kb\n const buffer = new SharedArrayBuffer(3072);\n\n // Initialize the buffer (sets write position to 4)\n SharedBufferReader.initializeBuffer(buffer);\n\n try {\n const template = {\n kind: event.kind,\n content: event.content,\n tags: event.tags || [],\n };\n\n const message: MainToWorkerMessage = {\n Publish: {\n publish_id: publish_id,\n template,\n },\n };\n\n const p = encode(message);\n\n this.worker.postMessage({ serializedMessage: p, sharedBuffer: buffer });\n\n this.publishes.set(publish_id, {buffer});\n return buffer;\n } catch (error) {\n console.error(\"Failed to publish event:\", error);\n throw error;\n }\n }\n\n setSigner(name: string, secretKeyHex: string): void {\n const message: MainToWorkerMessage = {\n SetSigner: {\n signer_type: name,\n private_key: secretKeyHex,\n },\n };\n\n const pack = encode(message);\n this.worker.postMessage(pack);\n this.signers.set(name, secretKeyHex);\n }\n\n signEvent(event: NostrEvent) {\n const template = {\n kind: event.kind,\n content: event.content,\n tags: event.tags,\n };\n\n const message: MainToWorkerMessage = {\n SignEvent: {\n template: template,\n },\n };\n const pack = encode(message);\n this.worker.postMessage(pack);\n }\n\n getPublicKey() {\n const message: MainToWorkerMessage = {\n GetPublicKey: {},\n };\n const pack = encode(message);\n this.worker.postMessage(pack);\n }\n\n cleanup(): void {\n const subscriptionsToDelete: string[] = [];\n\n for (const [subId, subscription] of this.subscriptions.entries()) {\n if (\n subscription.refCount <= 0 &&\n !this.PERPETUAL_SUBSCRIPTIONS.includes(subId)\n ) {\n subscriptionsToDelete.push(subId);\n }\n }\n\n for (const subId of subscriptionsToDelete) {\n const subscription = this.subscriptions.get(subId);\n if (subscription) {\n const message: MainToWorkerMessage = {\n Unsubscribe: {\n subscription_id: subId,\n },\n };\n const pack = encode(message);\n this.worker.postMessage(pack);\n this.subscriptions.delete(subId);\n }\n }\n }\n}\n\n/**\n * Factory function to create a new NostrManager instance.\n * @param config - Configuration for the NostrManager.\n * @returns A new instance of NostrManager.\n */\nexport function createNostrManager(\n config: NostrManagerConfig = {},\n): NostrManager {\n return new NostrManager(config);\n}\n\n/**\n * Default singleton instance of the NostrManager.\n * Useful for applications that only need one instance.\n */\nexport const nostrManager = new NostrManager();\n\nexport function cleanup(): void {\n nostrManager.cleanup();\n}\n\nexport * from \"./types\";\n"],"names":["WorkerToMainMessageBufferFull","SharedBufferReader","buffer","data","view","uint8View","currentWritePosition","requiredSpace","newWritePosition","message","encoded","encode","error","lastReadPosition","dataStartOffset","currentPos","maxMessages","messages","msgCount","scratch","eventLength","unpack","result","actualLastReadPosition","totalEventLimit","bytesPerEvent","dataSize","overhead","isKind39089","event","ReactionType","NostrManager","config","RustWorker","uint8Array","decode","content","signedEvent","publicKey","input","hash","i","char","subscriptionId","requests","options","subId","existingSubscription","defaultOptions","totalLimit","sum","req","bufferSize","initialMessage","pack","subscription","publish_id","template","p","name","secretKeyHex","subscriptionsToDelete","createNostrManager","nostrManager","cleanup"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@candypoets/nipworker",
3
- "version": "0.0.24",
3
+ "version": "0.0.26",
4
4
  "description": "Nostr client library with worker-based architecture using Rust WASM",
5
5
  "type": "module",
6
6
  "module": "./dist/index.js",
@@ -60,8 +60,8 @@
60
60
  "vite-plugin-wasm": "^3.0.0"
61
61
  },
62
62
  "dependencies": {
63
- "@candypoets/rust-main": "0.0.24",
64
- "@candypoets/rust-worker": "0.0.24",
63
+ "@candypoets/rust-main": "0.0.26",
64
+ "@candypoets/rust-worker": "0.0.26",
65
65
  "@msgpack/msgpack": "^3.0.0",
66
66
  "msgpackr": "^1.11.5",
67
67
  "nostr-tools": "^2.0.0"