@easbot/gateway 0.2.11 → 0.2.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.
@@ -1,2 +1,2 @@
1
- var s={name:"@easbot/gateway",version:"0.2.11",description:"EASBot Gateway - AI Agent Server and Multi-channel Integration Platform - \u652F\u6301 WebSocket\u3001HTTP\u3001Discord\u3001Telegram\u3001Slack \u7B49\u591A\u6E20\u9053\u96C6\u6210",type:"module",main:"dist/index.cjs",module:"dist/index.mjs",types:"dist/index.d.ts",exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.mjs",require:"./dist/index.cjs"},"./package.json":"./package.json"},scripts:{dev:"cross-env NODE_ENV=development tsx src/cli.ts -- start",start:"cross-env NODE_ENV=production node dist/cli.mjs start",build:"tsup --env.NODE_ENV production",test:"vitest","test:run":"vitest run",lint:"biome check .","lint:fix":"biome check --write .","lint:report":"biome check --reporter=summary .",format:"biome format .","format:fix":"biome format --write .","type-check":"tsc --noEmit",clean:"npx rimraf dist node_modules",prepare:"echo norun",prepublishOnly:"pnpm build","publish:npm":"bash scripts/publish.sh","publish:npm:win":"powershell -ExecutionPolicy Bypass -File scripts/publish.ps1"},keywords:["easbot","gateway","server","websocket","http","agent","multi-channel","discord","telegram","slack","feishu","wechat","chat","integration"],author:"houjallen",license:"MIT",repository:{type:"git",url:"https://github.com/houjallen/easbot.git",directory:"packages/gateway"},homepage:"https://github.com/houjallen/easbot/tree/main/packages/gateway#readme",bugs:{url:"https://github.com/houjallen/easbot/issues"},files:["dist","README.md","README.en.md","LICENSE"],dependencies:{"@ai-sdk/provider":"^3.0.10","@ai-sdk/provider-utils":"^4.0.27","@ai-sdk/openai-compatible":"^2.0.47","@ai-sdk/anthropic":"^3.0.76","@easbot/plugin":"workspace:*","@easbot/sdk":"workspace:*","@easbot/types":"workspace:*","@easbot/utils":"workspace:*","@hono/node-server":"^2.0.2","@hono/node-ws":"^1.3.0","@hono/standard-validator":"^0.2.2","@hono/zod-validator":"^0.7.6",commander:"^14.0.3",zod:"^4.4.3",hono:"^4.12.18","hono-openapi":"^1.3.0",ws:"^8.20.0",ai:"^6.0.176","better-sqlite3":"^12.9.0","jieba-wasm":"^2.4.0","xdg-basedir":"^5.1.0",minimatch:"^10.2.5"},devDependencies:{"@biomejs/biome":"^2.4.14","@types/better-sqlite3":"^7.6.13","@types/ws":"^8.18.1","@types/node":"^25.6.2","@vitest/coverage-v8":"^4.1.5",dotenv:"^17.4.2",tsup:"^8.5.1",typescript:"^6.0.3",vitest:"^4.1.5"},engines:{node:">=22.22.3"},publishConfig:{access:"public"}};
1
+ var s={name:"@easbot/gateway",version:"0.2.13",description:"EASBot Gateway - AI Agent Server and Multi-channel Integration Platform - \u652F\u6301 WebSocket\u3001HTTP\u3001Discord\u3001Telegram\u3001Slack \u7B49\u591A\u6E20\u9053\u96C6\u6210",type:"module",main:"dist/index.cjs",module:"dist/index.mjs",types:"dist/index.d.ts",exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.mjs",require:"./dist/index.cjs"},"./package.json":"./package.json"},scripts:{dev:"cross-env NODE_ENV=development tsx src/cli.ts -- start",start:"cross-env NODE_ENV=production node dist/cli.mjs start",build:"tsup --env.NODE_ENV production",test:"vitest","test:run":"vitest run",lint:"biome check .","lint:fix":"biome check --write .","lint:report":"biome check --reporter=summary .",format:"biome format .","format:fix":"biome format --write .","type-check":"tsc --noEmit",clean:"npx rimraf dist node_modules",prepare:"echo norun",prepublishOnly:"pnpm build","publish:npm":"bash scripts/publish.sh","publish:npm:win":"powershell -ExecutionPolicy Bypass -File scripts/publish.ps1"},keywords:["easbot","gateway","server","websocket","http","agent","multi-channel","discord","telegram","slack","feishu","wechat","chat","integration"],author:"houjallen",license:"MIT",repository:{type:"git",url:"https://github.com/houjallen/easbot.git",directory:"packages/gateway"},homepage:"https://github.com/houjallen/easbot/tree/main/packages/gateway#readme",bugs:{url:"https://github.com/houjallen/easbot/issues"},files:["dist","README.md","README.en.md","LICENSE"],dependencies:{"@ai-sdk/provider":"^3.0.10","@ai-sdk/provider-utils":"^4.0.27","@ai-sdk/openai-compatible":"^2.0.47","@ai-sdk/anthropic":"^3.0.76","@easbot/plugin":"workspace:*","@easbot/sdk":"workspace:*","@easbot/types":"workspace:*","@easbot/utils":"workspace:*","@hono/node-server":"^2.0.2","@hono/node-ws":"^1.3.0","@hono/standard-validator":"^0.2.2","@hono/zod-validator":"^0.7.6",commander:"^14.0.3",zod:"^4.4.3",hono:"^4.12.18","hono-openapi":"^1.3.0",ws:"^8.20.0",ai:"^6.0.176","better-sqlite3":"^12.9.0","jieba-wasm":"^2.4.0","xdg-basedir":"^5.1.0",minimatch:"^10.2.5"},devDependencies:{"@biomejs/biome":"^2.4.14","@types/better-sqlite3":"^7.6.13","@types/ws":"^8.18.1","@types/node":"^25.6.2","@vitest/coverage-v8":"^4.1.5",dotenv:"^17.4.2",tsup:"^8.5.1",typescript:"^6.0.3",vitest:"^4.1.5"},engines:{node:">=22.22.3"},publishConfig:{access:"public"}};
2
2
  export{s as a};
@@ -0,0 +1,2 @@
1
+ import {v as v$1,S as S$1,P as P$1,Q as Q$1}from'./chunk-LFGRYJV2.mjs';import {a}from'./chunk-6TRL3CVJ.mjs';import {a as a$1}from'./chunk-HAMGVOQD.mjs';import*as f from'fs';import f__default from'fs';import*as W from'path';import {createServer}from'@easbot/utils';import {WebSocketServer}from'ws';import Q from'https';import V from'http';import {HookEvent}from'@easbot/plugin';var I=class{constructor(e={}){a$1(this,"log",a.create({service:"gateway:lock"}));a$1(this,"config");a$1(this,"locks",new Map);a$1(this,"cleanupTimer",null);this.config={defaultTimeout:e.defaultTimeout??6e4,checkInterval:e.checkInterval??1e4};}tryAcquire(e,t,s,n){let r=this.locks.get(e);if(r)if(this.isLockExpired(r))this.log.warn("lock expired, releasing",{sessionId:e,oldHolder:r.holderId,newHolder:s}),this.locks.delete(e);else return this.log.debug("lock already held",{sessionId:e,holder:r.holderId,messageId:r.messageId}),false;let a={sessionId:e,messageId:t,holderId:s,acquiredAt:Date.now(),timeout:n??this.config.defaultTimeout};return this.locks.set(e,a),this.log.debug("lock acquired",{sessionId:e,messageId:t,subscriberId:s}),true}release(e,t){let s=this.locks.get(e);return s?s.holderId!==t?(this.log.warn("cannot release lock held by another",{sessionId:e,holder:s.holderId,requester:t}),false):(this.locks.delete(e),this.log.debug("lock released",{sessionId:e,subscriberId:t}),true):(this.log.debug("no lock to release",{sessionId:e}),false)}forceRelease(e){let t=this.locks.get(e);return t?(this.locks.delete(e),this.log.warn("lock force released",{sessionId:e,holder:t.holderId,messageId:t.messageId}),true):false}isLockExpired(e){return Date.now()-e.acquiredAt>e.timeout}getLock(e){return this.locks.get(e)}isLocked(e){let t=this.locks.get(e);return t?this.isLockExpired(t)?(this.locks.delete(e),false):true:false}getHolder(e){let t=this.locks.get(e);if(!(!t||this.isLockExpired(t)))return t.holderId}startCleanup(){this.cleanupTimer||(this.cleanupTimer=setInterval(()=>{this.cleanupExpired();},this.config.checkInterval),this.log.info("lock cleanup started",{interval:this.config.checkInterval}));}stopCleanup(){this.cleanupTimer&&(clearInterval(this.cleanupTimer),this.cleanupTimer=null,this.log.info("lock cleanup stopped"));}cleanupExpired(){let e=0;for(let[t,s]of this.locks)this.isLockExpired(s)&&(this.locks.delete(t),e++,this.log.warn("expired lock cleaned up",{sessionId:t,holder:s.holderId,messageId:s.messageId}));return e>0&&this.log.info("cleaned up expired locks",{count:e}),e}getAllLocks(){return [...this.locks.values()]}getLockCount(){return this.locks.size}};var k=class{constructor(e={}){a$1(this,"log",a.create({service:"gateway:retry"}));a$1(this,"policy");a$1(this,"retryQueue",new Map);a$1(this,"retryTimer",null);a$1(this,"handlers",new Map);this.policy={maxRetries:e.maxRetries??3,initialDelay:e.initialDelay??1e3,maxDelay:e.maxDelay??3e4,backoffMultiplier:e.backoffMultiplier??2};}registerHandler(e,t){this.handlers.set(e,t);}unregisterHandler(e){this.handlers.delete(e);}scheduleRetry(e,t){let s=e.id,n=this.retryQueue.get(s);if(n||(n={messageId:s,retryCount:0,nextRetryAt:0,errors:[]},this.retryQueue.set(s,n)),n.errors.push({error:t.message,timestamp:Date.now()}),n.retryCount>=this.policy.maxRetries)return this.log.error("max retries exceeded",{messageId:s,retryCount:n.retryCount,maxRetries:this.policy.maxRetries}),false;let r=Math.min(this.policy.initialDelay*this.policy.backoffMultiplier**n.retryCount,this.policy.maxDelay);return n.retryCount++,n.nextRetryAt=Date.now()+r,this.log.warn("retry scheduled",{messageId:s,retryCount:n.retryCount,delay:r,error:t.message}),true}getPendingRetries(){let e=Date.now(),t=[];for(let s of this.retryQueue.values())s.nextRetryAt<=e&&s.retryCount<=this.policy.maxRetries&&t.push(s);return t}async executeRetry(e){if(!this.handlers.get(e.messageId))return this.log.warn("no handler for retry",{messageId:e.messageId}),false;this.retryQueue.delete(e.messageId);try{return this.log.info("executing retry",{messageId:e.messageId,retryCount:e.retryCount}),!0}catch(s){return this.log.error("retry execution failed",{messageId:e.messageId,error:String(s)}),false}}clearRetry(e){this.retryQueue.delete(e),this.handlers.delete(e),this.log.debug("retry cleared",{messageId:e});}startProcessing(){this.retryTimer||(this.retryTimer=setInterval(async()=>{let e=this.getPendingRetries();for(let t of e)await this.executeRetry(t);},1e3),this.log.info("retry processing started"));}stopProcessing(){this.retryTimer&&(clearInterval(this.retryTimer),this.retryTimer=null,this.log.info("retry processing stopped"));}getStats(){let e=0;for(let t of this.retryQueue.values())e+=t.errors.length;return {pendingCount:this.retryQueue.size,totalErrors:e}}};function L(c,e,t,s){return {id:`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:c,type:e,content:t,metadata:{channel:{platform:s?.channel?.platform??"api",channelId:s?.channel?.channelId??"default",userId:s?.channel?.userId,chatId:s?.channel?.chatId,...Object.fromEntries(Object.entries(s?.channel??{}).filter(([n])=>!["platform","channelId","userId","chatId"].includes(n)))},context:s?.context,agent:s?.agent},timestamp:Date.now()}}function B(c,e,t,s){return L(c,e,[{type:"text",text:t}],s)}function R(c){let e=[c.platform,c.channelId];return c.chatId&&e.push(c.chatId),c.userId&&e.push(c.userId),e.join("_")}function E(){return {status:"active",messageCount:0,lastMessageAt:null}}function x(c,e){return `sub_${c}_${e}`}var D=v$1,S={maxConnections:1e3,connectionTimeout:6e4,sessionExpireMs:3e5};var q={type:"token",enabled:true,tokens:[],onInvalid:"reject",allowAnonymous:false,defaultPermissions:[]};var F={heartbeatInterval:3e4,heartbeatTimeout:9e4,heartbeatCheckInterval:1e4,autoDeregisterTimeout:3e5,maxAgents:100,maxRegistrations:1e3};var U={mode:"both",interval:3e4,pushEvents:["register","deregister","status_change"],conflictResolution:"latest",remoteNodes:[]};var $={maxConnectionsPerChannel:100,idleTimeout:3e5,reuseStrategy:"lru",healthCheckInterval:6e4,acquireTimeout:5e3,warmupCount:0};var P=class{constructor(){a$1(this,"log",a.create({service:"gateway:router"}));a$1(this,"subscribers",new Map);a$1(this,"subscriptions",new Map);a$1(this,"sessionSubscriptions",new Map);a$1(this,"channelPlugins",new Map);a$1(this,"lockManager");a$1(this,"retryManager");a$1(this,"cancelHandlers",new Set);this.lockManager=new I,this.retryManager=new k,this.lockManager.startCleanup();}onCancel(e){this.cancelHandlers.add(e);}offCancel(e){this.cancelHandlers.delete(e);}async emitCancel(e){for(let t of this.cancelHandlers)try{await t(e);}catch(s){this.log.error("cancel handler error",{error:String(s)});}}async route(e){this.log.debug("routing message",{id:e.id,sessionId:e.sessionId,type:e.type});let t=this.sessionSubscriptions.get(e.sessionId);if(!t||t.size===0)return this.log.debug("no subscribers for session",{sessionId:e.sessionId}),{success:true,broadcastCount:0,dispatchedToSubAgent:false};let s=0,n=null,r=[];for(let a of t){let g=this.subscriptions.get(a);if(!g)continue;let l=this.subscribers.get(g.subscriberId);if(l){if(e.type==="input"){if(!this.lockManager.tryAcquire(e.sessionId,e.id,l.id)){this.log.debug("skipping subscriber, lock held by another",{subscriberId:l.id,holder:this.lockManager.getHolder(e.sessionId)});continue}n=l.id;}try{await l.connection.send(e),s++,l.lastActiveAt=Date.now();}catch(p){this.log.warn("failed to send message to subscriber",{subscriberId:l.id,error:String(p)}),r.push(p instanceof Error?p:new Error(String(p))),n===l.id&&(this.lockManager.release(e.sessionId,l.id),n=null);}}}return this.log.debug("message routed",{id:e.id,broadcastCount:s,errorCount:r.length,processedBy:n}),{success:r.length===0,broadcastCount:s,dispatchedToSubAgent:false,error:r.length>0?r.map(a=>a.message).join("; "):void 0}}async completeMessage(e,t,s,n="completed"){this.lockManager.release(e,s),await this.emitCancel({sessionId:e,messageId:t,processedBy:s,reason:n,timestamp:Date.now()}),this.log.info("message completed",{sessionId:e,messageId:t,subscriberId:s,reason:n});}scheduleRetry(e,t){return this.retryManager.scheduleRetry(e,t)}async subscribe(e,t){let s=x(t.id,e);if(this.subscriptions.has(s))return this.log.debug("already subscribed",{subscriberId:t.id,sessionId:e}),this.subscriptions.get(s);let n={id:s,subscriberId:t.id,sessionId:e,createdAt:Date.now()};return this.subscribers.has(t.id)||this.subscribers.set(t.id,t),this.subscriptions.set(s,n),this.sessionSubscriptions.has(e)||this.sessionSubscriptions.set(e,new Set),this.sessionSubscriptions.get(e).add(s),t.sessions.add(e),this.log.info("subscribed to session",{subscriberId:t.id,sessionId:e,subscriptionId:s}),n}async unsubscribe(e){let t=this.subscriptions.get(e);if(!t){this.log.warn("subscription not found",{subscriptionId:e});return}this.subscriptions.delete(e);let s=this.sessionSubscriptions.get(t.sessionId);s&&(s.delete(e),s.size===0&&this.sessionSubscriptions.delete(t.sessionId));let n=this.subscribers.get(t.subscriberId);n&&n.sessions.delete(t.sessionId),this.log.info("unsubscribed from session",{subscriberId:t.subscriberId,sessionId:t.sessionId,subscriptionId:e});}async unsubscribeAll(e){let t=this.subscribers.get(e);if(!t)return;let s=[...t.sessions];for(let n of s){let r=x(e,n);await this.unsubscribe(r);}this.subscribers.delete(e),this.log.info("unsubscribed all sessions",{subscriberId:e});}async registerChannel(e){if(this.channelPlugins.has(e.id)){this.log.warn("channel plugin already registered",{id:e.id});return}this.channelPlugins.set(e.id,e),this.log.info("channel plugin registered",{id:e.id,platform:e.platform});}async unregisterChannel(e){let t=this.channelPlugins.get(e);if(!t){this.log.warn("channel plugin not found",{id:e});return}await t.stop(),this.channelPlugins.delete(e),this.log.info("channel plugin unregistered",{id:e});}getSubscriberCount(e){let t=this.sessionSubscriptions.get(e);return t?t.size:0}getAllSubscribers(){return [...this.subscribers.values()]}getChannelPlugin(e){return this.channelPlugins.get(e)}getAllChannelPlugins(){return [...this.channelPlugins.values()]}getLockManager(){return this.lockManager}getRetryManager(){return this.retryManager}async shutdown(){this.lockManager.stopCleanup(),this.retryManager.stopProcessing(),this.log.info("router shutdown");}};var b=class{constructor(e={}){a$1(this,"log",a.create({service:"gateway:message-store"}));a$1(this,"config");a$1(this,"messages",new Map);a$1(this,"sessionMessages",new Map);this.config={storagePath:e.storagePath??"./data/gateway-messages",maxRetentionDays:e.maxRetentionDays??30,maxRecords:e.maxRecords??1e5};}async store(e){let t={message:e,status:"pending",processedBy:null,processedAt:null,retryCount:0,error:null,createdAt:Date.now(),updatedAt:Date.now()};return this.messages.set(e.id,t),this.sessionMessages.has(e.sessionId)||this.sessionMessages.set(e.sessionId,new Set),this.sessionMessages.get(e.sessionId).add(e.id),this.log.debug("message stored",{id:e.id,sessionId:e.sessionId,type:e.type}),t}get(e){return this.messages.get(e)}async updateStatus(e,t,s={}){let n=this.messages.get(e);if(!n){this.log.warn("message not found for status update",{messageId:e});return}n.status=t,n.updatedAt=Date.now(),s.processedBy&&(n.processedBy=s.processedBy),(t==="completed"||t==="failed")&&(n.processedAt=Date.now()),s.error&&(n.error=s.error),this.log.debug("message status updated",{id:e,status:t,processedBy:s.processedBy});}incrementRetry(e){let t=this.messages.get(e);return t?(t.retryCount++,t.updatedAt=Date.now(),t.retryCount):0}isProcessed(e){let t=this.messages.get(e);return t?t.status==="completed"||t.status==="processing":false}getSessionHistory(e,t=100){let s=this.sessionMessages.get(e);if(!s)return [];let n=[];for(let r of s){let a=this.messages.get(r);a&&n.push(a);}return n.sort((r,a)=>r.createdAt-a.createdAt),n.slice(-t)}getPendingMessages(e){let t=[];for(let s of this.messages.values())s.status==="pending"&&(!e||s.message.sessionId===e)&&t.push(s);return t}getFailedMessages(e=3){let t=[];for(let s of this.messages.values())s.status==="failed"&&s.retryCount<e&&t.push(s);return t}async cleanup(){let e=Date.now(),t=this.config.maxRetentionDays*24*60*60*1e3,s=[];for(let[n,r]of this.messages)e-r.createdAt>t&&s.push(n);for(let n of s){let r=this.messages.get(n);if(r){let a=this.sessionMessages.get(r.message.sessionId);a&&a.delete(n),this.messages.delete(n);}}return s.length>0&&this.log.info("cleaned up expired messages",{count:s.length}),s.length}getStats(){let e={total:this.messages.size,pending:0,processing:0,completed:0,failed:0,cancelled:0};for(let t of this.messages.values())e[t.status]++;return e}};var v=class{constructor(e={}){a$1(this,"log",a.create({service:"gateway:session-store"}));a$1(this,"config");a$1(this,"sessions",new Map);a$1(this,"channelSessions",new Map);a$1(this,"autoSaveTimer",null);a$1(this,"isDirty",false);this.config={storagePath:e.storagePath??"./data/gateway-sessions",autoSaveInterval:e.autoSaveInterval??6e4,maxRetentionDays:e.maxRetentionDays??7,enablePersistence:e.enablePersistence??true};}async initialize(){if(!this.config.enablePersistence){this.log.info("session persistence disabled");return}await this.ensureStorageDir(),await this.load(),this.startAutoSave(),this.log.info("session store initialized",{sessionCount:this.sessions.size,storagePath:this.config.storagePath});}async close(){this.stopAutoSave(),this.isDirty&&await this.save(),this.log.info("session store closed");}async set(e){this.sessions.set(e.id,e),this.channelSessions.has(e.channel.channelId)||this.channelSessions.set(e.channel.channelId,new Set),this.channelSessions.get(e.channel.channelId).add(e.id),this.markDirty();}get(e){return this.sessions.get(e)}async delete(e){let t=this.sessions.get(e);if(!t)return;this.sessions.delete(e);let s=this.channelSessions.get(t.channel.channelId);s&&(s.delete(e),s.size===0&&this.channelSessions.delete(t.channel.channelId)),this.markDirty();}getAll(){return [...this.sessions.values()]}getByChannel(e){let t=this.channelSessions.get(e);return t?[...t].map(s=>this.sessions.get(s)).filter(s=>s!==void 0):[]}size(){return this.sessions.size}markDirty(){this.isDirty=true;}async save(){if(!this.config.enablePersistence)return;let e=this.getSessionFilePath(),t=this.serializeAll();try{await this.ensureStorageDir(),await f.promises.writeFile(e,JSON.stringify(t,null,2),"utf-8"),this.isDirty=!1,this.log.debug("sessions saved",{count:t.length});}catch(s){throw this.log.error("failed to save sessions",{error:s}),s}}async load(){if(!this.config.enablePersistence)return;let e=this.getSessionFilePath();try{if(!f.existsSync(e)){this.log.debug("no saved sessions found");return}let t=await f.promises.readFile(e,"utf-8"),s=JSON.parse(t);this.sessions.clear(),this.channelSessions.clear();let n=Date.now(),r=this.config.maxRetentionDays*24*60*60*1e3;for(let a of s){if(n-a.lastActiveAt>r)continue;let g={id:a.id,channel:a.channel,backendSessionId:a.backendSessionId,subscribers:new Set(a.subscribers),state:a.state,createdAt:a.createdAt,lastActiveAt:a.lastActiveAt};this.sessions.set(g.id,g),this.channelSessions.has(g.channel.channelId)||this.channelSessions.set(g.channel.channelId,new Set),this.channelSessions.get(g.channel.channelId).add(g.id);}this.log.info("sessions loaded",{loaded:this.sessions.size,skipped:s.length-this.sessions.size});}catch(t){this.log.error("failed to load sessions",{error:t});}}serializeAll(){let e=[];for(let t of this.sessions.values())e.push(this.serialize(t));return e}serialize(e){return {id:e.id,channel:e.channel,backendSessionId:e.backendSessionId,subscribers:[...e.subscribers],state:e.state,createdAt:e.createdAt,lastActiveAt:e.lastActiveAt}}getSessionFilePath(){return W.join(this.config.storagePath,"sessions.json")}async ensureStorageDir(){f.existsSync(this.config.storagePath)||await f.promises.mkdir(this.config.storagePath,{recursive:true});}startAutoSave(){this.autoSaveTimer||(this.autoSaveTimer=setInterval(async()=>{this.isDirty&&await this.save();},this.config.autoSaveInterval));}stopAutoSave(){this.autoSaveTimer&&(clearInterval(this.autoSaveTimer),this.autoSaveTimer=null);}async cleanup(){let e=Date.now(),t=this.config.maxRetentionDays*24*60*60*1e3,s=[];for(let n of this.sessions.values())e-n.lastActiveAt>t&&s.push(n.id);for(let n of s)await this.delete(n);return s.length>0&&(this.log.info("cleaned up expired sessions",{count:s.length}),await this.save()),s.length}getStats(){let e={total:this.sessions.size,byStatus:{active:0,idle:0,closed:0},byPlatform:{}};for(let t of this.sessions.values()){e.byStatus[t.state.status]++;let s=t.channel.platform;e.byPlatform[s]=(e.byPlatform[s]||0)+1;}return e}};var C=class{constructor(e={}){a$1(this,"log",a.create({service:"gateway:session"}));a$1(this,"sessionStore");a$1(this,"config");a$1(this,"initialized",false);this.config={enablePersistence:e.enablePersistence??true,storagePath:e.storagePath??"./data/gateway-sessions",autoSaveInterval:e.autoSaveInterval??6e4,maxRetentionDays:e.maxRetentionDays??7},this.sessionStore=new v({storagePath:this.config.storagePath,autoSaveInterval:this.config.autoSaveInterval,maxRetentionDays:this.config.maxRetentionDays,enablePersistence:this.config.enablePersistence});}async initialize(){this.initialized||(await this.sessionStore.initialize(),this.initialized=true,this.log.info("session manager initialized",{sessionCount:this.sessionStore.size(),persistenceEnabled:this.config.enablePersistence}));}async close(){await this.sessionStore.close(),this.log.info("session manager closed");}async getOrCreate(e,t={}){let s=R(e),n=this.sessionStore.get(s);return n?(n.lastActiveAt=Date.now(),this.log.debug("session found",{sessionId:s}),n):(n={id:s,channel:e,backendSessionId:null,subscribers:new Set,state:E(),createdAt:Date.now(),lastActiveAt:Date.now()},await this.sessionStore.set(n),this.log.info("session created",{sessionId:s,platform:e.platform,channelId:e.channelId,chatId:e.chatId,userId:e.userId}),n)}get(e){return this.sessionStore.get(e)}async closeSession(e){let t=this.sessionStore.get(e);if(!t){this.log.warn("session not found",{sessionId:e});return}t.state.status="closed",await this.sessionStore.delete(e),this.log.info("session closed",{sessionId:e});}async closeById(e){return this.closeSession(e)}async bindBackendSession(e,t){let s=this.sessionStore.get(e);if(!s)throw new Error(`Session not found: ${e}`);s.backendSessionId=t,await this.sessionStore.set(s),this.log.info("backend session bound",{gatewaySessionId:e,backendSessionId:t});}async unbindBackendSession(e){let t=this.sessionStore.get(e);if(!t){this.log.warn("session not found",{sessionId:e});return}t.backendSessionId=null,await this.sessionStore.set(t),this.log.info("backend session unbound",{gatewaySessionId:e});}getState(e){return this.sessionStore.get(e)?.state}async incrementMessageCountOrCreate(e,t){let s=this.sessionStore.get(e);if(!s){let n=t??{platform:"api",channelId:"default"};s=await this.getOrCreate(n);}return s.state.messageCount++,s.state.lastMessageAt=Date.now(),s.lastActiveAt=Date.now(),await this.sessionStore.set(s),s}async incrementMessageCount(e){let t=this.sessionStore.get(e);t&&(t.state.messageCount++,t.state.lastMessageAt=Date.now(),t.lastActiveAt=Date.now(),await this.sessionStore.set(t));}async addSubscriber(e,t){let s=this.sessionStore.get(e);if(!s){this.log.warn("session not found",{sessionId:e});return}s.subscribers.add(t),await this.sessionStore.set(s),this.log.debug("subscriber added to session",{sessionId:e,subscriberId:t});}async removeSubscriber(e,t){let s=this.sessionStore.get(e);s&&(s.subscribers.delete(t),await this.sessionStore.set(s),this.log.debug("subscriber removed from session",{sessionId:e,subscriberId:t}));}query(e){let t=this.sessionStore.getAll();return e.platform&&(t=t.filter(s=>s.channel.platform===e.platform)),e.channelId&&(t=t.filter(s=>s.channel.channelId===e.channelId)),e.userId&&(t=t.filter(s=>s.channel.userId===e.userId)),e.status&&(t=t.filter(s=>s.state.status===e.status)),t}getSessionsByChannel(e){return this.sessionStore.getByChannel(e)}getAllSessions(){return this.sessionStore.getAll()}getSessionCount(){return this.sessionStore.size()}async cleanupIdleSessions(e){let t=Date.now(),s=[];for(let n of this.sessionStore.getAll())n.state.status==="idle"&&t-n.lastActiveAt>e&&s.push(n.id);for(let n of s)await this.closeSession(n);return s.length>0&&this.log.info("cleaned up idle sessions",{count:s.length}),s.length}async save(){this.log.debug("session save triggered");}getStats(){return this.sessionStore.getStats()}};var w=class{constructor(e={}){a$1(this,"log",a.create({service:"gateway:plugin-loader"}));a$1(this,"plugins",new Map);a$1(this,"messageHandler",null);a$1(this,"healthCheckTimer",null);a$1(this,"config");this.config={pluginDir:e.pluginDir??"./plugins",autoDiscover:e.autoDiscover??false,healthCheckInterval:e.healthCheckInterval??6e4,healthCheckTimeout:e.healthCheckTimeout??5e3};}setMessageHandler(e){this.messageHandler=e;}async register(e,t){if(this.plugins.has(e.id)){this.log.warn("plugin already registered",{id:e.id});return}this.plugins.set(e.id,{plugin:e,config:t,started:false}),this.log.info("plugin registered",{id:e.id,platform:e.platform,name:e.name});}async unregister(e){let t=this.plugins.get(e);if(!t){this.log.warn("plugin not found",{id:e});return}t.started&&await this.stop(e),this.plugins.delete(e),this.log.info("plugin unregistered",{id:e});}async start(e){let t=this.plugins.get(e);if(!t)throw new Error(`Plugin not found: ${e}`);if(t.started){this.log.warn("plugin already started",{id:e});return}if(!this.messageHandler)throw new Error("Message handler not set");if(!t.config.enabled){this.log.warn("plugin is disabled",{id:e});return}this.log.info("starting plugin",{id:e});try{await t.plugin.start(t.config,this.messageHandler),t.started=!0,this.log.info("plugin started",{id:e});}catch(s){throw this.log.error("failed to start plugin",{id:e,error:String(s)}),s}}async stop(e){let t=this.plugins.get(e);if(!t){this.log.warn("plugin not found",{id:e});return}if(!t.started){this.log.warn("plugin not started",{id:e});return}this.log.info("stopping plugin",{id:e});try{await t.plugin.stop(),t.started=!1,this.log.info("plugin stopped",{id:e});}catch(s){throw this.log.error("failed to stop plugin",{id:e,error:String(s)}),s}}async startAll(){let e=[];for(let[t,s]of this.plugins){if(!s.config.enabled){this.log.debug("skipping disabled plugin",{id:t});continue}try{await this.start(t);}catch(n){e.push({id:t,error:n instanceof Error?n:new Error(String(n))});}}e.length>0&&this.log.warn("some plugins failed to start",{count:e.length,errors:e.map(t=>({id:t.id,error:t.error.message}))});}async stopAll(){let e=[];for(let[t,s]of this.plugins)if(s.started)try{await this.stop(t);}catch(n){e.push({id:t,error:n instanceof Error?n:new Error(String(n))});}e.length>0&&this.log.warn("some plugins failed to stop",{count:e.length,errors:e.map(t=>({id:t.id,error:t.error.message}))});}getPlugin(e){return this.plugins.get(e)?.plugin}getAllPlugins(){return [...this.plugins.values()].map(e=>e.plugin)}getRunningPlugins(){return [...this.plugins.values()].filter(e=>e.started).map(e=>e.plugin)}isRunning(e){return this.plugins.get(e)?.started??false}async healthCheck(e){let t=this.plugins.get(e);if(!t)return {healthy:false,lastCheck:Date.now(),error:"Plugin not found"};try{let s=await Promise.race([t.plugin.healthCheck(),new Promise((n,r)=>setTimeout(()=>r(new Error("Health check timeout")),this.config.healthCheckTimeout))]);return t.lastHealthCheck=s,s}catch(s){let n={healthy:false,lastCheck:Date.now(),error:s instanceof Error?s.message:String(s)};return t.lastHealthCheck=n,n}}async healthCheckAll(){let e=new Map;for(let[t]of this.plugins)e.set(t,await this.healthCheck(t));return e}startHealthCheckTimer(){if(this.healthCheckTimer){this.log.warn("health check timer already running");return}this.healthCheckTimer=setInterval(async()=>{let e=await this.healthCheckAll();for(let[t,s]of e)s.healthy||this.log.warn("plugin health check failed",{id:t,error:s.error});},this.config.healthCheckInterval),this.log.info("health check timer started",{interval:this.config.healthCheckInterval});}stopHealthCheckTimer(){this.healthCheckTimer&&(clearInterval(this.healthCheckTimer),this.healthCheckTimer=null,this.log.info("health check timer stopped"));}async sendToPlugin(e,t){let s=this.plugins.get(e);if(!s)throw new Error(`Plugin not found: ${e}`);if(!s.started)throw new Error(`Plugin not started: ${e}`);return s.plugin.send(t)}};var o=a.create({service:"gateway:websocket-server"});function K(){return `ws_${Date.now()}_${Math.random().toString(36).substring(2,9)}`}function A(c,e){for(let[t,s]of e.entries())if(s===c)return t;return null}var M=class{constructor(e={}){a$1(this,"config");a$1(this,"server",null);a$1(this,"wss",null);a$1(this,"httpsServer",null);a$1(this,"httpRedirectServer",null);a$1(this,"connections",new Map);a$1(this,"subscriptions",new Map);a$1(this,"sessionSubscriptions",new Map);a$1(this,"sessionLastActivity",new Map);a$1(this,"cleanupInterval",null);a$1(this,"heartbeatInterval",null);a$1(this,"startTime",0);a$1(this,"running",false);this.config={...D,...e};}async start(){if(this.running){o.warn("WebSocket server is already running");return}o.debug("WebSocket server start: checking configuration",{port:this.config.port,hostname:this.config.hostname,path:this.config.path,https:this.config.https?.enabled??false}),this.startTime=Date.now();let e={onOpen:(n,r)=>{this.handleConnection(r);},onMessage:async(n,r)=>{try{let a=JSON.parse(n.data.toString());await this.handleMessage(r,a);}catch(a){o.error("Error processing WebSocket message",{error:a.message}),this.sendError(r,a.message);}},onClose:(n,r)=>{this.handleDisconnection(r);},onError:(n,r)=>{o.error("WebSocket error",{error:n.type});}},t=this.config.https?.enabled??false;o.debug("WebSocket server start: choosing server type",{https:t});try{t?(o.debug("WebSocket server start: starting HTTPS server"),await this.startHTTPSServer(e)):(o.debug("WebSocket server start: starting HTTP server"),await this.startHTTPServer(e)),o.debug("WebSocket server start: HTTP/HTTPS server started successfully");}catch(n){throw o.error("WebSocket server start: HTTP/HTTPS server failed",{error:n instanceof Error?n.message:String(n)}),n}o.debug("WebSocket server start: server initialization complete"),this.cleanupInterval=setInterval(()=>this.cleanupExpiredSessions(),3e4),this.heartbeatInterval=setInterval(()=>this.checkConnectionTimeout(),this.config.heartbeatInterval),this.running=true;let s=t?"wss":"ws";o.info("WebSocket server started",{url:`${s}://${this.config.hostname}:${this.config.port}${this.config.path}`});}async startHTTPServer(e){o.debug("HTTP server start: creating server",{port:this.config.port,hostname:this.config.hostname}),this.server=createServer({port:this.config.port,hostname:this.config.hostname,routes:this.createRoutes(),timeout:this.config.connectionTimeout,cors:true}),o.debug("HTTP server start: server created, checking raw server");let t=this.server.raw;t?(o.debug("HTTP server start: raw server available, setting up WebSocket upgrade"),this.setupWebSocketUpgrade(t,e),o.debug("HTTP server start: WebSocket upgrade configured")):o.warn("HTTP server start: raw server is undefined");}async startHTTPSServer(e){let t=this.config.https,s=this.loadSSLCertificates(t);if(!s){o.error("Failed to load SSL certificates, falling back to HTTP"),await this.startHTTPServer(e);return}this.server=createServer({port:this.config.port,hostname:this.config.hostname,routes:this.createRoutes(),timeout:this.config.connectionTimeout,cors:true});let n=this.server.raw;this.httpsServer=Q.createServer(s,n.listeners("request")[0]),this.setupWebSocketUpgrade(this.httpsServer,e),await new Promise(r=>{this.httpsServer.listen(this.config.port,this.config.hostname,()=>{o.info("HTTPS server started",{port:this.config.port,hostname:this.config.hostname}),r();});}),t.forceRedirect&&t.httpPort&&this.startHTTPRedirectServer(t.httpPort);}loadSSLCertificates(e){try{return e.cert&&e.key?{cert:e.cert,key:e.key}:e.certPath&&e.keyPath?{cert:f__default.readFileSync(e.certPath),key:f__default.readFileSync(e.keyPath)}:null}catch(t){return o.error("Failed to load SSL certificates",{error:t.message}),null}}setupWebSocketUpgrade(e,t){this.wss=new WebSocketServer({noServer:true}),e.on("upgrade",(s,n,r)=>{try{let g=new URL(s.url||"/",`http://${s.headers.host}`).pathname;(g===this.config.path||this.config.path==="/"&&(g==="/"||g===""))&&this.wss.handleUpgrade(s,n,r,l=>{t.onOpen?.({type:"open"},l),l.on("message",p=>{t.onMessage?.({data:p},l);}),l.on("close",(p,G)=>{t.onClose?.({code:p,reason:G},l);}),l.on("error",p=>{t.onError?.({type:"error",message:p.message},l);});});}catch{}});}startHTTPRedirectServer(e){this.httpRedirectServer=V.createServer((t,s)=>{let r=`https://${t.headers.host?.split(":")[0]||this.config.hostname}:${this.config.port}${t.url}`;o.debug("Redirecting HTTP to HTTPS",{from:t.url,to:r}),s.writeHead(301,{Location:r,"Content-Type":"text/plain"}),s.end(`Redirecting to ${r}`);}),this.httpRedirectServer.listen(e,this.config.hostname,()=>{o.info("HTTP redirect server started",{port:e,hostname:this.config.hostname});});}async stop(){if(!this.running){o.warn("WebSocket server is not running");return}o.info("Stopping WebSocket server"),this.cleanupInterval&&(clearInterval(this.cleanupInterval),this.cleanupInterval=null),this.heartbeatInterval&&(clearInterval(this.heartbeatInterval),this.heartbeatInterval=null);for(let[e,t]of this.connections.entries())try{t.close(1e3,"Server shutting down");}catch(s){o.debug("Error closing connection",{clientId:e,error:String(s)});}this.connections.clear(),this.subscriptions.clear(),this.sessionSubscriptions.clear(),this.sessionLastActivity.clear(),this.wss&&(this.wss.close(),this.wss=null),this.httpsServer&&(await new Promise(e=>{this.httpsServer.close(()=>e());}),this.httpsServer=null),this.httpRedirectServer&&(await new Promise(e=>{this.httpRedirectServer.close(()=>e());}),this.httpRedirectServer=null),this.server&&(await this.server.stop(),this.server=null),this.running=false,o.info("WebSocket server stopped");}broadcastToSession(e,t){this.touchSession(e);let s=this.sessionSubscriptions.get(e);if(!s||s.size===0){o.debug("No clients subscribed to session",{sessionId:e});return}let n=t?.properties?.sessionID;if(typeof n=="string"&&n&&n!==e)return;let r=JSON.stringify({type:"event",sessionId:e,payload:t});for(let a of s){let g=this.subscriptions.get(a);if(g&&g.ws.readyState===g.ws.OPEN&&g.subscribedSessions.has(e)){if(g.subscribedEventTypes.size>0&&!g.subscribedEventTypes.has(t.type)){o.debug("Event type not subscribed, skipping",{clientId:a,sessionId:e,eventType:t.type,subscribedTypes:Array.from(g.subscribedEventTypes)});continue}g.ws.send(r),o.debug("Event pushed to client",{clientId:a,sessionId:e,eventType:t.type});}}}getConnectionCount(){return this.connections.size}getSubscriptionCount(){return this.sessionSubscriptions.size}getConnectionInfos(){let e=[];for(let[t,s]of this.subscriptions.entries())e.push({clientId:t,state:s.ws.readyState===s.ws.OPEN?"connected":"disconnected",connectedAt:s.connectedAt,lastActivityAt:s.lastActivityAt,authInfo:s.authInfo});return e}isRunning(){return this.running}handleConnection(e){if(this.connections.size>=(this.config.maxConnections??S.maxConnections)){o.warn("Maximum connections reached, rejecting new connection"),e.close(1013,"Maximum connections reached");return}let t=K();this.connections.set(t,e),this.subscriptions.set(t,{clientId:t,ws:e,subscribedSessions:new Set,subscribedEventTypes:new Set,connectedAt:Date.now(),lastActivityAt:Date.now()}),o.info("WebSocket client connected",{clientId:t,totalConnections:this.connections.size}),this.sendMessage(e,{type:"connected",clientId:t,payload:{version:"1.0.0",capabilities:["subscribe","message","interrupt","releaseSubscription"]}});}handleDisconnection(e){let t=A(e,this.connections);if(!t)return;this.connections.delete(t);let s=this.subscriptions.get(t);if(s){for(let n of s.subscribedSessions){let r=this.sessionSubscriptions.get(n);r&&(r.delete(t),r.size===0&&this.sessionSubscriptions.delete(n));}this.subscriptions.delete(t);}o.info("WebSocket client disconnected",{clientId:t,totalConnections:this.connections.size});}async handleMessage(e,t){let s=A(e,this.connections);if(!s){this.sendError(e,"Client not found");return}let n=this.subscriptions.get(s);switch(n&&(n.lastActivityAt=Date.now()),t.type){case "ping":this.sendMessage(e,{type:"pong"});break;case "initialize":await this.handleInitialize(e,t);break;case "subscribe":await this.handleSubscribe(e,t);break;case "unsubscribe":await this.handleUnsubscribe(e,t);break;case "message":await this.handleMessageRequest(e,t);break;case "interrupt":await this.handleInterrupt(e,t);break;case "releaseSubscription":await this.handleReleaseSubscription(e,t);break;default:this.sendError(e,`Unknown message type: ${t.type}`);}}async handleInitialize(e,t){this.sendMessage(e,{type:"initialize_response",success:true,result:{capabilities:["subscribe","message","interrupt","releaseSubscription"]}});}async handleSubscribe(e,t){if(t.type!=="subscribe"||!t.sessionId){this.sendError(e,"Invalid subscribe request");return}let s=A(e,this.connections);if(!s){this.sendError(e,"Client not found");return}let n=this.subscriptions.get(s);if(!n){this.sendError(e,"Subscription not found");return}if(n.subscribedSessions.add(t.sessionId),this.sessionSubscriptions.has(t.sessionId)||this.sessionSubscriptions.set(t.sessionId,new Set),this.sessionSubscriptions.get(t.sessionId).add(s),this.touchSession(t.sessionId),t.payload?.eventTypes)for(let r of t.payload.eventTypes)n.subscribedEventTypes.add(r);o.info("Client subscribed to session",{clientId:s,sessionId:t.sessionId}),this.sendMessage(e,{type:"subscribed",sessionId:t.sessionId});}async handleUnsubscribe(e,t){if(t.type!=="unsubscribe"||!t.sessionId){this.sendError(e,"Invalid unsubscribe request");return}let s=A(e,this.connections);if(!s)return;let n=this.subscriptions.get(s);if(!n)return;n.subscribedSessions.delete(t.sessionId);let r=this.sessionSubscriptions.get(t.sessionId);r&&(r.delete(s),r.size===0&&this.sessionSubscriptions.delete(t.sessionId)),o.info("Client unsubscribed from session",{clientId:s,sessionId:t.sessionId}),this.sendMessage(e,{type:"unsubscribed",sessionId:t.sessionId});}async handleMessageRequest(e,t){if(t.type!=="message"||!t.sessionId||!t.payload){this.sendError(e,"Invalid message request");return}o.debug("Message received",{sessionId:t.sessionId}),this.sendMessage(e,{type:"response",sessionId:t.sessionId,success:true,result:{received:true}});}async handleInterrupt(e,t){if(t.type!=="interrupt"||!t.sessionId){this.sendError(e,"Invalid interrupt request");return}o.info("Interrupt requested via WebSocket",{sessionId:t.sessionId}),this.sendMessage(e,{type:"interrupted",sessionId:t.sessionId,success:true});}async handleReleaseSubscription(e,t){if(t.type!=="releaseSubscription"||!t.sessionId){this.sendError(e,"Invalid releaseSubscription request");return}o.info("Release subscription requested via WebSocket",{sessionId:t.sessionId}),this.releaseSessionResources(t.sessionId),this.sendMessage(e,{type:"subscriptionReleased",sessionId:t.sessionId});}touchSession(e){this.sessionLastActivity.set(e,Date.now());}checkConnectionTimeout(){let e=Date.now(),t=this.config.connectionTimeout??S.connectionTimeout,s=[];for(let[n,r]of this.subscriptions.entries())e-r.lastActivityAt>t&&s.push(n);for(let n of s){let r=this.subscriptions.get(n);if(r){o.info("Closing inactive connection",{clientId:n,lastActivityAt:r.lastActivityAt,timeoutMs:t});try{r.ws.close(1001,"Connection timeout");}catch(a){o.debug("Error closing timed out connection",{clientId:n,error:String(a)});}}}s.length>0&&o.debug("Closed timed out connections",{count:s.length,remainingConnections:this.connections.size-s.length});}cleanupExpiredSessions(){let e=Date.now(),t=[],s=this.config.sessionExpireMs??S.sessionExpireMs;for(let[n,r]of this.sessionLastActivity.entries())e-r>s&&t.push(n);for(let n of t)o.info("Session expired due to inactivity, releasing resources",{sessionId:n,lastActivity:this.sessionLastActivity.get(n),expireMs:s}),this.releaseSessionResources(n),this.sessionLastActivity.delete(n);t.length>0&&o.debug("Cleaned up expired sessions",{count:t.length,remainingActive:this.sessionLastActivity.size});}releaseSessionResources(e){let t=this.sessionSubscriptions.get(e);if(t){for(let s of t){let n=this.subscriptions.get(s);n&&n.ws.readyState===n.ws.OPEN&&this.sendMessage(n.ws,{type:"session_closed",sessionId:e,payload:{reason:"Session resources released due to expiration or explicit release"}});}for(let s of t){let n=this.subscriptions.get(s);n&&n.subscribedSessions.delete(e);}this.sessionSubscriptions.delete(e),o.info("Session resources released",{sessionId:e,clientCount:t.size});}}sendMessage(e,t){e.readyState===e.OPEN&&e.send(JSON.stringify(t));}sendError(e,t,s){this.sendMessage(e,{type:"error",message:t,code:s});}createRoutes(){return [{method:"get",path:"/health",handler:e=>{let t={status:"healthy",timestamp:Date.now(),version:"1.0.0"};return e.json(t)}},{method:"get",path:"/status",handler:e=>{let t={status:this.running?"running":"stopped",connections:this.connections.size,sessions:this.sessionSubscriptions.size,subscriptions:this.subscriptions.size,uptime:this.running?Date.now()-this.startTime:0,version:"1.0.0"};return e.json(t)}},{method:"post",path:"/initialize",handler:async e=>{let t=await e.req.json().catch(()=>({}));return o.debug("ACP initialize request received",{body:t}),e.json({success:true,result:{capabilities:["subscribe","message","interrupt","releaseSubscription"],version:"1.0.0"}})}},{method:"post",path:"/session",handler:async e=>{await e.req.json().catch(()=>({}));let n={id:`session_${Date.now()}_${Math.random().toString(36).substring(2,9)}`,createdAt:Date.now()};return e.json(n)}},{method:"post",path:"/session/:sessionId/prompt",handler:async e=>{let t=e.req.param("sessionId");await e.req.json().catch(()=>({}));if(!t)return e.json({error:"No sessionId provided"},400);let r={messageId:`msg_${Date.now()}_${Math.random().toString(36).substring(2,9)}`,sessionId:t,createdAt:Date.now()};return e.status(202),e.json(r)}},{method:"get",path:"/subscriptions",handler:e=>{let t=Array.from(this.sessionSubscriptions.entries()).map(([n,r])=>({sessionId:n,subscriberCount:r.size,subscribers:Array.from(r)})),s={subscriptions:t,total:t.length};return e.json(s)}}]}};function J(c){let e={platform:c.channel?.platform??"api",channelId:c.channel?.channelId??"unknown",chatId:c.channel?.chatId,userId:c.channel?.userId};return {id:c.messageId??`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:c.sessionId,type:c.type,content:Array.isArray(c.content)?c.content:[{type:"text",text:String(c.content??"")}],metadata:{channel:e,replyToMessageId:c.replyToMessageId},timestamp:c.timestamp??Date.now()}}var N=class{constructor(e){a$1(this,"log",a.create({service:"gateway:server"}));a$1(this,"config");a$1(this,"router");a$1(this,"sessionManager");a$1(this,"pluginLoader");a$1(this,"messageStore");a$1(this,"wsServer",null);a$1(this,"running",false);a$1(this,"connections",0);this.config=e,this.router=new P,this.sessionManager=new C,this.pluginLoader=new w({healthCheckInterval:6e4});let t=S$1(),s=P$1();this.messageStore=new b({storagePath:`${t.Path.data}/gateway-messages`}),s&&Q$1().setMessageStorage({store:async r=>{let a=J(r);await this.messageStore.store(a);},updateStatus:async(r,a,g)=>{await this.messageStore.updateStatus(r,a,g);}}),this.pluginLoader.setMessageHandler({onMessage:n=>this.onMessage(n),onEvent:n=>this.onEvent(n)});}async start(){if(this.running){this.log.warn("gateway server is already running");return}this.log.info("starting gateway server",{port:this.config.port,hostname:this.config.hostname}),await this.sessionManager.initialize(),this.log.debug("starting WebSocket server"),this.wsServer=new M({port:this.config.port,hostname:this.config.hostname,path:String(this.config.path??"/")}),await this.wsServer.start(),this.log.debug("WebSocket server started"),await this.pluginLoader.startAll(),this.pluginLoader.startHealthCheckTimer(),this.running=true,this.log.info("gateway server started");}async stop(){if(!this.running){this.log.warn("gateway server is not running");return}this.log.info("stopping gateway server"),this.wsServer&&(this.log.debug("stopping WebSocket server"),await this.wsServer.stop(),this.wsServer=null,this.log.debug("WebSocket server stopped")),this.pluginLoader.stopHealthCheckTimer(),await this.pluginLoader.stopAll(),await this.router.shutdown(),await this.sessionManager.close(),this.running=false,this.log.info("gateway server stopped");}getStatus(){return {running:this.running,port:this.config.port,hostname:this.config.hostname,connections:this.connections}}async onMessage(e){if(this.log.debug("message received",{id:e.id,sessionId:e.sessionId,type:e.type}),this.messageStore.isProcessed(e.id)){this.log.warn("message already processed, skipping",{id:e.id});return}await this.storeMessage(e),await this.sessionManager.incrementMessageCountOrCreate(e.sessionId,e.metadata.channel),(await this.router.route(e)).broadcastCount===0&&e.type==="input"&&this.log.debug("no subscribers, message pending",{id:e.id});}async storeMessage(e){if(await this.messageStore.store(e),P$1())try{let t=Q$1(),s={name:HookEvent.GatewayMessageReceive,data:{messageId:e.id,sessionId:e.sessionId,type:e.type,content:e.content,channel:e.metadata.channel,replyToMessageId:e.metadata.replyToMessageId,timestamp:e.timestamp||Date.now()}};await t.hookRegistry.emit(s);}catch{this.log.debug("hook trigger skipped (no context)");}}async processMessage(e,t,s,n={}){if(await this.messageStore.updateStatus(t,"processing",{processedBy:s}),P$1())try{await Q$1().hookRegistry.emit({name:HookEvent.GatewayMessageProcess,data:{messageId:t,sessionId:e,processorId:s,processType:n.processType||"direct",agent:n.agent,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}}async completeMessage(e,t,s,n={}){let{success:r=true,error:a,responseMessageId:g,tokens:l,duration:p}=n;if(await this.messageStore.updateStatus(t,r?"completed":"failed",{processedBy:s,error:a}),P$1())try{await Q$1().hookRegistry.emit({name:HookEvent.GatewayMessageComplete,data:{messageId:t,sessionId:e,processorId:s,status:r?"completed":"failed",error:a,responseMessageId:g,tokens:l,duration:p,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}await this.router.completeMessage(e,t,s,r?"completed":"failed");}async onEvent(e){switch(this.log.debug("channel event",{type:e.type,channelId:e.channelId}),e.type){case "connected":this.connections++;break;case "disconnected":this.connections--;break;case "error":this.log.error("channel error",{channelId:e.channelId,data:e.data});break}}async registerChannel(e,t){let s=t??{enabled:true,platform:{},session:{autoCreate:true,resetPolicy:"onNewConversation"},routing:{broadcast:true,dispatchToSubAgent:false}};await this.pluginLoader.register(e,s),this.running&&await this.pluginLoader.start(e.id);}async unregisterChannel(e){await this.pluginLoader.unregister(e);}async subscribe(e,t){await this.router.subscribe(e,t),await this.sessionManager.addSubscriber(e,t.id);}async unsubscribe(e){await this.router.unsubscribe(e);}async send(e){await this.onMessage(e);}async sendToChannel(e,t,s={}){let{proactive:n=false,replyToMessageId:r}=s;if(P$1())try{await Q$1().hookRegistry.emit({name:HookEvent.GatewayMessageSend,data:{messageId:t.id,sessionId:t.sessionId,type:t.type,channel:t.metadata.channel,proactive:n,replyToMessageId:r,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}return this.pluginLoader.sendToPlugin(e,t)}async sendProactiveMessage(e,t,s,n){let r={id:`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:t,type:"output",content:s,metadata:{channel:n},timestamp:Date.now()};return this.sendToChannel(e,r,{proactive:true})}getSessionManager(){return this.sessionManager}getRouter(){return this.router}getPluginLoader(){return this.pluginLoader}getMessageStore(){return this.messageStore}async getMessageHistory(e,t){return this.messageStore.getSessionHistory(e,t)}};
2
+ export{I as a,k as b,L as c,B as d,R as e,E as f,x as g,D as h,q as i,F as j,U as k,$ as l,P as m,b as n,v as o,C as p,w as q,M as r,N as s};
@@ -0,0 +1,2 @@
1
+ 'use strict';var chunk5ETPYYQR_cjs=require('./chunk-5ETPYYQR.cjs'),e=require('zod'),_=require('path'),fs=require('fs'),be=require('fs/promises');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var e__default=/*#__PURE__*/_interopDefault(e);var ___default=/*#__PURE__*/_interopDefault(_);var be__default=/*#__PURE__*/_interopDefault(be);var Y=chunk5ETPYYQR_cjs.a.create({service:"gateway:config"});e__default.default.enum(["telegram","slack","feishu","wechat","webchat","signal","nostr"]);e__default.default.enum(["low","medium","high","critical"]);var m=e__default.default.object({enabled:e__default.default.boolean().default(true),botToken:e__default.default.string().describe("Telegram Bot Token"),webhookUrl:e__default.default.string().optional().describe("Webhook URL for receiving updates"),maxConnections:e__default.default.number().int().positive().optional().default(100),timeout:e__default.default.number().int().positive().optional().default(3e4)}),y=e__default.default.object({enabled:e__default.default.boolean().default(true),botToken:e__default.default.string().describe("Slack Bot Token (xoxb-*)"),appToken:e__default.default.string().optional().describe("Slack App Token (xapp-*) for Socket Mode"),signingSecret:e__default.default.string().optional().describe("Slack Signing Secret for webhook verification"),maxConnections:e__default.default.number().int().positive().optional().default(100),timeout:e__default.default.number().int().positive().optional().default(3e4)}),b=e__default.default.object({enabled:e__default.default.boolean().default(true),appId:e__default.default.string().describe("Feishu App ID"),appSecret:e__default.default.string().describe("Feishu App Secret"),encryptKey:e__default.default.string().optional().describe("Feishu Encrypt Key for message encryption"),verificationToken:e__default.default.string().optional().describe("Feishu Verification Token"),maxConnections:e__default.default.number().int().positive().optional().default(100),timeout:e__default.default.number().int().positive().optional().default(3e4)}),S=e__default.default.object({enabled:e__default.default.boolean().default(true),mode:e__default.default.enum(["official","wecom"]).default("official"),appId:e__default.default.string().describe("WeChat App ID"),appSecret:e__default.default.string().describe("WeChat App Secret"),token:e__default.default.string().optional().describe("WeChat Token for message verification"),encodingAESKey:e__default.default.string().optional().describe("WeChat Encoding AES Key"),agentId:e__default.default.string().optional().describe("WeCom Agent ID"),maxConnections:e__default.default.number().int().positive().optional().default(100),timeout:e__default.default.number().int().positive().optional().default(3e4)}),w=e__default.default.object({enabled:e__default.default.boolean().default(true),path:e__default.default.string().optional().default("/"),heartbeatInterval:e__default.default.number().int().positive().optional().default(3e4),connectionTimeout:e__default.default.number().int().positive().optional().default(6e4),maxConnections:e__default.default.number().int().positive().optional().default(1e3),cors:e__default.default.boolean().optional().default(true)}),A=e__default.default.object({enabled:e__default.default.boolean().default(true),serviceUrl:e__default.default.string().optional().default("https://chat.signal.org"),phoneNumber:e__default.default.string().describe("Signal phone number"),password:e__default.default.string().optional().describe("Signal account password"),maxConnections:e__default.default.number().int().positive().optional().default(100),timeout:e__default.default.number().int().positive().optional().default(3e4)}),x=e__default.default.object({enabled:e__default.default.boolean().default(true),relays:e__default.default.array(e__default.default.string()).min(1).describe("Nostr relay URLs"),privateKey:e__default.default.string().optional().describe("Nostr private key (hex)"),publicKey:e__default.default.string().optional().describe("Nostr public key (hex)"),maxConnections:e__default.default.number().int().positive().optional().default(100),timeout:e__default.default.number().int().positive().optional().default(3e4)}),O=e__default.default.object({telegram:m.optional(),slack:y.optional(),feishu:b.optional(),wechat:S.optional(),webchat:w.optional(),signal:A.optional(),nostr:x.optional()}),D=e__default.default.object({enabled:e__default.default.boolean().default(false),cert:e__default.default.string().describe("Path to SSL certificate file (\u6216 certPath)"),key:e__default.default.string().describe("Path to SSL key file (\u6216 keyPath)"),certPath:e__default.default.string().optional().describe("Path to SSL certificate file (alias of cert)"),keyPath:e__default.default.string().optional().describe("Path to SSL key file (alias of key)"),ca:e__default.default.string().optional().describe("Path to CA certificate file"),forceRedirect:e__default.default.boolean().optional().default(false).describe("Force redirect HTTP to HTTPS"),httpPort:e__default.default.number().int().positive().optional().default(80).describe("HTTP redirect port")}),J=e__default.default.object({name:e__default.default.string().describe("Token name/description"),value:e__default.default.string().describe("Token value"),permissions:e__default.default.array(e__default.default.string()).default(["*"]).describe("Associated permissions"),expiresAt:e__default.default.number().int().positive().optional().describe("Expiration timestamp (Unix timestamp)"),description:e__default.default.string().optional().describe("Token description")}),$=e__default.default.object({enabled:e__default.default.boolean().default(true),tokens:e__default.default.array(J).default([]).describe("List of valid tokens"),onInvalid:e__default.default.enum(["reject","anonymous"]).default("reject").describe("Behavior when token validation fails"),allowAnonymous:e__default.default.boolean().default(false).describe("Allow anonymous access"),defaultPermissions:e__default.default.array(e__default.default.string()).default([]).describe("Default permissions for anonymous users")}),X=e__default.default.object({enabled:e__default.default.boolean().default(false),authorizationServer:e__default.default.string().describe("OAuth2 authorization server URL"),clientId:e__default.default.string().describe("OAuth2 client ID"),clientSecret:e__default.default.string().describe("OAuth2 client secret"),tokenValidationEndpoint:e__default.default.string().describe("OAuth2 token validation endpoint"),scope:e__default.default.array(e__default.default.string()).default([]).describe("OAuth2 scopes")}),j=e__default.default.object({type:e__default.default.enum(["token","oauth2"]).default("token"),token:$.optional(),oauth2:X.optional()}),z=e__default.default.object({mode:e__default.default.enum(["pull","push","both"]).default("both"),interval:e__default.default.number().int().positive().default(3e4),pushEvents:e__default.default.array(e__default.default.enum(["register","deregister","heartbeat","status_change"])).default(["register","deregister","status_change"]),conflictResolution:e__default.default.enum(["latest","local","remote"]).default("latest")}),Z=e__default.default.object({id:e__default.default.string().describe("Gateway node ID"),address:e__default.default.string().describe("Gateway node address (e.g., http://localhost:3001)"),enabled:e__default.default.boolean().default(true),metadata:e__default.default.record(e__default.default.string(),e__default.default.any()).optional()}),U=e__default.default.object({nodeId:e__default.default.string().describe("Current gateway node ID"),nodes:e__default.default.array(Z).default([]).describe("Other gateway nodes in the cluster"),sync:z.optional()}),v=e__default.default.object({maxConnectionsPerChannel:e__default.default.number().int().positive().default(100),idleTimeout:e__default.default.number().int().positive().default(3e5),reuseStrategy:e__default.default.enum(["lru","fifo","random"]).default("lru"),healthCheckInterval:e__default.default.number().int().positive().default(6e4),acquireTimeout:e__default.default.number().int().positive().default(5e3),warmupCount:e__default.default.number().int().nonnegative().default(0).describe("Number of connections to warm up")}),T=e__default.default.object({maxLength:e__default.default.number().int().positive().default(1e4),concurrency:e__default.default.number().int().positive().default(100),batchSize:e__default.default.number().int().positive().default(10),batchTimeout:e__default.default.number().int().positive().default(100),priorityLevels:e__default.default.number().int().positive().default(4)}),G=e__default.default.object({failureThreshold:e__default.default.number().int().positive().default(5),successThreshold:e__default.default.number().int().positive().default(3),timeout:e__default.default.number().int().positive().default(3e4),halfOpenDuration:e__default.default.number().int().positive().default(15e3)}),P=e__default.default.object({heartbeatInterval:e__default.default.number().int().positive().default(3e4),heartbeatTimeout:e__default.default.number().int().positive().default(9e4),heartbeatCheckInterval:e__default.default.number().int().positive().default(1e4),autoDeregisterTimeout:e__default.default.number().int().positive().default(3e5),maxAgents:e__default.default.number().int().positive().default(100),persistenceFile:e__default.default.string().optional().describe("Path to persist agent registrations"),maxRegistrations:e__default.default.number().int().positive().default(1e3)}),k=e__default.default.object({sessionTimeout:e__default.default.number().int().positive().default(18e5),resumeGracePeriod:e__default.default.number().int().positive().default(3e5),maxSessions:e__default.default.number().int().positive().default(1e4),persistenceFile:e__default.default.string().optional().describe("Path to persist sessions")}),I=e__default.default.object({enabled:e__default.default.boolean().default(true),port:e__default.default.number().int().positive().default(8080),hostname:e__default.default.string().default("localhost"),path:e__default.default.string().optional().default("/"),maxConnections:e__default.default.number().int().positive().optional().default(1e3),connectionTimeout:e__default.default.number().int().positive().optional().default(6e4),heartbeatInterval:e__default.default.number().int().positive().optional().default(3e4),sessionExpireMs:e__default.default.number().int().positive().optional().default(3e5),https:D.optional(),auth:j.optional(),cors:e__default.default.boolean().default(true),corsOrigins:e__default.default.array(e__default.default.string()).optional(),mdns:e__default.default.boolean().default(false),mdnsDomain:e__default.default.string().optional().default("gateway.easbot.local")}),d=e__default.default.object({server:I.optional(),channels:O.optional(),cluster:U.optional(),connectionPool:v.optional(),messageQueue:T.optional(),circuitBreaker:G.optional(),agentRegistry:P.optional(),session:k.optional(),defaultAgent:e__default.default.string().optional().describe("Default agent ID for routing"),logLevel:e__default.default.enum(["DEBUG","INFO","WARN","ERROR"]).optional().default("INFO")}).catchall(e__default.default.unknown());function N(t){let n=d.safeParse(t);return n.success||Y.warn("Invalid gateway config",{issues:n.error.issues}),n.success?n.data:d.parse({})}function ee(t){return d.parse(t)}var Ce=chunk5ETPYYQR_cjs.a.create({service:"gateway:interfaces"}),f;function Je(t){f=t,Ce.debug("Agent adapter registered",{hasSubAgentRunner:!!t.subAgentRunner});}function $e(){return f}function Q(){return f!==void 0}function K(){if(!f)throw new Error("Agent adapter not set. Call setAgentAdapter() before using Gateway.");return f}var he={Path:{home:process.env.HOME||process.env.USERPROFILE||"",data:".easbot",cache:".easbot/cache",config:".easbot",state:".easbot/state",log:".easbot/log",bin:".easbot/bin"}},de={directory:process.cwd(),worktree:process.cwd()},F=class{async load(n){return E(n)}},me=new F;function Xe(){return f?f.configLoader:me}function q(){return f?f.global:he}function R(){return f?f.instance:de}var l=chunk5ETPYYQR_cjs.a.create({service:"gateway:config"}),W={enabled:true,port:8080,hostname:"localhost",path:"/",maxConnections:1e3,connectionTimeout:6e4,heartbeatInterval:3e4,sessionExpireMs:3e5,cors:true,mdns:false,mdnsDomain:"gateway.easbot.local",https:{enabled:false,cert:"",key:"",certPath:void 0,keyPath:void 0,ca:void 0,forceRedirect:false,httpPort:80}},Se={maxConnectionsPerChannel:100,idleTimeout:3e5,reuseStrategy:"lru",healthCheckInterval:6e4,acquireTimeout:5e3,warmupCount:0},we={maxLength:1e4,concurrency:100,batchSize:10,batchTimeout:100,priorityLevels:4},Ae={failureThreshold:5,successThreshold:3,timeout:3e4,halfOpenDuration:15e3},xe={heartbeatInterval:3e4,heartbeatTimeout:9e4,heartbeatCheckInterval:1e4,autoDeregisterTimeout:3e5,maxAgents:100,maxRegistrations:1e3},ve={sessionTimeout:18e5,resumeGracePeriod:3e5,maxSessions:1e4},Te={enabled:true,botToken:"",maxConnections:100,timeout:3e4},Ge={enabled:true,botToken:"",maxConnections:100,timeout:3e4},Pe={enabled:true,appId:"",appSecret:"",maxConnections:100,timeout:3e4},ke={enabled:true,mode:"official",appId:"",appSecret:"",maxConnections:100,timeout:3e4},Ie={enabled:true,path:"/ws",heartbeatInterval:3e4,connectionTimeout:6e4,maxConnections:1e3,cors:true},Ne={enabled:true,serviceUrl:"https://chat.signal.org",phoneNumber:"",maxConnections:100,timeout:3e4},Le={enabled:true,relays:[],maxConnections:100,timeout:3e4},C,p=null;async function V(t){try{if(!fs.existsSync(t))return null;let n=await be__default.default.readFile(t,"utf-8"),o=n,a=!1,r=!1,s=[];for(let i=0;i<n.length;i++){let g=n[i];if(r){s.push(g),r=!1;continue}if(g==="\\"){s.push(g),r=!0;continue}if(g==='"'){a=!a,s.push(g);continue}if(!a&&g==="/"&&n[i+1]==="/"){for(;i<n.length&&n[i]!==`
2
+ `&&n[i]!=="\r";)i++;continue}if(!a&&g==="/"&&n[i+1]==="*"){for(i+=2;i<n.length-1;){if(n[i]==="*"&&n[i+1]==="/"){i+=2;break}i++;}continue}s.push(g);}return o=s.join("").trim(),o?JSON.parse(o):null}catch(n){return l.debug("Failed to load config file",{file:t,error:String(n)}),null}}function h(t,n){let o={...t,...n};return (t.server||n.server)&&(o.server={...t.server,...n.server}),(t.channels||n.channels)&&(o.channels={...t.channels,...n.channels}),o}async function Ee(){let t=q(),n=R(),o=[],a=___default.default.join(t.Path.config);o.push(a),o.push(n.directory);try{let{Filesystem:s}=await import('@easbot/utils'),i=await Array.fromAsync(s.up({targets:[".easbot"],start:n.directory,stop:n.worktree}));o.push(...i);}catch{l.debug("Filesystem utils not available, using basic directories");}let r=process.env.EASBOT_CONFIG_DIR;return r&&(o.push(r),l.debug("Using EASBOT_CONFIG_DIR",{path:r})),[...new Set(o)]}async function Fe(t){let n=["gateway.json"],o={};for(let a of n){let r=___default.default.join(t,a),s=await V(r);if(s)if("gateway"in s&&s.gateway){let i=s.gateway;"server"in i||"channels"in i||"cluster"in i?o=h(o,i):o=h(o,{server:i}),l.debug("Loaded gateway config from file (nested format)",{file:r});}else "gateway"in s||(o=h(o,s),l.debug("Loaded gateway config from file (direct format)",{file:r}));}return o}async function Re(t){let n=["easbot.json"];for(let o of n){let a=___default.default.join(t,o),r=await V(a);if(r?.gateway)return l.debug("Loaded gateway config from easbot.json",{file:a}),r.gateway}return {}}function _e(){return C||(C=async()=>{if(Q())try{let a=K(),r=await a.configLoader.load(a.instance.directory);if(r&&typeof r=="object"){let s=r.gateway;if(s){l.debug("Loaded gateway config from agent adapter");let i=N(s);return p=i,{config:i,directories:Array.from(new Set([a.instance.directory,a.instance.worktree,a.global.Path.config]).values())}}}}catch(a){l.debug("Failed to load config from agent adapter",{error:String(a)});}let t=await Ee(),n={};for(let a of t){let r=await Fe(a);n=h(n,r);let s=await Re(a);n=h(n,s);}let o=N(n);return p=o,l.info("Gateway config loaded",{directories:t.length,port:o.server?.port,logLevel:o.logLevel}),{config:o,directories:t}}),C}async function c(){return _e()()}async function E(t){return t&&(C=void 0,p=null),(await c()).config}function H(){return p}function B(){return p!==null}function te(){C=void 0,p=null,l.debug("Config cache cleared");}function ne(){return R().directory}async function oe(){let{config:t}=await c();return I.parse({...W,...t.server})}async function ie(t){let{config:n}=await c(),o=n.channels?.[t],r={telegram:m.parse(Te),slack:y.parse(Ge),feishu:b.parse(Pe),wechat:S.parse(ke),webchat:w.parse(Ie),signal:A.parse(Ne),nostr:x.parse(Le)}[t];if(!r)throw new Error(`Unknown channel type: ${t}`);return {...r,...o}}async function ae(){let{config:t}=await c();return v.parse({...Se,...t.connectionPool})}async function re(){let{config:t}=await c();return T.parse({...we,...t.messageQueue})}async function se(){let{config:t}=await c();return G.parse({...Ae,...t.circuitBreaker})}async function ce(){let{config:t}=await c();return P.parse({...xe,...t.agentRegistry})}async function fe(){let{config:t}=await c();return k.parse({...ve,...t.session})}async function le(){let{config:t}=await c();return t.server?.https}async function ge(){let{config:t}=await c();return t.server?.auth}async function ue(){let{config:t}=await c();return t.cluster}async function pe(){let{config:t}=await c();return t.defaultAgent}async function M(){let{config:t}=await c();return t.logLevel??"INFO"}exports.A=ne;exports.B=oe;exports.C=ie;exports.D=ae;exports.E=re;exports.F=se;exports.G=ce;exports.H=fe;exports.I=le;exports.J=ge;exports.K=ue;exports.L=pe;exports.M=M;exports.N=Je;exports.O=$e;exports.P=Q;exports.Q=K;exports.R=Xe;exports.S=q;exports.T=R;exports.a=m;exports.b=y;exports.c=b;exports.d=S;exports.e=w;exports.f=A;exports.g=x;exports.h=O;exports.i=D;exports.j=j;exports.k=z;exports.l=U;exports.m=v;exports.n=T;exports.o=G;exports.p=P;exports.q=k;exports.r=I;exports.s=d;exports.t=N;exports.u=ee;exports.v=W;exports.w=E;exports.x=H;exports.y=B;exports.z=te;
@@ -1 +1 @@
1
- import {P,R}from'./chunk-YU4JG7OA.mjs';import {a as a$1}from'./chunk-6TRL3CVJ.mjs';import {a as a$2}from'./chunk-HAMGVOQD.mjs';var a=a$1.create({service:"gateway"}),c=class t extends Error{constructor(r,y,m){super(r);a$2(this,"type");a$2(this,"stage");a$2(this,"cause");this.type="GatewayInitializationError",this.stage=y,this.cause=m,Error.captureStackTrace&&Error.captureStackTrace(this,t);}};function E(t){let s=["ECONNRESET","EPIPE","ETIMEDOUT","ECONNREFUSED","ENOTFOUND","EHOSTUNREACH","ENETUNREACH"],r=t.code;return s.includes(r??"")}function C(){process.on("unhandledRejection",t=>{let s=t instanceof Error?t:new Error(String(t));E(s)||a.error("unhandled promise rejection",{error:s.message,stack:s.stack});}),process.on("uncaughtException",t=>{E(t)||(a.error("uncaught exception",{error:t.message,stack:t.stack}),process.exit(1));});}var h;(k=>{let t={initialized:false,initPromise:null,directory:".easbot",config:null},s,r={server:null,config:null,initialized:false,status:"stopped"};async function y(o={}){if(t.initialized)return;if(t.initPromise)return t.initPromise;let i={directory:o.directory??".easbot",printLogs:o.printLogs??false,logLevel:o.logLevel??"INFO"};return t.directory=i.directory,t.initPromise=(async()=>{let{Global:n}=await import('./global-ZNJCD2DC.mjs');try{await n.init();}catch(e){throw new c("Failed to initialize global directories","global",e instanceof Error?e:void 0)}try{await a$1.init({logDir:n.Path.log,print:i.printLogs,level:i.logLevel});}catch(e){throw new c("Failed to initialize log infrastructure","log",e instanceof Error?e:void 0)}try{C();}catch(e){throw new c("Failed to install unhandled exception handlers","server",e instanceof Error?e:void 0)}try{let e=P();t.config=await e.load(i.directory);}catch(e){throw new c("Failed to load gateway configuration","config",e instanceof Error?e:void 0)}t.initialized=true,a.info("gateway runtime initialized",{directory:i.directory,logLevel:i.logLevel});})(),t.initPromise}k.init=y;function m(){return t.initialized}k.isInitialized=m;function S(){return s||(s=async()=>{if(!t.initialized)throw new c("Gateway not initialized. Call init() first.","server");let o=P(),i=R(),n=await o.load(i.directory);return n.server?.enabled?(r={server:null,config:n,initialized:true,status:"stopped"},a.info("gateway initialized",{port:n.server?.port,hostname:n.server?.hostname}),r):(a.debug("gateway server disabled or not configured"),{server:null,config:n,initialized:true,status:"stopped"})}),s}function u(){return S()()}k.state=u;async function z(){return (await u()).server}k.get=z;async function R$1(){return (await u()).config?.server?.enabled??false}k.isEnabled=R$1;async function L(){return (await u()).config}k.config=L;function N(){return r.status}k.getStatus=N;async function b(o){t.initialized||await y();let i=await u();if(!(i.config?.server?.enabled??true))throw a.warn("gateway server is disabled"),new Error("Gateway server is disabled");if(r.status==="running"){a.info("gateway server is already running");return}if(r.status==="starting"){a.info("gateway server is starting");return}r.status="starting";try{let e={...i.config?.server,...o};a.debug("gateway start: creating server with config",{port:e.port,hostname:e.hostname,path:e.path,https:e.https?.enabled});let g=await G(e);a.debug("gateway start: server created, updating state"),r.server=g,r.status="running",r.error=void 0,a.info("gateway server started",{port:e.port,hostname:e.hostname});}catch(e){throw r.status="error",r.error=e instanceof Error?e.message:String(e),a.error("failed to start gateway server",{error:r.error}),e}}k.start=b;async function P$1(){if(r.status==="stopped"){a.info("gateway server is already stopped");return}if(r.status==="stopping"){a.info("gateway server is stopping");return}r.status="stopping";try{r.server&&await r.server.stop(),r.server=null,r.status="stopped",r.error=void 0,a.info("gateway server stopped");}catch(o){throw r.status="error",r.error=o instanceof Error?o.message:String(o),a.error("failed to stop gateway server",{error:r.error}),o}}k.stop=P$1;async function T(o){a.info("restarting gateway server"),(r.status==="running"||r.status==="starting")&&await P$1(),await b(o);}k.restart=T;async function I(){let o=P(),i=R(),n=await o.load(i.directory);return r.config=n,t.config=n,a.info("gateway config reloaded"),n}k.reloadConfig=I;async function G(o,i){let{GatewayServer:n}=await import('./server-C5W7D4CU.mjs'),e=new n(o);a.debug("createGatewayServer: created server instance, calling start()");let g=i?.startupTimeout??3e4;if(await(async()=>{let l=new Promise((d,x)=>{setTimeout(()=>{x(new Error(`Gateway server startup timeout after ${g}ms`));},g);});try{a.debug("createGatewayServer: calling server.start()"),await Promise.race([e.start(),l]),a.debug("createGatewayServer: server.start() completed");}catch(d){a.error("gateway server start failed, attempting cleanup",{error:d instanceof Error?d.message:String(d)});try{await e.stop();}catch{}throw d}})(),i?.onStarted)try{await i.onStarted(e);}catch(l){a.warn("server started but onStarted callback failed",{error:l instanceof Error?l.message:String(l)});}return a.info("gateway server created and started",{port:o.port,hostname:o.hostname}),e}k.createGatewayServer=G;})(h||(h={}));export{h as a};
1
+ import {R,T}from'./chunk-LFGRYJV2.mjs';import {a as a$1}from'./chunk-6TRL3CVJ.mjs';import {a as a$2}from'./chunk-HAMGVOQD.mjs';var a=a$1.create({service:"gateway"}),c=class t extends Error{constructor(r,y,m){super(r);a$2(this,"type");a$2(this,"stage");a$2(this,"cause");this.type="GatewayInitializationError",this.stage=y,this.cause=m,Error.captureStackTrace&&Error.captureStackTrace(this,t);}};function E(t){let s=["ECONNRESET","EPIPE","ETIMEDOUT","ECONNREFUSED","ENOTFOUND","EHOSTUNREACH","ENETUNREACH"],r=t.code;return s.includes(r??"")}function C(){process.on("unhandledRejection",t=>{let s=t instanceof Error?t:new Error(String(t));E(s)||a.error("unhandled promise rejection",{error:s.message,stack:s.stack});}),process.on("uncaughtException",t=>{E(t)||(a.error("uncaught exception",{error:t.message,stack:t.stack}),process.exit(1));});}var h;(k=>{let t={initialized:false,initPromise:null,directory:".easbot",config:null},s,r={server:null,config:null,initialized:false,status:"stopped"};async function y(o={}){if(t.initialized)return;if(t.initPromise)return t.initPromise;let i={directory:o.directory??".easbot",printLogs:o.printLogs??false,logLevel:o.logLevel??"INFO"};return t.directory=i.directory,t.initPromise=(async()=>{let{Global:n}=await import('./global-XNAA35J4.mjs');try{await n.init();}catch(e){throw new c("Failed to initialize global directories","global",e instanceof Error?e:void 0)}try{await a$1.init({logDir:n.Path.log,print:i.printLogs,level:i.logLevel});}catch(e){throw new c("Failed to initialize log infrastructure","log",e instanceof Error?e:void 0)}try{C();}catch(e){throw new c("Failed to install unhandled exception handlers","server",e instanceof Error?e:void 0)}try{let e=R();t.config=await e.load(i.directory);}catch(e){throw new c("Failed to load gateway configuration","config",e instanceof Error?e:void 0)}t.initialized=true,a.info("gateway runtime initialized",{directory:i.directory,logLevel:i.logLevel});})(),t.initPromise}k.init=y;function m(){return t.initialized}k.isInitialized=m;function S(){return s||(s=async()=>{if(!t.initialized)throw new c("Gateway not initialized. Call init() first.","server");let o=R(),i=T(),n=await o.load(i.directory);return n.server?.enabled?(r={server:null,config:n,initialized:true,status:"stopped"},a.info("gateway initialized",{port:n.server?.port,hostname:n.server?.hostname}),r):(a.debug("gateway server disabled or not configured"),{server:null,config:n,initialized:true,status:"stopped"})}),s}function u(){return S()()}k.state=u;async function z(){return (await u()).server}k.get=z;async function R$1(){return (await u()).config?.server?.enabled??false}k.isEnabled=R$1;async function L(){return (await u()).config}k.config=L;function N(){return r.status}k.getStatus=N;async function b(o){t.initialized||await y();let i=await u();if(!(i.config?.server?.enabled??true))throw a.warn("gateway server is disabled"),new Error("Gateway server is disabled");if(r.status==="running"){a.info("gateway server is already running");return}if(r.status==="starting"){a.info("gateway server is starting");return}r.status="starting";try{let e={...i.config?.server,...o};a.debug("gateway start: creating server with config",{port:e.port,hostname:e.hostname,path:e.path,https:e.https?.enabled});let g=await G(e);a.debug("gateway start: server created, updating state"),r.server=g,r.status="running",r.error=void 0,a.info("gateway server started",{port:e.port,hostname:e.hostname});}catch(e){throw r.status="error",r.error=e instanceof Error?e.message:String(e),a.error("failed to start gateway server",{error:r.error}),e}}k.start=b;async function P(){if(r.status==="stopped"){a.info("gateway server is already stopped");return}if(r.status==="stopping"){a.info("gateway server is stopping");return}r.status="stopping";try{r.server&&await r.server.stop(),r.server=null,r.status="stopped",r.error=void 0,a.info("gateway server stopped");}catch(o){throw r.status="error",r.error=o instanceof Error?o.message:String(o),a.error("failed to stop gateway server",{error:r.error}),o}}k.stop=P;async function T$1(o){a.info("restarting gateway server"),(r.status==="running"||r.status==="starting")&&await P(),await b(o);}k.restart=T$1;async function I(){let o=R(),i=T(),n=await o.load(i.directory);return r.config=n,t.config=n,a.info("gateway config reloaded"),n}k.reloadConfig=I;async function G(o,i){let{GatewayServer:n}=await import('./server-MEM7CIBN.mjs'),e=new n(o);a.debug("createGatewayServer: created server instance, calling start()");let g=i?.startupTimeout??3e4;if(await(async()=>{let l=new Promise((d,x)=>{setTimeout(()=>{x(new Error(`Gateway server startup timeout after ${g}ms`));},g);});try{a.debug("createGatewayServer: calling server.start()"),await Promise.race([e.start(),l]),a.debug("createGatewayServer: server.start() completed");}catch(d){a.error("gateway server start failed, attempting cleanup",{error:d instanceof Error?d.message:String(d)});try{await e.stop();}catch{}throw d}})(),i?.onStarted)try{await i.onStarted(e);}catch(l){a.warn("server started but onStarted callback failed",{error:l instanceof Error?l.message:String(l)});}return a.info("gateway server created and started",{port:o.port,hostname:o.hostname}),e}k.createGatewayServer=G;})(h||(h={}));export{h as a};
@@ -0,0 +1,2 @@
1
+ import {a}from'./chunk-6TRL3CVJ.mjs';import e from'zod';import O from'path';import {existsSync}from'fs';import Se from'fs/promises';var J=a.create({service:"gateway:config"});e.enum(["telegram","slack","feishu","wechat","webchat","signal","nostr"]);e.enum(["low","medium","high","critical"]);var m=e.object({enabled:e.boolean().default(true),botToken:e.string().describe("Telegram Bot Token"),webhookUrl:e.string().optional().describe("Webhook URL for receiving updates"),maxConnections:e.number().int().positive().optional().default(100),timeout:e.number().int().positive().optional().default(3e4)}),y=e.object({enabled:e.boolean().default(true),botToken:e.string().describe("Slack Bot Token (xoxb-*)"),appToken:e.string().optional().describe("Slack App Token (xapp-*) for Socket Mode"),signingSecret:e.string().optional().describe("Slack Signing Secret for webhook verification"),maxConnections:e.number().int().positive().optional().default(100),timeout:e.number().int().positive().optional().default(3e4)}),b=e.object({enabled:e.boolean().default(true),appId:e.string().describe("Feishu App ID"),appSecret:e.string().describe("Feishu App Secret"),encryptKey:e.string().optional().describe("Feishu Encrypt Key for message encryption"),verificationToken:e.string().optional().describe("Feishu Verification Token"),maxConnections:e.number().int().positive().optional().default(100),timeout:e.number().int().positive().optional().default(3e4)}),S=e.object({enabled:e.boolean().default(true),mode:e.enum(["official","wecom"]).default("official"),appId:e.string().describe("WeChat App ID"),appSecret:e.string().describe("WeChat App Secret"),token:e.string().optional().describe("WeChat Token for message verification"),encodingAESKey:e.string().optional().describe("WeChat Encoding AES Key"),agentId:e.string().optional().describe("WeCom Agent ID"),maxConnections:e.number().int().positive().optional().default(100),timeout:e.number().int().positive().optional().default(3e4)}),w=e.object({enabled:e.boolean().default(true),path:e.string().optional().default("/"),heartbeatInterval:e.number().int().positive().optional().default(3e4),connectionTimeout:e.number().int().positive().optional().default(6e4),maxConnections:e.number().int().positive().optional().default(1e3),cors:e.boolean().optional().default(true)}),A=e.object({enabled:e.boolean().default(true),serviceUrl:e.string().optional().default("https://chat.signal.org"),phoneNumber:e.string().describe("Signal phone number"),password:e.string().optional().describe("Signal account password"),maxConnections:e.number().int().positive().optional().default(100),timeout:e.number().int().positive().optional().default(3e4)}),x=e.object({enabled:e.boolean().default(true),relays:e.array(e.string()).min(1).describe("Nostr relay URLs"),privateKey:e.string().optional().describe("Nostr private key (hex)"),publicKey:e.string().optional().describe("Nostr public key (hex)"),maxConnections:e.number().int().positive().optional().default(100),timeout:e.number().int().positive().optional().default(3e4)}),D=e.object({telegram:m.optional(),slack:y.optional(),feishu:b.optional(),wechat:S.optional(),webchat:w.optional(),signal:A.optional(),nostr:x.optional()}),j=e.object({enabled:e.boolean().default(false),cert:e.string().describe("Path to SSL certificate file (\u6216 certPath)"),key:e.string().describe("Path to SSL key file (\u6216 keyPath)"),certPath:e.string().optional().describe("Path to SSL certificate file (alias of cert)"),keyPath:e.string().optional().describe("Path to SSL key file (alias of key)"),ca:e.string().optional().describe("Path to CA certificate file"),forceRedirect:e.boolean().optional().default(false).describe("Force redirect HTTP to HTTPS"),httpPort:e.number().int().positive().optional().default(80).describe("HTTP redirect port")}),$=e.object({name:e.string().describe("Token name/description"),value:e.string().describe("Token value"),permissions:e.array(e.string()).default(["*"]).describe("Associated permissions"),expiresAt:e.number().int().positive().optional().describe("Expiration timestamp (Unix timestamp)"),description:e.string().optional().describe("Token description")}),X=e.object({enabled:e.boolean().default(true),tokens:e.array($).default([]).describe("List of valid tokens"),onInvalid:e.enum(["reject","anonymous"]).default("reject").describe("Behavior when token validation fails"),allowAnonymous:e.boolean().default(false).describe("Allow anonymous access"),defaultPermissions:e.array(e.string()).default([]).describe("Default permissions for anonymous users")}),Z=e.object({enabled:e.boolean().default(false),authorizationServer:e.string().describe("OAuth2 authorization server URL"),clientId:e.string().describe("OAuth2 client ID"),clientSecret:e.string().describe("OAuth2 client secret"),tokenValidationEndpoint:e.string().describe("OAuth2 token validation endpoint"),scope:e.array(e.string()).default([]).describe("OAuth2 scopes")}),z=e.object({type:e.enum(["token","oauth2"]).default("token"),token:X.optional(),oauth2:Z.optional()}),U=e.object({mode:e.enum(["pull","push","both"]).default("both"),interval:e.number().int().positive().default(3e4),pushEvents:e.array(e.enum(["register","deregister","heartbeat","status_change"])).default(["register","deregister","status_change"]),conflictResolution:e.enum(["latest","local","remote"]).default("latest")}),ee=e.object({id:e.string().describe("Gateway node ID"),address:e.string().describe("Gateway node address (e.g., http://localhost:3001)"),enabled:e.boolean().default(true),metadata:e.record(e.string(),e.any()).optional()}),W=e.object({nodeId:e.string().describe("Current gateway node ID"),nodes:e.array(ee).default([]).describe("Other gateway nodes in the cluster"),sync:U.optional()}),v=e.object({maxConnectionsPerChannel:e.number().int().positive().default(100),idleTimeout:e.number().int().positive().default(3e5),reuseStrategy:e.enum(["lru","fifo","random"]).default("lru"),healthCheckInterval:e.number().int().positive().default(6e4),acquireTimeout:e.number().int().positive().default(5e3),warmupCount:e.number().int().nonnegative().default(0).describe("Number of connections to warm up")}),T=e.object({maxLength:e.number().int().positive().default(1e4),concurrency:e.number().int().positive().default(100),batchSize:e.number().int().positive().default(10),batchTimeout:e.number().int().positive().default(100),priorityLevels:e.number().int().positive().default(4)}),G=e.object({failureThreshold:e.number().int().positive().default(5),successThreshold:e.number().int().positive().default(3),timeout:e.number().int().positive().default(3e4),halfOpenDuration:e.number().int().positive().default(15e3)}),P=e.object({heartbeatInterval:e.number().int().positive().default(3e4),heartbeatTimeout:e.number().int().positive().default(9e4),heartbeatCheckInterval:e.number().int().positive().default(1e4),autoDeregisterTimeout:e.number().int().positive().default(3e5),maxAgents:e.number().int().positive().default(100),persistenceFile:e.string().optional().describe("Path to persist agent registrations"),maxRegistrations:e.number().int().positive().default(1e3)}),k=e.object({sessionTimeout:e.number().int().positive().default(18e5),resumeGracePeriod:e.number().int().positive().default(3e5),maxSessions:e.number().int().positive().default(1e4),persistenceFile:e.string().optional().describe("Path to persist sessions")}),I=e.object({enabled:e.boolean().default(true),port:e.number().int().positive().default(8080),hostname:e.string().default("localhost"),path:e.string().optional().default("/"),maxConnections:e.number().int().positive().optional().default(1e3),connectionTimeout:e.number().int().positive().optional().default(6e4),heartbeatInterval:e.number().int().positive().optional().default(3e4),sessionExpireMs:e.number().int().positive().optional().default(3e5),https:j.optional(),auth:z.optional(),cors:e.boolean().default(true),corsOrigins:e.array(e.string()).optional(),mdns:e.boolean().default(false),mdnsDomain:e.string().optional().default("gateway.easbot.local")}),d=e.object({server:I.optional(),channels:D.optional(),cluster:W.optional(),connectionPool:v.optional(),messageQueue:T.optional(),circuitBreaker:G.optional(),agentRegistry:P.optional(),session:k.optional(),defaultAgent:e.string().optional().describe("Default agent ID for routing"),logLevel:e.enum(["DEBUG","INFO","WARN","ERROR"]).optional().default("INFO")}).catchall(e.unknown());function N(t){let n=d.safeParse(t);return n.success||J.warn("Invalid gateway config",{issues:n.error.issues}),n.success?n.data:d.parse({})}function te(t){return d.parse(t)}var he=a.create({service:"gateway:interfaces"}),f;function $e(t){f=t,he.debug("Agent adapter registered",{hasSubAgentRunner:!!t.subAgentRunner});}function Xe(){return f}function K(){return f!==void 0}function q(){if(!f)throw new Error("Agent adapter not set. Call setAgentAdapter() before using Gateway.");return f}var de={Path:{home:process.env.HOME||process.env.USERPROFILE||"",data:".easbot",cache:".easbot/cache",config:".easbot",state:".easbot/state",log:".easbot/log",bin:".easbot/bin"}},me={directory:process.cwd(),worktree:process.cwd()},R=class{async load(n){return F(n)}},ye=new R;function Ze(){return f?f.configLoader:ye}function V(){return f?f.global:de}function _(){return f?f.instance:me}var l=a.create({service:"gateway:config"}),H={enabled:true,port:8080,hostname:"localhost",path:"/",maxConnections:1e3,connectionTimeout:6e4,heartbeatInterval:3e4,sessionExpireMs:3e5,cors:true,mdns:false,mdnsDomain:"gateway.easbot.local",https:{enabled:false,cert:"",key:"",certPath:void 0,keyPath:void 0,ca:void 0,forceRedirect:false,httpPort:80}},we={maxConnectionsPerChannel:100,idleTimeout:3e5,reuseStrategy:"lru",healthCheckInterval:6e4,acquireTimeout:5e3,warmupCount:0},Ae={maxLength:1e4,concurrency:100,batchSize:10,batchTimeout:100,priorityLevels:4},xe={failureThreshold:5,successThreshold:3,timeout:3e4,halfOpenDuration:15e3},ve={heartbeatInterval:3e4,heartbeatTimeout:9e4,heartbeatCheckInterval:1e4,autoDeregisterTimeout:3e5,maxAgents:100,maxRegistrations:1e3},Te={sessionTimeout:18e5,resumeGracePeriod:3e5,maxSessions:1e4},Ge={enabled:true,botToken:"",maxConnections:100,timeout:3e4},Pe={enabled:true,botToken:"",maxConnections:100,timeout:3e4},ke={enabled:true,appId:"",appSecret:"",maxConnections:100,timeout:3e4},Ie={enabled:true,mode:"official",appId:"",appSecret:"",maxConnections:100,timeout:3e4},Ne={enabled:true,path:"/ws",heartbeatInterval:3e4,connectionTimeout:6e4,maxConnections:1e3,cors:true},Le={enabled:true,serviceUrl:"https://chat.signal.org",phoneNumber:"",maxConnections:100,timeout:3e4},Ee={enabled:true,relays:[],maxConnections:100,timeout:3e4},C,p=null;async function Y(t){try{if(!existsSync(t))return null;let n=await Se.readFile(t,"utf-8"),o=n,a=!1,r=!1,s=[];for(let i=0;i<n.length;i++){let g=n[i];if(r){s.push(g),r=!1;continue}if(g==="\\"){s.push(g),r=!0;continue}if(g==='"'){a=!a,s.push(g);continue}if(!a&&g==="/"&&n[i+1]==="/"){for(;i<n.length&&n[i]!==`
2
+ `&&n[i]!=="\r";)i++;continue}if(!a&&g==="/"&&n[i+1]==="*"){for(i+=2;i<n.length-1;){if(n[i]==="*"&&n[i+1]==="/"){i+=2;break}i++;}continue}s.push(g);}return o=s.join("").trim(),o?JSON.parse(o):null}catch(n){return l.debug("Failed to load config file",{file:t,error:String(n)}),null}}function h(t,n){let o={...t,...n};return (t.server||n.server)&&(o.server={...t.server,...n.server}),(t.channels||n.channels)&&(o.channels={...t.channels,...n.channels}),o}async function Fe(){let t=V(),n=_(),o=[],a=O.join(t.Path.config);o.push(a),o.push(n.directory);try{let{Filesystem:s}=await import('@easbot/utils'),i=await Array.fromAsync(s.up({targets:[".easbot"],start:n.directory,stop:n.worktree}));o.push(...i);}catch{l.debug("Filesystem utils not available, using basic directories");}let r=process.env.EASBOT_CONFIG_DIR;return r&&(o.push(r),l.debug("Using EASBOT_CONFIG_DIR",{path:r})),[...new Set(o)]}async function Re(t){let n=["gateway.json"],o={};for(let a of n){let r=O.join(t,a),s=await Y(r);if(s)if("gateway"in s&&s.gateway){let i=s.gateway;"server"in i||"channels"in i||"cluster"in i?o=h(o,i):o=h(o,{server:i}),l.debug("Loaded gateway config from file (nested format)",{file:r});}else "gateway"in s||(o=h(o,s),l.debug("Loaded gateway config from file (direct format)",{file:r}));}return o}async function _e(t){let n=["easbot.json"];for(let o of n){let a=O.join(t,o),r=await Y(a);if(r?.gateway)return l.debug("Loaded gateway config from easbot.json",{file:a}),r.gateway}return {}}function Oe(){return C||(C=async()=>{if(K())try{let a=q(),r=await a.configLoader.load(a.instance.directory);if(r&&typeof r=="object"){let s=r.gateway;if(s){l.debug("Loaded gateway config from agent adapter");let i=N(s);return p=i,{config:i,directories:Array.from(new Set([a.instance.directory,a.instance.worktree,a.global.Path.config]).values())}}}}catch(a){l.debug("Failed to load config from agent adapter",{error:String(a)});}let t=await Fe(),n={};for(let a of t){let r=await Re(a);n=h(n,r);let s=await _e(a);n=h(n,s);}let o=N(n);return p=o,l.info("Gateway config loaded",{directories:t.length,port:o.server?.port,logLevel:o.logLevel}),{config:o,directories:t}}),C}async function c(){return Oe()()}async function F(t){return t&&(C=void 0,p=null),(await c()).config}function B(){return p}function M(){return p!==null}function ne(){C=void 0,p=null,l.debug("Config cache cleared");}function oe(){return _().directory}async function ie(){let{config:t}=await c();return I.parse({...H,...t.server})}async function ae(t){let{config:n}=await c(),o=n.channels?.[t],r={telegram:m.parse(Ge),slack:y.parse(Pe),feishu:b.parse(ke),wechat:S.parse(Ie),webchat:w.parse(Ne),signal:A.parse(Le),nostr:x.parse(Ee)}[t];if(!r)throw new Error(`Unknown channel type: ${t}`);return {...r,...o}}async function re(){let{config:t}=await c();return v.parse({...we,...t.connectionPool})}async function se(){let{config:t}=await c();return T.parse({...Ae,...t.messageQueue})}async function ce(){let{config:t}=await c();return G.parse({...xe,...t.circuitBreaker})}async function fe(){let{config:t}=await c();return P.parse({...ve,...t.agentRegistry})}async function le(){let{config:t}=await c();return k.parse({...Te,...t.session})}async function ge(){let{config:t}=await c();return t.server?.https}async function ue(){let{config:t}=await c();return t.server?.auth}async function pe(){let{config:t}=await c();return t.cluster}async function Ce(){let{config:t}=await c();return t.defaultAgent}async function Q(){let{config:t}=await c();return t.logLevel??"INFO"}export{oe as A,ie as B,ae as C,re as D,se as E,ce as F,fe as G,le as H,ge as I,ue as J,pe as K,Ce as L,Q as M,$e as N,Xe as O,K as P,q as Q,Ze as R,V as S,_ as T,m as a,y as b,b as c,S as d,w as e,A as f,x as g,D as h,j as i,z as j,U as k,W as l,v as m,T as n,G as o,P as p,k as q,I as r,d as s,N as t,te as u,H as v,F as w,B as x,M as y,ne as z};
@@ -1 +1 @@
1
- 'use strict';var chunk4PHQN25U_cjs=require('./chunk-4PHQN25U.cjs'),chunk5ETPYYQR_cjs=require('./chunk-5ETPYYQR.cjs'),chunkGY3SWWW3_cjs=require('./chunk-GY3SWWW3.cjs');var a=chunk5ETPYYQR_cjs.a.create({service:"gateway"}),c=class t extends Error{constructor(r,y,m){super(r);chunkGY3SWWW3_cjs.a(this,"type");chunkGY3SWWW3_cjs.a(this,"stage");chunkGY3SWWW3_cjs.a(this,"cause");this.type="GatewayInitializationError",this.stage=y,this.cause=m,Error.captureStackTrace&&Error.captureStackTrace(this,t);}};function E(t){let s=["ECONNRESET","EPIPE","ETIMEDOUT","ECONNREFUSED","ENOTFOUND","EHOSTUNREACH","ENETUNREACH"],r=t.code;return s.includes(r??"")}function C(){process.on("unhandledRejection",t=>{let s=t instanceof Error?t:new Error(String(t));E(s)||a.error("unhandled promise rejection",{error:s.message,stack:s.stack});}),process.on("uncaughtException",t=>{E(t)||(a.error("uncaught exception",{error:t.message,stack:t.stack}),process.exit(1));});}exports.a=void 0;(k=>{let t={initialized:false,initPromise:null,directory:".easbot",config:null},s,r={server:null,config:null,initialized:false,status:"stopped"};async function y(o={}){if(t.initialized)return;if(t.initPromise)return t.initPromise;let i={directory:o.directory??".easbot",printLogs:o.printLogs??false,logLevel:o.logLevel??"INFO"};return t.directory=i.directory,t.initPromise=(async()=>{let{Global:n}=await import('./global-LIE6RHEM.cjs');try{await n.init();}catch(e){throw new c("Failed to initialize global directories","global",e instanceof Error?e:void 0)}try{await chunk5ETPYYQR_cjs.a.init({logDir:n.Path.log,print:i.printLogs,level:i.logLevel});}catch(e){throw new c("Failed to initialize log infrastructure","log",e instanceof Error?e:void 0)}try{C();}catch(e){throw new c("Failed to install unhandled exception handlers","server",e instanceof Error?e:void 0)}try{let e=chunk4PHQN25U_cjs.P();t.config=await e.load(i.directory);}catch(e){throw new c("Failed to load gateway configuration","config",e instanceof Error?e:void 0)}t.initialized=true,a.info("gateway runtime initialized",{directory:i.directory,logLevel:i.logLevel});})(),t.initPromise}k.init=y;function m(){return t.initialized}k.isInitialized=m;function S(){return s||(s=async()=>{if(!t.initialized)throw new c("Gateway not initialized. Call init() first.","server");let o=chunk4PHQN25U_cjs.P(),i=chunk4PHQN25U_cjs.R(),n=await o.load(i.directory);return n.server?.enabled?(r={server:null,config:n,initialized:true,status:"stopped"},a.info("gateway initialized",{port:n.server?.port,hostname:n.server?.hostname}),r):(a.debug("gateway server disabled or not configured"),{server:null,config:n,initialized:true,status:"stopped"})}),s}function u(){return S()()}k.state=u;async function z(){return (await u()).server}k.get=z;async function R(){return (await u()).config?.server?.enabled??false}k.isEnabled=R;async function L(){return (await u()).config}k.config=L;function N(){return r.status}k.getStatus=N;async function b(o){t.initialized||await y();let i=await u();if(!(i.config?.server?.enabled??true))throw a.warn("gateway server is disabled"),new Error("Gateway server is disabled");if(r.status==="running"){a.info("gateway server is already running");return}if(r.status==="starting"){a.info("gateway server is starting");return}r.status="starting";try{let e={...i.config?.server,...o};a.debug("gateway start: creating server with config",{port:e.port,hostname:e.hostname,path:e.path,https:e.https?.enabled});let g=await G(e);a.debug("gateway start: server created, updating state"),r.server=g,r.status="running",r.error=void 0,a.info("gateway server started",{port:e.port,hostname:e.hostname});}catch(e){throw r.status="error",r.error=e instanceof Error?e.message:String(e),a.error("failed to start gateway server",{error:r.error}),e}}k.start=b;async function P(){if(r.status==="stopped"){a.info("gateway server is already stopped");return}if(r.status==="stopping"){a.info("gateway server is stopping");return}r.status="stopping";try{r.server&&await r.server.stop(),r.server=null,r.status="stopped",r.error=void 0,a.info("gateway server stopped");}catch(o){throw r.status="error",r.error=o instanceof Error?o.message:String(o),a.error("failed to stop gateway server",{error:r.error}),o}}k.stop=P;async function T(o){a.info("restarting gateway server"),(r.status==="running"||r.status==="starting")&&await P(),await b(o);}k.restart=T;async function I(){let o=chunk4PHQN25U_cjs.P(),i=chunk4PHQN25U_cjs.R(),n=await o.load(i.directory);return r.config=n,t.config=n,a.info("gateway config reloaded"),n}k.reloadConfig=I;async function G(o,i){let{GatewayServer:n}=await import('./server-HQL7QNZY.cjs'),e=new n(o);a.debug("createGatewayServer: created server instance, calling start()");let g=i?.startupTimeout??3e4;if(await(async()=>{let l=new Promise((d,x)=>{setTimeout(()=>{x(new Error(`Gateway server startup timeout after ${g}ms`));},g);});try{a.debug("createGatewayServer: calling server.start()"),await Promise.race([e.start(),l]),a.debug("createGatewayServer: server.start() completed");}catch(d){a.error("gateway server start failed, attempting cleanup",{error:d instanceof Error?d.message:String(d)});try{await e.stop();}catch{}throw d}})(),i?.onStarted)try{await i.onStarted(e);}catch(l){a.warn("server started but onStarted callback failed",{error:l instanceof Error?l.message:String(l)});}return a.info("gateway server created and started",{port:o.port,hostname:o.hostname}),e}k.createGatewayServer=G;})(exports.a||(exports.a={}));
1
+ 'use strict';var chunkEK6MKXZH_cjs=require('./chunk-EK6MKXZH.cjs'),chunk5ETPYYQR_cjs=require('./chunk-5ETPYYQR.cjs'),chunkGY3SWWW3_cjs=require('./chunk-GY3SWWW3.cjs');var a=chunk5ETPYYQR_cjs.a.create({service:"gateway"}),c=class t extends Error{constructor(r,y,m){super(r);chunkGY3SWWW3_cjs.a(this,"type");chunkGY3SWWW3_cjs.a(this,"stage");chunkGY3SWWW3_cjs.a(this,"cause");this.type="GatewayInitializationError",this.stage=y,this.cause=m,Error.captureStackTrace&&Error.captureStackTrace(this,t);}};function E(t){let s=["ECONNRESET","EPIPE","ETIMEDOUT","ECONNREFUSED","ENOTFOUND","EHOSTUNREACH","ENETUNREACH"],r=t.code;return s.includes(r??"")}function C(){process.on("unhandledRejection",t=>{let s=t instanceof Error?t:new Error(String(t));E(s)||a.error("unhandled promise rejection",{error:s.message,stack:s.stack});}),process.on("uncaughtException",t=>{E(t)||(a.error("uncaught exception",{error:t.message,stack:t.stack}),process.exit(1));});}exports.a=void 0;(k=>{let t={initialized:false,initPromise:null,directory:".easbot",config:null},s,r={server:null,config:null,initialized:false,status:"stopped"};async function y(o={}){if(t.initialized)return;if(t.initPromise)return t.initPromise;let i={directory:o.directory??".easbot",printLogs:o.printLogs??false,logLevel:o.logLevel??"INFO"};return t.directory=i.directory,t.initPromise=(async()=>{let{Global:n}=await import('./global-XG2WUH2W.cjs');try{await n.init();}catch(e){throw new c("Failed to initialize global directories","global",e instanceof Error?e:void 0)}try{await chunk5ETPYYQR_cjs.a.init({logDir:n.Path.log,print:i.printLogs,level:i.logLevel});}catch(e){throw new c("Failed to initialize log infrastructure","log",e instanceof Error?e:void 0)}try{C();}catch(e){throw new c("Failed to install unhandled exception handlers","server",e instanceof Error?e:void 0)}try{let e=chunkEK6MKXZH_cjs.R();t.config=await e.load(i.directory);}catch(e){throw new c("Failed to load gateway configuration","config",e instanceof Error?e:void 0)}t.initialized=true,a.info("gateway runtime initialized",{directory:i.directory,logLevel:i.logLevel});})(),t.initPromise}k.init=y;function m(){return t.initialized}k.isInitialized=m;function S(){return s||(s=async()=>{if(!t.initialized)throw new c("Gateway not initialized. Call init() first.","server");let o=chunkEK6MKXZH_cjs.R(),i=chunkEK6MKXZH_cjs.T(),n=await o.load(i.directory);return n.server?.enabled?(r={server:null,config:n,initialized:true,status:"stopped"},a.info("gateway initialized",{port:n.server?.port,hostname:n.server?.hostname}),r):(a.debug("gateway server disabled or not configured"),{server:null,config:n,initialized:true,status:"stopped"})}),s}function u(){return S()()}k.state=u;async function z(){return (await u()).server}k.get=z;async function R(){return (await u()).config?.server?.enabled??false}k.isEnabled=R;async function L(){return (await u()).config}k.config=L;function N(){return r.status}k.getStatus=N;async function b(o){t.initialized||await y();let i=await u();if(!(i.config?.server?.enabled??true))throw a.warn("gateway server is disabled"),new Error("Gateway server is disabled");if(r.status==="running"){a.info("gateway server is already running");return}if(r.status==="starting"){a.info("gateway server is starting");return}r.status="starting";try{let e={...i.config?.server,...o};a.debug("gateway start: creating server with config",{port:e.port,hostname:e.hostname,path:e.path,https:e.https?.enabled});let g=await G(e);a.debug("gateway start: server created, updating state"),r.server=g,r.status="running",r.error=void 0,a.info("gateway server started",{port:e.port,hostname:e.hostname});}catch(e){throw r.status="error",r.error=e instanceof Error?e.message:String(e),a.error("failed to start gateway server",{error:r.error}),e}}k.start=b;async function P(){if(r.status==="stopped"){a.info("gateway server is already stopped");return}if(r.status==="stopping"){a.info("gateway server is stopping");return}r.status="stopping";try{r.server&&await r.server.stop(),r.server=null,r.status="stopped",r.error=void 0,a.info("gateway server stopped");}catch(o){throw r.status="error",r.error=o instanceof Error?o.message:String(o),a.error("failed to stop gateway server",{error:r.error}),o}}k.stop=P;async function T(o){a.info("restarting gateway server"),(r.status==="running"||r.status==="starting")&&await P(),await b(o);}k.restart=T;async function I(){let o=chunkEK6MKXZH_cjs.R(),i=chunkEK6MKXZH_cjs.T(),n=await o.load(i.directory);return r.config=n,t.config=n,a.info("gateway config reloaded"),n}k.reloadConfig=I;async function G(o,i){let{GatewayServer:n}=await import('./server-NST3VRWT.cjs'),e=new n(o);a.debug("createGatewayServer: created server instance, calling start()");let g=i?.startupTimeout??3e4;if(await(async()=>{let l=new Promise((d,x)=>{setTimeout(()=>{x(new Error(`Gateway server startup timeout after ${g}ms`));},g);});try{a.debug("createGatewayServer: calling server.start()"),await Promise.race([e.start(),l]),a.debug("createGatewayServer: server.start() completed");}catch(d){a.error("gateway server start failed, attempting cleanup",{error:d instanceof Error?d.message:String(d)});try{await e.stop();}catch{}throw d}})(),i?.onStarted)try{await i.onStarted(e);}catch(l){a.warn("server started but onStarted callback failed",{error:l instanceof Error?l.message:String(l)});}return a.info("gateway server created and started",{port:o.port,hostname:o.hostname}),e}k.createGatewayServer=G;})(exports.a||(exports.a={}));
@@ -1,2 +1,2 @@
1
- 'use strict';var s={name:"@easbot/gateway",version:"0.2.11",description:"EASBot Gateway - AI Agent Server and Multi-channel Integration Platform - \u652F\u6301 WebSocket\u3001HTTP\u3001Discord\u3001Telegram\u3001Slack \u7B49\u591A\u6E20\u9053\u96C6\u6210",type:"module",main:"dist/index.cjs",module:"dist/index.mjs",types:"dist/index.d.ts",exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.mjs",require:"./dist/index.cjs"},"./package.json":"./package.json"},scripts:{dev:"cross-env NODE_ENV=development tsx src/cli.ts -- start",start:"cross-env NODE_ENV=production node dist/cli.mjs start",build:"tsup --env.NODE_ENV production",test:"vitest","test:run":"vitest run",lint:"biome check .","lint:fix":"biome check --write .","lint:report":"biome check --reporter=summary .",format:"biome format .","format:fix":"biome format --write .","type-check":"tsc --noEmit",clean:"npx rimraf dist node_modules",prepare:"echo norun",prepublishOnly:"pnpm build","publish:npm":"bash scripts/publish.sh","publish:npm:win":"powershell -ExecutionPolicy Bypass -File scripts/publish.ps1"},keywords:["easbot","gateway","server","websocket","http","agent","multi-channel","discord","telegram","slack","feishu","wechat","chat","integration"],author:"houjallen",license:"MIT",repository:{type:"git",url:"https://github.com/houjallen/easbot.git",directory:"packages/gateway"},homepage:"https://github.com/houjallen/easbot/tree/main/packages/gateway#readme",bugs:{url:"https://github.com/houjallen/easbot/issues"},files:["dist","README.md","README.en.md","LICENSE"],dependencies:{"@ai-sdk/provider":"^3.0.10","@ai-sdk/provider-utils":"^4.0.27","@ai-sdk/openai-compatible":"^2.0.47","@ai-sdk/anthropic":"^3.0.76","@easbot/plugin":"workspace:*","@easbot/sdk":"workspace:*","@easbot/types":"workspace:*","@easbot/utils":"workspace:*","@hono/node-server":"^2.0.2","@hono/node-ws":"^1.3.0","@hono/standard-validator":"^0.2.2","@hono/zod-validator":"^0.7.6",commander:"^14.0.3",zod:"^4.4.3",hono:"^4.12.18","hono-openapi":"^1.3.0",ws:"^8.20.0",ai:"^6.0.176","better-sqlite3":"^12.9.0","jieba-wasm":"^2.4.0","xdg-basedir":"^5.1.0",minimatch:"^10.2.5"},devDependencies:{"@biomejs/biome":"^2.4.14","@types/better-sqlite3":"^7.6.13","@types/ws":"^8.18.1","@types/node":"^25.6.2","@vitest/coverage-v8":"^4.1.5",dotenv:"^17.4.2",tsup:"^8.5.1",typescript:"^6.0.3",vitest:"^4.1.5"},engines:{node:">=22.22.3"},publishConfig:{access:"public"}};
1
+ 'use strict';var s={name:"@easbot/gateway",version:"0.2.13",description:"EASBot Gateway - AI Agent Server and Multi-channel Integration Platform - \u652F\u6301 WebSocket\u3001HTTP\u3001Discord\u3001Telegram\u3001Slack \u7B49\u591A\u6E20\u9053\u96C6\u6210",type:"module",main:"dist/index.cjs",module:"dist/index.mjs",types:"dist/index.d.ts",exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.mjs",require:"./dist/index.cjs"},"./package.json":"./package.json"},scripts:{dev:"cross-env NODE_ENV=development tsx src/cli.ts -- start",start:"cross-env NODE_ENV=production node dist/cli.mjs start",build:"tsup --env.NODE_ENV production",test:"vitest","test:run":"vitest run",lint:"biome check .","lint:fix":"biome check --write .","lint:report":"biome check --reporter=summary .",format:"biome format .","format:fix":"biome format --write .","type-check":"tsc --noEmit",clean:"npx rimraf dist node_modules",prepare:"echo norun",prepublishOnly:"pnpm build","publish:npm":"bash scripts/publish.sh","publish:npm:win":"powershell -ExecutionPolicy Bypass -File scripts/publish.ps1"},keywords:["easbot","gateway","server","websocket","http","agent","multi-channel","discord","telegram","slack","feishu","wechat","chat","integration"],author:"houjallen",license:"MIT",repository:{type:"git",url:"https://github.com/houjallen/easbot.git",directory:"packages/gateway"},homepage:"https://github.com/houjallen/easbot/tree/main/packages/gateway#readme",bugs:{url:"https://github.com/houjallen/easbot/issues"},files:["dist","README.md","README.en.md","LICENSE"],dependencies:{"@ai-sdk/provider":"^3.0.10","@ai-sdk/provider-utils":"^4.0.27","@ai-sdk/openai-compatible":"^2.0.47","@ai-sdk/anthropic":"^3.0.76","@easbot/plugin":"workspace:*","@easbot/sdk":"workspace:*","@easbot/types":"workspace:*","@easbot/utils":"workspace:*","@hono/node-server":"^2.0.2","@hono/node-ws":"^1.3.0","@hono/standard-validator":"^0.2.2","@hono/zod-validator":"^0.7.6",commander:"^14.0.3",zod:"^4.4.3",hono:"^4.12.18","hono-openapi":"^1.3.0",ws:"^8.20.0",ai:"^6.0.176","better-sqlite3":"^12.9.0","jieba-wasm":"^2.4.0","xdg-basedir":"^5.1.0",minimatch:"^10.2.5"},devDependencies:{"@biomejs/biome":"^2.4.14","@types/better-sqlite3":"^7.6.13","@types/ws":"^8.18.1","@types/node":"^25.6.2","@vitest/coverage-v8":"^4.1.5",dotenv:"^17.4.2",tsup:"^8.5.1",typescript:"^6.0.3",vitest:"^4.1.5"},engines:{node:">=22.22.3"},publishConfig:{access:"public"}};
2
2
  exports.a=s;
@@ -1,2 +1,2 @@
1
- 'use strict';var chunk4PHQN25U_cjs=require('./chunk-4PHQN25U.cjs'),chunk5ETPYYQR_cjs=require('./chunk-5ETPYYQR.cjs'),chunkGY3SWWW3_cjs=require('./chunk-GY3SWWW3.cjs'),p=require('fs'),L=require('path'),utils=require('@easbot/utils'),ws=require('ws'),j=require('https'),Q=require('http'),plugin=require('@easbot/plugin');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}function _interopNamespace(e){if(e&&e.__esModule)return e;var n=Object.create(null);if(e){Object.keys(e).forEach(function(k){if(k!=='default'){var d=Object.getOwnPropertyDescriptor(e,k);Object.defineProperty(n,k,d.get?d:{enumerable:true,get:function(){return e[k]}});}})}n.default=e;return Object.freeze(n)}var p__namespace=/*#__PURE__*/_interopNamespace(p);var L__namespace=/*#__PURE__*/_interopNamespace(L);var j__default=/*#__PURE__*/_interopDefault(j);var Q__default=/*#__PURE__*/_interopDefault(Q);var M=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:lock"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"locks",new Map);chunkGY3SWWW3_cjs.a(this,"cleanupTimer",null);this.config={defaultTimeout:e.defaultTimeout??6e4,checkInterval:e.checkInterval??1e4};}tryAcquire(e,t,s,n){let r=this.locks.get(e);if(r)if(this.isLockExpired(r))this.log.warn("lock expired, releasing",{sessionId:e,oldHolder:r.holderId,newHolder:s}),this.locks.delete(e);else return this.log.debug("lock already held",{sessionId:e,holder:r.holderId,messageId:r.messageId}),false;let a={sessionId:e,messageId:t,holderId:s,acquiredAt:Date.now(),timeout:n??this.config.defaultTimeout};return this.locks.set(e,a),this.log.debug("lock acquired",{sessionId:e,messageId:t,subscriberId:s}),true}release(e,t){let s=this.locks.get(e);return s?s.holderId!==t?(this.log.warn("cannot release lock held by another",{sessionId:e,holder:s.holderId,requester:t}),false):(this.locks.delete(e),this.log.debug("lock released",{sessionId:e,subscriberId:t}),true):(this.log.debug("no lock to release",{sessionId:e}),false)}forceRelease(e){let t=this.locks.get(e);return t?(this.locks.delete(e),this.log.warn("lock force released",{sessionId:e,holder:t.holderId,messageId:t.messageId}),true):false}isLockExpired(e){return Date.now()-e.acquiredAt>e.timeout}getLock(e){return this.locks.get(e)}isLocked(e){let t=this.locks.get(e);return t?this.isLockExpired(t)?(this.locks.delete(e),false):true:false}getHolder(e){let t=this.locks.get(e);if(!(!t||this.isLockExpired(t)))return t.holderId}startCleanup(){this.cleanupTimer||(this.cleanupTimer=setInterval(()=>{this.cleanupExpired();},this.config.checkInterval),this.log.info("lock cleanup started",{interval:this.config.checkInterval}));}stopCleanup(){this.cleanupTimer&&(clearInterval(this.cleanupTimer),this.cleanupTimer=null,this.log.info("lock cleanup stopped"));}cleanupExpired(){let e=0;for(let[t,s]of this.locks)this.isLockExpired(s)&&(this.locks.delete(t),e++,this.log.warn("expired lock cleaned up",{sessionId:t,holder:s.holderId,messageId:s.messageId}));return e>0&&this.log.info("cleaned up expired locks",{count:e}),e}getAllLocks(){return [...this.locks.values()]}getLockCount(){return this.locks.size}};var I=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:retry"}));chunkGY3SWWW3_cjs.a(this,"policy");chunkGY3SWWW3_cjs.a(this,"retryQueue",new Map);chunkGY3SWWW3_cjs.a(this,"retryTimer",null);chunkGY3SWWW3_cjs.a(this,"handlers",new Map);this.policy={maxRetries:e.maxRetries??3,initialDelay:e.initialDelay??1e3,maxDelay:e.maxDelay??3e4,backoffMultiplier:e.backoffMultiplier??2};}registerHandler(e,t){this.handlers.set(e,t);}unregisterHandler(e){this.handlers.delete(e);}scheduleRetry(e,t){let s=e.id,n=this.retryQueue.get(s);if(n||(n={messageId:s,retryCount:0,nextRetryAt:0,errors:[]},this.retryQueue.set(s,n)),n.errors.push({error:t.message,timestamp:Date.now()}),n.retryCount>=this.policy.maxRetries)return this.log.error("max retries exceeded",{messageId:s,retryCount:n.retryCount,maxRetries:this.policy.maxRetries}),false;let r=Math.min(this.policy.initialDelay*this.policy.backoffMultiplier**n.retryCount,this.policy.maxDelay);return n.retryCount++,n.nextRetryAt=Date.now()+r,this.log.warn("retry scheduled",{messageId:s,retryCount:n.retryCount,delay:r,error:t.message}),true}getPendingRetries(){let e=Date.now(),t=[];for(let s of this.retryQueue.values())s.nextRetryAt<=e&&s.retryCount<=this.policy.maxRetries&&t.push(s);return t}async executeRetry(e){if(!this.handlers.get(e.messageId))return this.log.warn("no handler for retry",{messageId:e.messageId}),false;this.retryQueue.delete(e.messageId);try{return this.log.info("executing retry",{messageId:e.messageId,retryCount:e.retryCount}),!0}catch(s){return this.log.error("retry execution failed",{messageId:e.messageId,error:String(s)}),false}}clearRetry(e){this.retryQueue.delete(e),this.handlers.delete(e),this.log.debug("retry cleared",{messageId:e});}startProcessing(){this.retryTimer||(this.retryTimer=setInterval(async()=>{let e=this.getPendingRetries();for(let t of e)await this.executeRetry(t);},1e3),this.log.info("retry processing started"));}stopProcessing(){this.retryTimer&&(clearInterval(this.retryTimer),this.retryTimer=null,this.log.info("retry processing stopped"));}getStats(){let e=0;for(let t of this.retryQueue.values())e+=t.errors.length;return {pendingCount:this.retryQueue.size,totalErrors:e}}};function H(c,e,t,s){return {id:`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:c,type:e,content:t,metadata:{channel:{platform:s?.channel?.platform??"api",channelId:s?.channel?.channelId??"default",userId:s?.channel?.userId,chatId:s?.channel?.chatId,...Object.fromEntries(Object.entries(s?.channel??{}).filter(([n])=>!["platform","channelId","userId","chatId"].includes(n)))},context:s?.context,agent:s?.agent},timestamp:Date.now()}}function N(c,e,t,s){return H(c,e,[{type:"text",text:t}],s)}function T(c){let e=[c.platform,c.channelId];return c.chatId&&e.push(c.chatId),c.userId&&e.push(c.userId),e.join("_")}function R(){return {status:"active",messageCount:0,lastMessageAt:null}}function k(c,e){return `sub_${c}_${e}`}var E=chunk4PHQN25U_cjs.v,y={maxConnections:1e3,connectionTimeout:6e4,sessionExpireMs:3e5};var B={type:"token",enabled:true,tokens:[],onInvalid:"reject",allowAnonymous:false,defaultPermissions:[]};var q={heartbeatInterval:3e4,heartbeatTimeout:9e4,heartbeatCheckInterval:1e4,autoDeregisterTimeout:3e5,maxAgents:100,maxRegistrations:1e3};var F={mode:"both",interval:3e4,pushEvents:["register","deregister","status_change"],conflictResolution:"latest",remoteNodes:[]};var U={maxConnectionsPerChannel:100,idleTimeout:3e5,reuseStrategy:"lru",healthCheckInterval:6e4,acquireTimeout:5e3,warmupCount:0};var x=class{constructor(){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:router"}));chunkGY3SWWW3_cjs.a(this,"subscribers",new Map);chunkGY3SWWW3_cjs.a(this,"subscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"sessionSubscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"channelPlugins",new Map);chunkGY3SWWW3_cjs.a(this,"lockManager");chunkGY3SWWW3_cjs.a(this,"retryManager");chunkGY3SWWW3_cjs.a(this,"cancelHandlers",new Set);this.lockManager=new M,this.retryManager=new I,this.lockManager.startCleanup();}onCancel(e){this.cancelHandlers.add(e);}offCancel(e){this.cancelHandlers.delete(e);}async emitCancel(e){for(let t of this.cancelHandlers)try{await t(e);}catch(s){this.log.error("cancel handler error",{error:String(s)});}}async route(e){this.log.debug("routing message",{id:e.id,sessionId:e.sessionId,type:e.type});let t=this.sessionSubscriptions.get(e.sessionId);if(!t||t.size===0)return this.log.debug("no subscribers for session",{sessionId:e.sessionId}),{success:true,broadcastCount:0,dispatchedToSubAgent:false};let s=0,n=null,r=[];for(let a of t){let g=this.subscriptions.get(a);if(!g)continue;let l=this.subscribers.get(g.subscriberId);if(l){if(e.type==="input"){if(!this.lockManager.tryAcquire(e.sessionId,e.id,l.id)){this.log.debug("skipping subscriber, lock held by another",{subscriberId:l.id,holder:this.lockManager.getHolder(e.sessionId)});continue}n=l.id;}try{await l.connection.send(e),s++,l.lastActiveAt=Date.now();}catch(h){this.log.warn("failed to send message to subscriber",{subscriberId:l.id,error:String(h)}),r.push(h instanceof Error?h:new Error(String(h))),n===l.id&&(this.lockManager.release(e.sessionId,l.id),n=null);}}}return this.log.debug("message routed",{id:e.id,broadcastCount:s,errorCount:r.length,processedBy:n}),{success:r.length===0,broadcastCount:s,dispatchedToSubAgent:false,error:r.length>0?r.map(a=>a.message).join("; "):void 0}}async completeMessage(e,t,s,n="completed"){this.lockManager.release(e,s),await this.emitCancel({sessionId:e,messageId:t,processedBy:s,reason:n,timestamp:Date.now()}),this.log.info("message completed",{sessionId:e,messageId:t,subscriberId:s,reason:n});}scheduleRetry(e,t){return this.retryManager.scheduleRetry(e,t)}async subscribe(e,t){let s=k(t.id,e);if(this.subscriptions.has(s))return this.log.debug("already subscribed",{subscriberId:t.id,sessionId:e}),this.subscriptions.get(s);let n={id:s,subscriberId:t.id,sessionId:e,createdAt:Date.now()};return this.subscribers.has(t.id)||this.subscribers.set(t.id,t),this.subscriptions.set(s,n),this.sessionSubscriptions.has(e)||this.sessionSubscriptions.set(e,new Set),this.sessionSubscriptions.get(e).add(s),t.sessions.add(e),this.log.info("subscribed to session",{subscriberId:t.id,sessionId:e,subscriptionId:s}),n}async unsubscribe(e){let t=this.subscriptions.get(e);if(!t){this.log.warn("subscription not found",{subscriptionId:e});return}this.subscriptions.delete(e);let s=this.sessionSubscriptions.get(t.sessionId);s&&(s.delete(e),s.size===0&&this.sessionSubscriptions.delete(t.sessionId));let n=this.subscribers.get(t.subscriberId);n&&n.sessions.delete(t.sessionId),this.log.info("unsubscribed from session",{subscriberId:t.subscriberId,sessionId:t.sessionId,subscriptionId:e});}async unsubscribeAll(e){let t=this.subscribers.get(e);if(!t)return;let s=[...t.sessions];for(let n of s){let r=k(e,n);await this.unsubscribe(r);}this.subscribers.delete(e),this.log.info("unsubscribed all sessions",{subscriberId:e});}async registerChannel(e){if(this.channelPlugins.has(e.id)){this.log.warn("channel plugin already registered",{id:e.id});return}this.channelPlugins.set(e.id,e),this.log.info("channel plugin registered",{id:e.id,platform:e.platform});}async unregisterChannel(e){let t=this.channelPlugins.get(e);if(!t){this.log.warn("channel plugin not found",{id:e});return}await t.stop(),this.channelPlugins.delete(e),this.log.info("channel plugin unregistered",{id:e});}getSubscriberCount(e){let t=this.sessionSubscriptions.get(e);return t?t.size:0}getAllSubscribers(){return [...this.subscribers.values()]}getChannelPlugin(e){return this.channelPlugins.get(e)}getAllChannelPlugins(){return [...this.channelPlugins.values()]}getLockManager(){return this.lockManager}getRetryManager(){return this.retryManager}async shutdown(){this.lockManager.stopCleanup(),this.retryManager.stopProcessing(),this.log.info("router shutdown");}};var S=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:message-store"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"messages",new Map);chunkGY3SWWW3_cjs.a(this,"sessionMessages",new Map);this.config={storagePath:e.storagePath??"./data/gateway-messages",maxRetentionDays:e.maxRetentionDays??30,maxRecords:e.maxRecords??1e5};}async store(e){let t={message:e,status:"pending",processedBy:null,processedAt:null,retryCount:0,error:null,createdAt:Date.now(),updatedAt:Date.now()};return this.messages.set(e.id,t),this.sessionMessages.has(e.sessionId)||this.sessionMessages.set(e.sessionId,new Set),this.sessionMessages.get(e.sessionId).add(e.id),this.log.debug("message stored",{id:e.id,sessionId:e.sessionId,type:e.type}),t}get(e){return this.messages.get(e)}async updateStatus(e,t,s={}){let n=this.messages.get(e);if(!n){this.log.warn("message not found for status update",{messageId:e});return}n.status=t,n.updatedAt=Date.now(),s.processedBy&&(n.processedBy=s.processedBy),(t==="completed"||t==="failed")&&(n.processedAt=Date.now()),s.error&&(n.error=s.error),this.log.debug("message status updated",{id:e,status:t,processedBy:s.processedBy});}incrementRetry(e){let t=this.messages.get(e);return t?(t.retryCount++,t.updatedAt=Date.now(),t.retryCount):0}isProcessed(e){let t=this.messages.get(e);return t?t.status==="completed"||t.status==="processing":false}getSessionHistory(e,t=100){let s=this.sessionMessages.get(e);if(!s)return [];let n=[];for(let r of s){let a=this.messages.get(r);a&&n.push(a);}return n.sort((r,a)=>r.createdAt-a.createdAt),n.slice(-t)}getPendingMessages(e){let t=[];for(let s of this.messages.values())s.status==="pending"&&(!e||s.message.sessionId===e)&&t.push(s);return t}getFailedMessages(e=3){let t=[];for(let s of this.messages.values())s.status==="failed"&&s.retryCount<e&&t.push(s);return t}async cleanup(){let e=Date.now(),t=this.config.maxRetentionDays*24*60*60*1e3,s=[];for(let[n,r]of this.messages)e-r.createdAt>t&&s.push(n);for(let n of s){let r=this.messages.get(n);if(r){let a=this.sessionMessages.get(r.message.sessionId);a&&a.delete(n),this.messages.delete(n);}}return s.length>0&&this.log.info("cleaned up expired messages",{count:s.length}),s.length}getStats(){let e={total:this.messages.size,pending:0,processing:0,completed:0,failed:0,cancelled:0};for(let t of this.messages.values())e[t.status]++;return e}};var b=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:session-store"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"sessions",new Map);chunkGY3SWWW3_cjs.a(this,"channelSessions",new Map);chunkGY3SWWW3_cjs.a(this,"autoSaveTimer",null);chunkGY3SWWW3_cjs.a(this,"isDirty",false);this.config={storagePath:e.storagePath??"./data/gateway-sessions",autoSaveInterval:e.autoSaveInterval??6e4,maxRetentionDays:e.maxRetentionDays??7,enablePersistence:e.enablePersistence??true};}async initialize(){if(!this.config.enablePersistence){this.log.info("session persistence disabled");return}await this.ensureStorageDir(),await this.load(),this.startAutoSave(),this.log.info("session store initialized",{sessionCount:this.sessions.size,storagePath:this.config.storagePath});}async close(){this.stopAutoSave(),this.isDirty&&await this.save(),this.log.info("session store closed");}async set(e){this.sessions.set(e.id,e),this.channelSessions.has(e.channel.channelId)||this.channelSessions.set(e.channel.channelId,new Set),this.channelSessions.get(e.channel.channelId).add(e.id),this.markDirty();}get(e){return this.sessions.get(e)}async delete(e){let t=this.sessions.get(e);if(!t)return;this.sessions.delete(e);let s=this.channelSessions.get(t.channel.channelId);s&&(s.delete(e),s.size===0&&this.channelSessions.delete(t.channel.channelId)),this.markDirty();}getAll(){return [...this.sessions.values()]}getByChannel(e){let t=this.channelSessions.get(e);return t?[...t].map(s=>this.sessions.get(s)).filter(s=>s!==void 0):[]}size(){return this.sessions.size}markDirty(){this.isDirty=true;}async save(){if(!this.config.enablePersistence)return;let e=this.getSessionFilePath(),t=this.serializeAll();try{await this.ensureStorageDir(),await p__namespace.promises.writeFile(e,JSON.stringify(t,null,2),"utf-8"),this.isDirty=!1,this.log.debug("sessions saved",{count:t.length});}catch(s){throw this.log.error("failed to save sessions",{error:s}),s}}async load(){if(!this.config.enablePersistence)return;let e=this.getSessionFilePath();try{if(!p__namespace.existsSync(e)){this.log.debug("no saved sessions found");return}let t=await p__namespace.promises.readFile(e,"utf-8"),s=JSON.parse(t);this.sessions.clear(),this.channelSessions.clear();let n=Date.now(),r=this.config.maxRetentionDays*24*60*60*1e3;for(let a of s){if(n-a.lastActiveAt>r)continue;let g={id:a.id,channel:a.channel,backendSessionId:a.backendSessionId,subscribers:new Set(a.subscribers),state:a.state,createdAt:a.createdAt,lastActiveAt:a.lastActiveAt};this.sessions.set(g.id,g),this.channelSessions.has(g.channel.channelId)||this.channelSessions.set(g.channel.channelId,new Set),this.channelSessions.get(g.channel.channelId).add(g.id);}this.log.info("sessions loaded",{loaded:this.sessions.size,skipped:s.length-this.sessions.size});}catch(t){this.log.error("failed to load sessions",{error:t});}}serializeAll(){let e=[];for(let t of this.sessions.values())e.push(this.serialize(t));return e}serialize(e){return {id:e.id,channel:e.channel,backendSessionId:e.backendSessionId,subscribers:[...e.subscribers],state:e.state,createdAt:e.createdAt,lastActiveAt:e.lastActiveAt}}getSessionFilePath(){return L__namespace.join(this.config.storagePath,"sessions.json")}async ensureStorageDir(){p__namespace.existsSync(this.config.storagePath)||await p__namespace.promises.mkdir(this.config.storagePath,{recursive:true});}startAutoSave(){this.autoSaveTimer||(this.autoSaveTimer=setInterval(async()=>{this.isDirty&&await this.save();},this.config.autoSaveInterval));}stopAutoSave(){this.autoSaveTimer&&(clearInterval(this.autoSaveTimer),this.autoSaveTimer=null);}async cleanup(){let e=Date.now(),t=this.config.maxRetentionDays*24*60*60*1e3,s=[];for(let n of this.sessions.values())e-n.lastActiveAt>t&&s.push(n.id);for(let n of s)await this.delete(n);return s.length>0&&(this.log.info("cleaned up expired sessions",{count:s.length}),await this.save()),s.length}getStats(){let e={total:this.sessions.size,byStatus:{active:0,idle:0,closed:0},byPlatform:{}};for(let t of this.sessions.values()){e.byStatus[t.state.status]++;let s=t.channel.platform;e.byPlatform[s]=(e.byPlatform[s]||0)+1;}return e}};var v=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:session"}));chunkGY3SWWW3_cjs.a(this,"sessionStore");chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"initialized",false);this.config={enablePersistence:e.enablePersistence??true,storagePath:e.storagePath??"./data/gateway-sessions",autoSaveInterval:e.autoSaveInterval??6e4,maxRetentionDays:e.maxRetentionDays??7},this.sessionStore=new b({storagePath:this.config.storagePath,autoSaveInterval:this.config.autoSaveInterval,maxRetentionDays:this.config.maxRetentionDays,enablePersistence:this.config.enablePersistence});}async initialize(){this.initialized||(await this.sessionStore.initialize(),this.initialized=true,this.log.info("session manager initialized",{sessionCount:this.sessionStore.size(),persistenceEnabled:this.config.enablePersistence}));}async close(){await this.sessionStore.close(),this.log.info("session manager closed");}async getOrCreate(e,t={}){let s=T(e),n=this.sessionStore.get(s);return n?(n.lastActiveAt=Date.now(),this.log.debug("session found",{sessionId:s}),n):(n={id:s,channel:e,backendSessionId:null,subscribers:new Set,state:R(),createdAt:Date.now(),lastActiveAt:Date.now()},await this.sessionStore.set(n),this.log.info("session created",{sessionId:s,platform:e.platform,channelId:e.channelId,chatId:e.chatId,userId:e.userId}),n)}get(e){return this.sessionStore.get(e)}async closeSession(e){let t=this.sessionStore.get(e);if(!t){this.log.warn("session not found",{sessionId:e});return}t.state.status="closed",await this.sessionStore.delete(e),this.log.info("session closed",{sessionId:e});}async closeById(e){return this.closeSession(e)}async bindBackendSession(e,t){let s=this.sessionStore.get(e);if(!s)throw new Error(`Session not found: ${e}`);s.backendSessionId=t,await this.sessionStore.set(s),this.log.info("backend session bound",{gatewaySessionId:e,backendSessionId:t});}async unbindBackendSession(e){let t=this.sessionStore.get(e);if(!t){this.log.warn("session not found",{sessionId:e});return}t.backendSessionId=null,await this.sessionStore.set(t),this.log.info("backend session unbound",{gatewaySessionId:e});}getState(e){return this.sessionStore.get(e)?.state}async incrementMessageCountOrCreate(e,t){let s=this.sessionStore.get(e);if(!s){let n=t??{platform:"api",channelId:"default"};s=await this.getOrCreate(n);}return s.state.messageCount++,s.state.lastMessageAt=Date.now(),s.lastActiveAt=Date.now(),await this.sessionStore.set(s),s}async incrementMessageCount(e){let t=this.sessionStore.get(e);t&&(t.state.messageCount++,t.state.lastMessageAt=Date.now(),t.lastActiveAt=Date.now(),await this.sessionStore.set(t));}async addSubscriber(e,t){let s=this.sessionStore.get(e);if(!s){this.log.warn("session not found",{sessionId:e});return}s.subscribers.add(t),await this.sessionStore.set(s),this.log.debug("subscriber added to session",{sessionId:e,subscriberId:t});}async removeSubscriber(e,t){let s=this.sessionStore.get(e);s&&(s.subscribers.delete(t),await this.sessionStore.set(s),this.log.debug("subscriber removed from session",{sessionId:e,subscriberId:t}));}query(e){let t=this.sessionStore.getAll();return e.platform&&(t=t.filter(s=>s.channel.platform===e.platform)),e.channelId&&(t=t.filter(s=>s.channel.channelId===e.channelId)),e.userId&&(t=t.filter(s=>s.channel.userId===e.userId)),e.status&&(t=t.filter(s=>s.state.status===e.status)),t}getSessionsByChannel(e){return this.sessionStore.getByChannel(e)}getAllSessions(){return this.sessionStore.getAll()}getSessionCount(){return this.sessionStore.size()}async cleanupIdleSessions(e){let t=Date.now(),s=[];for(let n of this.sessionStore.getAll())n.state.status==="idle"&&t-n.lastActiveAt>e&&s.push(n.id);for(let n of s)await this.closeSession(n);return s.length>0&&this.log.info("cleaned up idle sessions",{count:s.length}),s.length}async save(){this.log.debug("session save triggered");}getStats(){return this.sessionStore.getStats()}};var C=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:plugin-loader"}));chunkGY3SWWW3_cjs.a(this,"plugins",new Map);chunkGY3SWWW3_cjs.a(this,"messageHandler",null);chunkGY3SWWW3_cjs.a(this,"healthCheckTimer",null);chunkGY3SWWW3_cjs.a(this,"config");this.config={pluginDir:e.pluginDir??"./plugins",autoDiscover:e.autoDiscover??false,healthCheckInterval:e.healthCheckInterval??6e4,healthCheckTimeout:e.healthCheckTimeout??5e3};}setMessageHandler(e){this.messageHandler=e;}async register(e,t){if(this.plugins.has(e.id)){this.log.warn("plugin already registered",{id:e.id});return}this.plugins.set(e.id,{plugin:e,config:t,started:false}),this.log.info("plugin registered",{id:e.id,platform:e.platform,name:e.name});}async unregister(e){let t=this.plugins.get(e);if(!t){this.log.warn("plugin not found",{id:e});return}t.started&&await this.stop(e),this.plugins.delete(e),this.log.info("plugin unregistered",{id:e});}async start(e){let t=this.plugins.get(e);if(!t)throw new Error(`Plugin not found: ${e}`);if(t.started){this.log.warn("plugin already started",{id:e});return}if(!this.messageHandler)throw new Error("Message handler not set");if(!t.config.enabled){this.log.warn("plugin is disabled",{id:e});return}this.log.info("starting plugin",{id:e});try{await t.plugin.start(t.config,this.messageHandler),t.started=!0,this.log.info("plugin started",{id:e});}catch(s){throw this.log.error("failed to start plugin",{id:e,error:String(s)}),s}}async stop(e){let t=this.plugins.get(e);if(!t){this.log.warn("plugin not found",{id:e});return}if(!t.started){this.log.warn("plugin not started",{id:e});return}this.log.info("stopping plugin",{id:e});try{await t.plugin.stop(),t.started=!1,this.log.info("plugin stopped",{id:e});}catch(s){throw this.log.error("failed to stop plugin",{id:e,error:String(s)}),s}}async startAll(){let e=[];for(let[t,s]of this.plugins){if(!s.config.enabled){this.log.debug("skipping disabled plugin",{id:t});continue}try{await this.start(t);}catch(n){e.push({id:t,error:n instanceof Error?n:new Error(String(n))});}}e.length>0&&this.log.warn("some plugins failed to start",{count:e.length,errors:e.map(t=>({id:t.id,error:t.error.message}))});}async stopAll(){let e=[];for(let[t,s]of this.plugins)if(s.started)try{await this.stop(t);}catch(n){e.push({id:t,error:n instanceof Error?n:new Error(String(n))});}e.length>0&&this.log.warn("some plugins failed to stop",{count:e.length,errors:e.map(t=>({id:t.id,error:t.error.message}))});}getPlugin(e){return this.plugins.get(e)?.plugin}getAllPlugins(){return [...this.plugins.values()].map(e=>e.plugin)}getRunningPlugins(){return [...this.plugins.values()].filter(e=>e.started).map(e=>e.plugin)}isRunning(e){return this.plugins.get(e)?.started??false}async healthCheck(e){let t=this.plugins.get(e);if(!t)return {healthy:false,lastCheck:Date.now(),error:"Plugin not found"};try{let s=await Promise.race([t.plugin.healthCheck(),new Promise((n,r)=>setTimeout(()=>r(new Error("Health check timeout")),this.config.healthCheckTimeout))]);return t.lastHealthCheck=s,s}catch(s){let n={healthy:false,lastCheck:Date.now(),error:s instanceof Error?s.message:String(s)};return t.lastHealthCheck=n,n}}async healthCheckAll(){let e=new Map;for(let[t]of this.plugins)e.set(t,await this.healthCheck(t));return e}startHealthCheckTimer(){if(this.healthCheckTimer){this.log.warn("health check timer already running");return}this.healthCheckTimer=setInterval(async()=>{let e=await this.healthCheckAll();for(let[t,s]of e)s.healthy||this.log.warn("plugin health check failed",{id:t,error:s.error});},this.config.healthCheckInterval),this.log.info("health check timer started",{interval:this.config.healthCheckInterval});}stopHealthCheckTimer(){this.healthCheckTimer&&(clearInterval(this.healthCheckTimer),this.healthCheckTimer=null,this.log.info("health check timer stopped"));}async sendToPlugin(e,t){let s=this.plugins.get(e);if(!s)throw new Error(`Plugin not found: ${e}`);if(!s.started)throw new Error(`Plugin not started: ${e}`);return s.plugin.send(t)}};var o=chunk5ETPYYQR_cjs.a.create({service:"gateway:websocket-server"});function V(){return `ws_${Date.now()}_${Math.random().toString(36).substring(2,9)}`}function P(c,e){for(let[t,s]of e.entries())if(s===c)return t;return null}var w=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"server",null);chunkGY3SWWW3_cjs.a(this,"wss",null);chunkGY3SWWW3_cjs.a(this,"httpsServer",null);chunkGY3SWWW3_cjs.a(this,"httpRedirectServer",null);chunkGY3SWWW3_cjs.a(this,"connections",new Map);chunkGY3SWWW3_cjs.a(this,"subscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"sessionSubscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"sessionLastActivity",new Map);chunkGY3SWWW3_cjs.a(this,"cleanupInterval",null);chunkGY3SWWW3_cjs.a(this,"heartbeatInterval",null);chunkGY3SWWW3_cjs.a(this,"startTime",0);chunkGY3SWWW3_cjs.a(this,"running",false);this.config={...E,...e};}async start(){if(this.running){o.warn("WebSocket server is already running");return}o.debug("WebSocket server start: checking configuration",{port:this.config.port,hostname:this.config.hostname,path:this.config.path,https:this.config.https?.enabled??false}),this.startTime=Date.now();let e={onOpen:(n,r)=>{this.handleConnection(r);},onMessage:async(n,r)=>{try{let a=JSON.parse(n.data.toString());await this.handleMessage(r,a);}catch(a){o.error("Error processing WebSocket message",{error:a.message}),this.sendError(r,a.message);}},onClose:(n,r)=>{this.handleDisconnection(r);},onError:(n,r)=>{o.error("WebSocket error",{error:n.type});}},t=this.config.https?.enabled??false;o.debug("WebSocket server start: choosing server type",{https:t});try{t?(o.debug("WebSocket server start: starting HTTPS server"),await this.startHTTPSServer(e)):(o.debug("WebSocket server start: starting HTTP server"),await this.startHTTPServer(e)),o.debug("WebSocket server start: HTTP/HTTPS server started successfully");}catch(n){throw o.error("WebSocket server start: HTTP/HTTPS server failed",{error:n instanceof Error?n.message:String(n)}),n}o.debug("WebSocket server start: server initialization complete"),this.cleanupInterval=setInterval(()=>this.cleanupExpiredSessions(),3e4),this.heartbeatInterval=setInterval(()=>this.checkConnectionTimeout(),this.config.heartbeatInterval),this.running=true;let s=t?"wss":"ws";o.info("WebSocket server started",{url:`${s}://${this.config.hostname}:${this.config.port}${this.config.path}`});}async startHTTPServer(e){o.debug("HTTP server start: creating server",{port:this.config.port,hostname:this.config.hostname}),this.server=utils.createServer({port:this.config.port,hostname:this.config.hostname,routes:this.createRoutes(),timeout:this.config.connectionTimeout,cors:true}),o.debug("HTTP server start: server created, checking raw server");let t=this.server.raw;t?(o.debug("HTTP server start: raw server available, setting up WebSocket upgrade"),this.setupWebSocketUpgrade(t,e),o.debug("HTTP server start: WebSocket upgrade configured")):o.warn("HTTP server start: raw server is undefined");}async startHTTPSServer(e){let t=this.config.https,s=this.loadSSLCertificates(t);if(!s){o.error("Failed to load SSL certificates, falling back to HTTP"),await this.startHTTPServer(e);return}this.server=utils.createServer({port:this.config.port,hostname:this.config.hostname,routes:this.createRoutes(),timeout:this.config.connectionTimeout,cors:true});let n=this.server.raw;this.httpsServer=j__default.default.createServer(s,n.listeners("request")[0]),this.setupWebSocketUpgrade(this.httpsServer,e),await new Promise(r=>{this.httpsServer.listen(this.config.port,this.config.hostname,()=>{o.info("HTTPS server started",{port:this.config.port,hostname:this.config.hostname}),r();});}),t.forceRedirect&&t.httpPort&&this.startHTTPRedirectServer(t.httpPort);}loadSSLCertificates(e){try{return e.cert&&e.key?{cert:e.cert,key:e.key}:e.certPath&&e.keyPath?{cert:p__namespace.default.readFileSync(e.certPath),key:p__namespace.default.readFileSync(e.keyPath)}:null}catch(t){return o.error("Failed to load SSL certificates",{error:t.message}),null}}setupWebSocketUpgrade(e,t){this.wss=new ws.WebSocketServer({noServer:true}),e.on("upgrade",(s,n,r)=>{try{let g=new URL(s.url||"/",`http://${s.headers.host}`).pathname;(g===this.config.path||this.config.path==="/"&&(g==="/"||g===""))&&this.wss.handleUpgrade(s,n,r,l=>{t.onOpen?.({type:"open"},l),l.on("message",h=>{t.onMessage?.({data:h},l);}),l.on("close",(h,D)=>{t.onClose?.({code:h,reason:D},l);}),l.on("error",h=>{t.onError?.({type:"error",message:h.message},l);});});}catch{}});}startHTTPRedirectServer(e){this.httpRedirectServer=Q__default.default.createServer((t,s)=>{let r=`https://${t.headers.host?.split(":")[0]||this.config.hostname}:${this.config.port}${t.url}`;o.debug("Redirecting HTTP to HTTPS",{from:t.url,to:r}),s.writeHead(301,{Location:r,"Content-Type":"text/plain"}),s.end(`Redirecting to ${r}`);}),this.httpRedirectServer.listen(e,this.config.hostname,()=>{o.info("HTTP redirect server started",{port:e,hostname:this.config.hostname});});}async stop(){if(!this.running){o.warn("WebSocket server is not running");return}o.info("Stopping WebSocket server"),this.cleanupInterval&&(clearInterval(this.cleanupInterval),this.cleanupInterval=null),this.heartbeatInterval&&(clearInterval(this.heartbeatInterval),this.heartbeatInterval=null);for(let[e,t]of this.connections.entries())try{t.close(1e3,"Server shutting down");}catch(s){o.debug("Error closing connection",{clientId:e,error:String(s)});}this.connections.clear(),this.subscriptions.clear(),this.sessionSubscriptions.clear(),this.sessionLastActivity.clear(),this.wss&&(this.wss.close(),this.wss=null),this.httpsServer&&(await new Promise(e=>{this.httpsServer.close(()=>e());}),this.httpsServer=null),this.httpRedirectServer&&(await new Promise(e=>{this.httpRedirectServer.close(()=>e());}),this.httpRedirectServer=null),this.server&&(await this.server.stop(),this.server=null),this.running=false,o.info("WebSocket server stopped");}broadcastToSession(e,t){this.touchSession(e);let s=this.sessionSubscriptions.get(e);if(!s||s.size===0){o.debug("No clients subscribed to session",{sessionId:e});return}let n=t?.properties?.sessionID;if(typeof n=="string"&&n&&n!==e)return;let r=JSON.stringify({type:"event",sessionId:e,payload:t});for(let a of s){let g=this.subscriptions.get(a);if(g&&g.ws.readyState===g.ws.OPEN&&g.subscribedSessions.has(e)){if(g.subscribedEventTypes.size>0&&!g.subscribedEventTypes.has(t.type)){o.debug("Event type not subscribed, skipping",{clientId:a,sessionId:e,eventType:t.type,subscribedTypes:Array.from(g.subscribedEventTypes)});continue}g.ws.send(r),o.debug("Event pushed to client",{clientId:a,sessionId:e,eventType:t.type});}}}getConnectionCount(){return this.connections.size}getSubscriptionCount(){return this.sessionSubscriptions.size}getConnectionInfos(){let e=[];for(let[t,s]of this.subscriptions.entries())e.push({clientId:t,state:s.ws.readyState===s.ws.OPEN?"connected":"disconnected",connectedAt:s.connectedAt,lastActivityAt:s.lastActivityAt,authInfo:s.authInfo});return e}isRunning(){return this.running}handleConnection(e){if(this.connections.size>=(this.config.maxConnections??y.maxConnections)){o.warn("Maximum connections reached, rejecting new connection"),e.close(1013,"Maximum connections reached");return}let t=V();this.connections.set(t,e),this.subscriptions.set(t,{clientId:t,ws:e,subscribedSessions:new Set,subscribedEventTypes:new Set,connectedAt:Date.now(),lastActivityAt:Date.now()}),o.info("WebSocket client connected",{clientId:t,totalConnections:this.connections.size}),this.sendMessage(e,{type:"connected",clientId:t,payload:{version:"1.0.0",capabilities:["subscribe","message","interrupt","releaseSubscription"]}});}handleDisconnection(e){let t=P(e,this.connections);if(!t)return;this.connections.delete(t);let s=this.subscriptions.get(t);if(s){for(let n of s.subscribedSessions){let r=this.sessionSubscriptions.get(n);r&&(r.delete(t),r.size===0&&this.sessionSubscriptions.delete(n));}this.subscriptions.delete(t);}o.info("WebSocket client disconnected",{clientId:t,totalConnections:this.connections.size});}async handleMessage(e,t){let s=P(e,this.connections);if(!s){this.sendError(e,"Client not found");return}let n=this.subscriptions.get(s);switch(n&&(n.lastActivityAt=Date.now()),t.type){case "ping":this.sendMessage(e,{type:"pong"});break;case "initialize":await this.handleInitialize(e,t);break;case "subscribe":await this.handleSubscribe(e,t);break;case "unsubscribe":await this.handleUnsubscribe(e,t);break;case "message":await this.handleMessageRequest(e,t);break;case "interrupt":await this.handleInterrupt(e,t);break;case "releaseSubscription":await this.handleReleaseSubscription(e,t);break;default:this.sendError(e,`Unknown message type: ${t.type}`);}}async handleInitialize(e,t){this.sendMessage(e,{type:"initialize_response",success:true,result:{capabilities:["subscribe","message","interrupt","releaseSubscription"]}});}async handleSubscribe(e,t){if(t.type!=="subscribe"||!t.sessionId){this.sendError(e,"Invalid subscribe request");return}let s=P(e,this.connections);if(!s){this.sendError(e,"Client not found");return}let n=this.subscriptions.get(s);if(!n){this.sendError(e,"Subscription not found");return}if(n.subscribedSessions.add(t.sessionId),this.sessionSubscriptions.has(t.sessionId)||this.sessionSubscriptions.set(t.sessionId,new Set),this.sessionSubscriptions.get(t.sessionId).add(s),this.touchSession(t.sessionId),t.payload?.eventTypes)for(let r of t.payload.eventTypes)n.subscribedEventTypes.add(r);o.info("Client subscribed to session",{clientId:s,sessionId:t.sessionId}),this.sendMessage(e,{type:"subscribed",sessionId:t.sessionId});}async handleUnsubscribe(e,t){if(t.type!=="unsubscribe"||!t.sessionId){this.sendError(e,"Invalid unsubscribe request");return}let s=P(e,this.connections);if(!s)return;let n=this.subscriptions.get(s);if(!n)return;n.subscribedSessions.delete(t.sessionId);let r=this.sessionSubscriptions.get(t.sessionId);r&&(r.delete(s),r.size===0&&this.sessionSubscriptions.delete(t.sessionId)),o.info("Client unsubscribed from session",{clientId:s,sessionId:t.sessionId}),this.sendMessage(e,{type:"unsubscribed",sessionId:t.sessionId});}async handleMessageRequest(e,t){if(t.type!=="message"||!t.sessionId||!t.payload){this.sendError(e,"Invalid message request");return}o.debug("Message received",{sessionId:t.sessionId}),this.sendMessage(e,{type:"response",sessionId:t.sessionId,success:true,result:{received:true}});}async handleInterrupt(e,t){if(t.type!=="interrupt"||!t.sessionId){this.sendError(e,"Invalid interrupt request");return}o.info("Interrupt requested via WebSocket",{sessionId:t.sessionId}),this.sendMessage(e,{type:"interrupted",sessionId:t.sessionId,success:true});}async handleReleaseSubscription(e,t){if(t.type!=="releaseSubscription"||!t.sessionId){this.sendError(e,"Invalid releaseSubscription request");return}o.info("Release subscription requested via WebSocket",{sessionId:t.sessionId}),this.releaseSessionResources(t.sessionId),this.sendMessage(e,{type:"subscriptionReleased",sessionId:t.sessionId});}touchSession(e){this.sessionLastActivity.set(e,Date.now());}checkConnectionTimeout(){let e=Date.now(),t=this.config.connectionTimeout??y.connectionTimeout,s=[];for(let[n,r]of this.subscriptions.entries())e-r.lastActivityAt>t&&s.push(n);for(let n of s){let r=this.subscriptions.get(n);if(r){o.info("Closing inactive connection",{clientId:n,lastActivityAt:r.lastActivityAt,timeoutMs:t});try{r.ws.close(1001,"Connection timeout");}catch(a){o.debug("Error closing timed out connection",{clientId:n,error:String(a)});}}}s.length>0&&o.debug("Closed timed out connections",{count:s.length,remainingConnections:this.connections.size-s.length});}cleanupExpiredSessions(){let e=Date.now(),t=[],s=this.config.sessionExpireMs??y.sessionExpireMs;for(let[n,r]of this.sessionLastActivity.entries())e-r>s&&t.push(n);for(let n of t)o.info("Session expired due to inactivity, releasing resources",{sessionId:n,lastActivity:this.sessionLastActivity.get(n),expireMs:s}),this.releaseSessionResources(n),this.sessionLastActivity.delete(n);t.length>0&&o.debug("Cleaned up expired sessions",{count:t.length,remainingActive:this.sessionLastActivity.size});}releaseSessionResources(e){let t=this.sessionSubscriptions.get(e);if(t){for(let s of t){let n=this.subscriptions.get(s);n&&n.ws.readyState===n.ws.OPEN&&this.sendMessage(n.ws,{type:"session_closed",sessionId:e,payload:{reason:"Session resources released due to expiration or explicit release"}});}for(let s of t){let n=this.subscriptions.get(s);n&&n.subscribedSessions.delete(e);}this.sessionSubscriptions.delete(e),o.info("Session resources released",{sessionId:e,clientCount:t.size});}}sendMessage(e,t){e.readyState===e.OPEN&&e.send(JSON.stringify(t));}sendError(e,t,s){this.sendMessage(e,{type:"error",message:t,code:s});}createRoutes(){return [{method:"get",path:"/health",handler:e=>{let t={status:"healthy",timestamp:Date.now(),version:"1.0.0"};return e.json(t)}},{method:"get",path:"/status",handler:e=>{let t={status:this.running?"running":"stopped",connections:this.connections.size,sessions:this.sessionSubscriptions.size,subscriptions:this.subscriptions.size,uptime:this.running?Date.now()-this.startTime:0,version:"1.0.0"};return e.json(t)}},{method:"post",path:"/initialize",handler:async e=>{let t=await e.req.json().catch(()=>({}));return o.debug("ACP initialize request received",{body:t}),e.json({success:true,result:{capabilities:["subscribe","message","interrupt","releaseSubscription"],version:"1.0.0"}})}},{method:"post",path:"/session",handler:async e=>{await e.req.json().catch(()=>({}));let n={id:`session_${Date.now()}_${Math.random().toString(36).substring(2,9)}`,createdAt:Date.now()};return e.json(n)}},{method:"post",path:"/session/:sessionId/prompt",handler:async e=>{let t=e.req.param("sessionId");await e.req.json().catch(()=>({}));if(!t)return e.json({error:"No sessionId provided"},400);let r={messageId:`msg_${Date.now()}_${Math.random().toString(36).substring(2,9)}`,sessionId:t,createdAt:Date.now()};return e.status(202),e.json(r)}},{method:"get",path:"/subscriptions",handler:e=>{let t=Array.from(this.sessionSubscriptions.entries()).map(([n,r])=>({sessionId:n,subscriberCount:r.size,subscribers:Array.from(r)})),s={subscriptions:t,total:t.length};return e.json(s)}}]}};function K(c){let e={platform:c.channel?.platform??"api",channelId:c.channel?.channelId??"unknown",chatId:c.channel?.chatId,userId:c.channel?.userId};return {id:c.messageId??`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:c.sessionId,type:c.type,content:Array.isArray(c.content)?c.content:[{type:"text",text:String(c.content??"")}],metadata:{channel:e,replyToMessageId:c.replyToMessageId},timestamp:c.timestamp??Date.now()}}var O=class{constructor(e){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:server"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"router");chunkGY3SWWW3_cjs.a(this,"sessionManager");chunkGY3SWWW3_cjs.a(this,"pluginLoader");chunkGY3SWWW3_cjs.a(this,"messageStore");chunkGY3SWWW3_cjs.a(this,"wsServer",null);chunkGY3SWWW3_cjs.a(this,"running",false);chunkGY3SWWW3_cjs.a(this,"connections",0);this.config=e,this.router=new x,this.sessionManager=new v,this.pluginLoader=new C({healthCheckInterval:6e4});let t=chunk4PHQN25U_cjs.Q(),s=chunk4PHQN25U_cjs.N();this.messageStore=new S({storagePath:`${t.Path.data}/gateway-messages`}),s&&chunk4PHQN25U_cjs.O().setMessageStorage({store:async r=>{let a=K(r);await this.messageStore.store(a);},updateStatus:async(r,a,g)=>{await this.messageStore.updateStatus(r,a,g);}}),this.pluginLoader.setMessageHandler({onMessage:n=>this.onMessage(n),onEvent:n=>this.onEvent(n)});}async start(){if(this.running){this.log.warn("gateway server is already running");return}this.log.info("starting gateway server",{port:this.config.port,hostname:this.config.hostname}),await this.sessionManager.initialize(),this.log.debug("starting WebSocket server"),this.wsServer=new w({port:this.config.port,hostname:this.config.hostname,path:String(this.config.path??"/")}),await this.wsServer.start(),this.log.debug("WebSocket server started"),await this.pluginLoader.startAll(),this.pluginLoader.startHealthCheckTimer(),this.running=true,this.log.info("gateway server started");}async stop(){if(!this.running){this.log.warn("gateway server is not running");return}this.log.info("stopping gateway server"),this.wsServer&&(this.log.debug("stopping WebSocket server"),await this.wsServer.stop(),this.wsServer=null,this.log.debug("WebSocket server stopped")),this.pluginLoader.stopHealthCheckTimer(),await this.pluginLoader.stopAll(),await this.router.shutdown(),await this.sessionManager.close(),this.running=false,this.log.info("gateway server stopped");}getStatus(){return {running:this.running,port:this.config.port,hostname:this.config.hostname,connections:this.connections}}async onMessage(e){if(this.log.debug("message received",{id:e.id,sessionId:e.sessionId,type:e.type}),this.messageStore.isProcessed(e.id)){this.log.warn("message already processed, skipping",{id:e.id});return}await this.storeMessage(e),await this.sessionManager.incrementMessageCountOrCreate(e.sessionId,e.metadata.channel),(await this.router.route(e)).broadcastCount===0&&e.type==="input"&&this.log.debug("no subscribers, message pending",{id:e.id});}async storeMessage(e){if(await this.messageStore.store(e),chunk4PHQN25U_cjs.N())try{let t=chunk4PHQN25U_cjs.O(),s={name:plugin.HookEvent.GatewayMessageReceive,data:{messageId:e.id,sessionId:e.sessionId,type:e.type,content:e.content,channel:e.metadata.channel,replyToMessageId:e.metadata.replyToMessageId,timestamp:e.timestamp||Date.now()}};await t.hookRegistry.emit(s);}catch{this.log.debug("hook trigger skipped (no context)");}}async processMessage(e,t,s,n={}){if(await this.messageStore.updateStatus(t,"processing",{processedBy:s}),chunk4PHQN25U_cjs.N())try{await chunk4PHQN25U_cjs.O().hookRegistry.emit({name:plugin.HookEvent.GatewayMessageProcess,data:{messageId:t,sessionId:e,processorId:s,processType:n.processType||"direct",agent:n.agent,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}}async completeMessage(e,t,s,n={}){let{success:r=true,error:a,responseMessageId:g,tokens:l,duration:h}=n;if(await this.messageStore.updateStatus(t,r?"completed":"failed",{processedBy:s,error:a}),chunk4PHQN25U_cjs.N())try{await chunk4PHQN25U_cjs.O().hookRegistry.emit({name:plugin.HookEvent.GatewayMessageComplete,data:{messageId:t,sessionId:e,processorId:s,status:r?"completed":"failed",error:a,responseMessageId:g,tokens:l,duration:h,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}await this.router.completeMessage(e,t,s,r?"completed":"failed");}async onEvent(e){switch(this.log.debug("channel event",{type:e.type,channelId:e.channelId}),e.type){case "connected":this.connections++;break;case "disconnected":this.connections--;break;case "error":this.log.error("channel error",{channelId:e.channelId,data:e.data});break}}async registerChannel(e,t){let s=t??{enabled:true,platform:{},session:{autoCreate:true,resetPolicy:"onNewConversation"},routing:{broadcast:true,dispatchToSubAgent:false}};await this.pluginLoader.register(e,s),this.running&&await this.pluginLoader.start(e.id);}async unregisterChannel(e){await this.pluginLoader.unregister(e);}async subscribe(e,t){await this.router.subscribe(e,t),await this.sessionManager.addSubscriber(e,t.id);}async unsubscribe(e){await this.router.unsubscribe(e);}async send(e){await this.onMessage(e);}async sendToChannel(e,t,s={}){let{proactive:n=false,replyToMessageId:r}=s;if(chunk4PHQN25U_cjs.N())try{await chunk4PHQN25U_cjs.O().hookRegistry.emit({name:plugin.HookEvent.GatewayMessageSend,data:{messageId:t.id,sessionId:t.sessionId,type:t.type,channel:t.metadata.channel,proactive:n,replyToMessageId:r,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}return this.pluginLoader.sendToPlugin(e,t)}async sendProactiveMessage(e,t,s,n){let r={id:`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:t,type:"output",content:s,metadata:{channel:n},timestamp:Date.now()};return this.sendToChannel(e,r,{proactive:true})}getSessionManager(){return this.sessionManager}getRouter(){return this.router}getPluginLoader(){return this.pluginLoader}getMessageStore(){return this.messageStore}async getMessageHistory(e,t){return this.messageStore.getSessionHistory(e,t)}};
1
+ 'use strict';var chunkEK6MKXZH_cjs=require('./chunk-EK6MKXZH.cjs'),chunk5ETPYYQR_cjs=require('./chunk-5ETPYYQR.cjs'),chunkGY3SWWW3_cjs=require('./chunk-GY3SWWW3.cjs'),p=require('fs'),L=require('path'),utils=require('@easbot/utils'),ws=require('ws'),j=require('https'),Q=require('http'),plugin=require('@easbot/plugin');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}function _interopNamespace(e){if(e&&e.__esModule)return e;var n=Object.create(null);if(e){Object.keys(e).forEach(function(k){if(k!=='default'){var d=Object.getOwnPropertyDescriptor(e,k);Object.defineProperty(n,k,d.get?d:{enumerable:true,get:function(){return e[k]}});}})}n.default=e;return Object.freeze(n)}var p__namespace=/*#__PURE__*/_interopNamespace(p);var L__namespace=/*#__PURE__*/_interopNamespace(L);var j__default=/*#__PURE__*/_interopDefault(j);var Q__default=/*#__PURE__*/_interopDefault(Q);var M=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:lock"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"locks",new Map);chunkGY3SWWW3_cjs.a(this,"cleanupTimer",null);this.config={defaultTimeout:e.defaultTimeout??6e4,checkInterval:e.checkInterval??1e4};}tryAcquire(e,t,s,n){let r=this.locks.get(e);if(r)if(this.isLockExpired(r))this.log.warn("lock expired, releasing",{sessionId:e,oldHolder:r.holderId,newHolder:s}),this.locks.delete(e);else return this.log.debug("lock already held",{sessionId:e,holder:r.holderId,messageId:r.messageId}),false;let a={sessionId:e,messageId:t,holderId:s,acquiredAt:Date.now(),timeout:n??this.config.defaultTimeout};return this.locks.set(e,a),this.log.debug("lock acquired",{sessionId:e,messageId:t,subscriberId:s}),true}release(e,t){let s=this.locks.get(e);return s?s.holderId!==t?(this.log.warn("cannot release lock held by another",{sessionId:e,holder:s.holderId,requester:t}),false):(this.locks.delete(e),this.log.debug("lock released",{sessionId:e,subscriberId:t}),true):(this.log.debug("no lock to release",{sessionId:e}),false)}forceRelease(e){let t=this.locks.get(e);return t?(this.locks.delete(e),this.log.warn("lock force released",{sessionId:e,holder:t.holderId,messageId:t.messageId}),true):false}isLockExpired(e){return Date.now()-e.acquiredAt>e.timeout}getLock(e){return this.locks.get(e)}isLocked(e){let t=this.locks.get(e);return t?this.isLockExpired(t)?(this.locks.delete(e),false):true:false}getHolder(e){let t=this.locks.get(e);if(!(!t||this.isLockExpired(t)))return t.holderId}startCleanup(){this.cleanupTimer||(this.cleanupTimer=setInterval(()=>{this.cleanupExpired();},this.config.checkInterval),this.log.info("lock cleanup started",{interval:this.config.checkInterval}));}stopCleanup(){this.cleanupTimer&&(clearInterval(this.cleanupTimer),this.cleanupTimer=null,this.log.info("lock cleanup stopped"));}cleanupExpired(){let e=0;for(let[t,s]of this.locks)this.isLockExpired(s)&&(this.locks.delete(t),e++,this.log.warn("expired lock cleaned up",{sessionId:t,holder:s.holderId,messageId:s.messageId}));return e>0&&this.log.info("cleaned up expired locks",{count:e}),e}getAllLocks(){return [...this.locks.values()]}getLockCount(){return this.locks.size}};var I=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:retry"}));chunkGY3SWWW3_cjs.a(this,"policy");chunkGY3SWWW3_cjs.a(this,"retryQueue",new Map);chunkGY3SWWW3_cjs.a(this,"retryTimer",null);chunkGY3SWWW3_cjs.a(this,"handlers",new Map);this.policy={maxRetries:e.maxRetries??3,initialDelay:e.initialDelay??1e3,maxDelay:e.maxDelay??3e4,backoffMultiplier:e.backoffMultiplier??2};}registerHandler(e,t){this.handlers.set(e,t);}unregisterHandler(e){this.handlers.delete(e);}scheduleRetry(e,t){let s=e.id,n=this.retryQueue.get(s);if(n||(n={messageId:s,retryCount:0,nextRetryAt:0,errors:[]},this.retryQueue.set(s,n)),n.errors.push({error:t.message,timestamp:Date.now()}),n.retryCount>=this.policy.maxRetries)return this.log.error("max retries exceeded",{messageId:s,retryCount:n.retryCount,maxRetries:this.policy.maxRetries}),false;let r=Math.min(this.policy.initialDelay*this.policy.backoffMultiplier**n.retryCount,this.policy.maxDelay);return n.retryCount++,n.nextRetryAt=Date.now()+r,this.log.warn("retry scheduled",{messageId:s,retryCount:n.retryCount,delay:r,error:t.message}),true}getPendingRetries(){let e=Date.now(),t=[];for(let s of this.retryQueue.values())s.nextRetryAt<=e&&s.retryCount<=this.policy.maxRetries&&t.push(s);return t}async executeRetry(e){if(!this.handlers.get(e.messageId))return this.log.warn("no handler for retry",{messageId:e.messageId}),false;this.retryQueue.delete(e.messageId);try{return this.log.info("executing retry",{messageId:e.messageId,retryCount:e.retryCount}),!0}catch(s){return this.log.error("retry execution failed",{messageId:e.messageId,error:String(s)}),false}}clearRetry(e){this.retryQueue.delete(e),this.handlers.delete(e),this.log.debug("retry cleared",{messageId:e});}startProcessing(){this.retryTimer||(this.retryTimer=setInterval(async()=>{let e=this.getPendingRetries();for(let t of e)await this.executeRetry(t);},1e3),this.log.info("retry processing started"));}stopProcessing(){this.retryTimer&&(clearInterval(this.retryTimer),this.retryTimer=null,this.log.info("retry processing stopped"));}getStats(){let e=0;for(let t of this.retryQueue.values())e+=t.errors.length;return {pendingCount:this.retryQueue.size,totalErrors:e}}};function H(c,e,t,s){return {id:`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:c,type:e,content:t,metadata:{channel:{platform:s?.channel?.platform??"api",channelId:s?.channel?.channelId??"default",userId:s?.channel?.userId,chatId:s?.channel?.chatId,...Object.fromEntries(Object.entries(s?.channel??{}).filter(([n])=>!["platform","channelId","userId","chatId"].includes(n)))},context:s?.context,agent:s?.agent},timestamp:Date.now()}}function N(c,e,t,s){return H(c,e,[{type:"text",text:t}],s)}function T(c){let e=[c.platform,c.channelId];return c.chatId&&e.push(c.chatId),c.userId&&e.push(c.userId),e.join("_")}function R(){return {status:"active",messageCount:0,lastMessageAt:null}}function k(c,e){return `sub_${c}_${e}`}var E=chunkEK6MKXZH_cjs.v,y={maxConnections:1e3,connectionTimeout:6e4,sessionExpireMs:3e5};var B={type:"token",enabled:true,tokens:[],onInvalid:"reject",allowAnonymous:false,defaultPermissions:[]};var q={heartbeatInterval:3e4,heartbeatTimeout:9e4,heartbeatCheckInterval:1e4,autoDeregisterTimeout:3e5,maxAgents:100,maxRegistrations:1e3};var F={mode:"both",interval:3e4,pushEvents:["register","deregister","status_change"],conflictResolution:"latest",remoteNodes:[]};var U={maxConnectionsPerChannel:100,idleTimeout:3e5,reuseStrategy:"lru",healthCheckInterval:6e4,acquireTimeout:5e3,warmupCount:0};var x=class{constructor(){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:router"}));chunkGY3SWWW3_cjs.a(this,"subscribers",new Map);chunkGY3SWWW3_cjs.a(this,"subscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"sessionSubscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"channelPlugins",new Map);chunkGY3SWWW3_cjs.a(this,"lockManager");chunkGY3SWWW3_cjs.a(this,"retryManager");chunkGY3SWWW3_cjs.a(this,"cancelHandlers",new Set);this.lockManager=new M,this.retryManager=new I,this.lockManager.startCleanup();}onCancel(e){this.cancelHandlers.add(e);}offCancel(e){this.cancelHandlers.delete(e);}async emitCancel(e){for(let t of this.cancelHandlers)try{await t(e);}catch(s){this.log.error("cancel handler error",{error:String(s)});}}async route(e){this.log.debug("routing message",{id:e.id,sessionId:e.sessionId,type:e.type});let t=this.sessionSubscriptions.get(e.sessionId);if(!t||t.size===0)return this.log.debug("no subscribers for session",{sessionId:e.sessionId}),{success:true,broadcastCount:0,dispatchedToSubAgent:false};let s=0,n=null,r=[];for(let a of t){let g=this.subscriptions.get(a);if(!g)continue;let l=this.subscribers.get(g.subscriberId);if(l){if(e.type==="input"){if(!this.lockManager.tryAcquire(e.sessionId,e.id,l.id)){this.log.debug("skipping subscriber, lock held by another",{subscriberId:l.id,holder:this.lockManager.getHolder(e.sessionId)});continue}n=l.id;}try{await l.connection.send(e),s++,l.lastActiveAt=Date.now();}catch(h){this.log.warn("failed to send message to subscriber",{subscriberId:l.id,error:String(h)}),r.push(h instanceof Error?h:new Error(String(h))),n===l.id&&(this.lockManager.release(e.sessionId,l.id),n=null);}}}return this.log.debug("message routed",{id:e.id,broadcastCount:s,errorCount:r.length,processedBy:n}),{success:r.length===0,broadcastCount:s,dispatchedToSubAgent:false,error:r.length>0?r.map(a=>a.message).join("; "):void 0}}async completeMessage(e,t,s,n="completed"){this.lockManager.release(e,s),await this.emitCancel({sessionId:e,messageId:t,processedBy:s,reason:n,timestamp:Date.now()}),this.log.info("message completed",{sessionId:e,messageId:t,subscriberId:s,reason:n});}scheduleRetry(e,t){return this.retryManager.scheduleRetry(e,t)}async subscribe(e,t){let s=k(t.id,e);if(this.subscriptions.has(s))return this.log.debug("already subscribed",{subscriberId:t.id,sessionId:e}),this.subscriptions.get(s);let n={id:s,subscriberId:t.id,sessionId:e,createdAt:Date.now()};return this.subscribers.has(t.id)||this.subscribers.set(t.id,t),this.subscriptions.set(s,n),this.sessionSubscriptions.has(e)||this.sessionSubscriptions.set(e,new Set),this.sessionSubscriptions.get(e).add(s),t.sessions.add(e),this.log.info("subscribed to session",{subscriberId:t.id,sessionId:e,subscriptionId:s}),n}async unsubscribe(e){let t=this.subscriptions.get(e);if(!t){this.log.warn("subscription not found",{subscriptionId:e});return}this.subscriptions.delete(e);let s=this.sessionSubscriptions.get(t.sessionId);s&&(s.delete(e),s.size===0&&this.sessionSubscriptions.delete(t.sessionId));let n=this.subscribers.get(t.subscriberId);n&&n.sessions.delete(t.sessionId),this.log.info("unsubscribed from session",{subscriberId:t.subscriberId,sessionId:t.sessionId,subscriptionId:e});}async unsubscribeAll(e){let t=this.subscribers.get(e);if(!t)return;let s=[...t.sessions];for(let n of s){let r=k(e,n);await this.unsubscribe(r);}this.subscribers.delete(e),this.log.info("unsubscribed all sessions",{subscriberId:e});}async registerChannel(e){if(this.channelPlugins.has(e.id)){this.log.warn("channel plugin already registered",{id:e.id});return}this.channelPlugins.set(e.id,e),this.log.info("channel plugin registered",{id:e.id,platform:e.platform});}async unregisterChannel(e){let t=this.channelPlugins.get(e);if(!t){this.log.warn("channel plugin not found",{id:e});return}await t.stop(),this.channelPlugins.delete(e),this.log.info("channel plugin unregistered",{id:e});}getSubscriberCount(e){let t=this.sessionSubscriptions.get(e);return t?t.size:0}getAllSubscribers(){return [...this.subscribers.values()]}getChannelPlugin(e){return this.channelPlugins.get(e)}getAllChannelPlugins(){return [...this.channelPlugins.values()]}getLockManager(){return this.lockManager}getRetryManager(){return this.retryManager}async shutdown(){this.lockManager.stopCleanup(),this.retryManager.stopProcessing(),this.log.info("router shutdown");}};var S=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:message-store"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"messages",new Map);chunkGY3SWWW3_cjs.a(this,"sessionMessages",new Map);this.config={storagePath:e.storagePath??"./data/gateway-messages",maxRetentionDays:e.maxRetentionDays??30,maxRecords:e.maxRecords??1e5};}async store(e){let t={message:e,status:"pending",processedBy:null,processedAt:null,retryCount:0,error:null,createdAt:Date.now(),updatedAt:Date.now()};return this.messages.set(e.id,t),this.sessionMessages.has(e.sessionId)||this.sessionMessages.set(e.sessionId,new Set),this.sessionMessages.get(e.sessionId).add(e.id),this.log.debug("message stored",{id:e.id,sessionId:e.sessionId,type:e.type}),t}get(e){return this.messages.get(e)}async updateStatus(e,t,s={}){let n=this.messages.get(e);if(!n){this.log.warn("message not found for status update",{messageId:e});return}n.status=t,n.updatedAt=Date.now(),s.processedBy&&(n.processedBy=s.processedBy),(t==="completed"||t==="failed")&&(n.processedAt=Date.now()),s.error&&(n.error=s.error),this.log.debug("message status updated",{id:e,status:t,processedBy:s.processedBy});}incrementRetry(e){let t=this.messages.get(e);return t?(t.retryCount++,t.updatedAt=Date.now(),t.retryCount):0}isProcessed(e){let t=this.messages.get(e);return t?t.status==="completed"||t.status==="processing":false}getSessionHistory(e,t=100){let s=this.sessionMessages.get(e);if(!s)return [];let n=[];for(let r of s){let a=this.messages.get(r);a&&n.push(a);}return n.sort((r,a)=>r.createdAt-a.createdAt),n.slice(-t)}getPendingMessages(e){let t=[];for(let s of this.messages.values())s.status==="pending"&&(!e||s.message.sessionId===e)&&t.push(s);return t}getFailedMessages(e=3){let t=[];for(let s of this.messages.values())s.status==="failed"&&s.retryCount<e&&t.push(s);return t}async cleanup(){let e=Date.now(),t=this.config.maxRetentionDays*24*60*60*1e3,s=[];for(let[n,r]of this.messages)e-r.createdAt>t&&s.push(n);for(let n of s){let r=this.messages.get(n);if(r){let a=this.sessionMessages.get(r.message.sessionId);a&&a.delete(n),this.messages.delete(n);}}return s.length>0&&this.log.info("cleaned up expired messages",{count:s.length}),s.length}getStats(){let e={total:this.messages.size,pending:0,processing:0,completed:0,failed:0,cancelled:0};for(let t of this.messages.values())e[t.status]++;return e}};var b=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:session-store"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"sessions",new Map);chunkGY3SWWW3_cjs.a(this,"channelSessions",new Map);chunkGY3SWWW3_cjs.a(this,"autoSaveTimer",null);chunkGY3SWWW3_cjs.a(this,"isDirty",false);this.config={storagePath:e.storagePath??"./data/gateway-sessions",autoSaveInterval:e.autoSaveInterval??6e4,maxRetentionDays:e.maxRetentionDays??7,enablePersistence:e.enablePersistence??true};}async initialize(){if(!this.config.enablePersistence){this.log.info("session persistence disabled");return}await this.ensureStorageDir(),await this.load(),this.startAutoSave(),this.log.info("session store initialized",{sessionCount:this.sessions.size,storagePath:this.config.storagePath});}async close(){this.stopAutoSave(),this.isDirty&&await this.save(),this.log.info("session store closed");}async set(e){this.sessions.set(e.id,e),this.channelSessions.has(e.channel.channelId)||this.channelSessions.set(e.channel.channelId,new Set),this.channelSessions.get(e.channel.channelId).add(e.id),this.markDirty();}get(e){return this.sessions.get(e)}async delete(e){let t=this.sessions.get(e);if(!t)return;this.sessions.delete(e);let s=this.channelSessions.get(t.channel.channelId);s&&(s.delete(e),s.size===0&&this.channelSessions.delete(t.channel.channelId)),this.markDirty();}getAll(){return [...this.sessions.values()]}getByChannel(e){let t=this.channelSessions.get(e);return t?[...t].map(s=>this.sessions.get(s)).filter(s=>s!==void 0):[]}size(){return this.sessions.size}markDirty(){this.isDirty=true;}async save(){if(!this.config.enablePersistence)return;let e=this.getSessionFilePath(),t=this.serializeAll();try{await this.ensureStorageDir(),await p__namespace.promises.writeFile(e,JSON.stringify(t,null,2),"utf-8"),this.isDirty=!1,this.log.debug("sessions saved",{count:t.length});}catch(s){throw this.log.error("failed to save sessions",{error:s}),s}}async load(){if(!this.config.enablePersistence)return;let e=this.getSessionFilePath();try{if(!p__namespace.existsSync(e)){this.log.debug("no saved sessions found");return}let t=await p__namespace.promises.readFile(e,"utf-8"),s=JSON.parse(t);this.sessions.clear(),this.channelSessions.clear();let n=Date.now(),r=this.config.maxRetentionDays*24*60*60*1e3;for(let a of s){if(n-a.lastActiveAt>r)continue;let g={id:a.id,channel:a.channel,backendSessionId:a.backendSessionId,subscribers:new Set(a.subscribers),state:a.state,createdAt:a.createdAt,lastActiveAt:a.lastActiveAt};this.sessions.set(g.id,g),this.channelSessions.has(g.channel.channelId)||this.channelSessions.set(g.channel.channelId,new Set),this.channelSessions.get(g.channel.channelId).add(g.id);}this.log.info("sessions loaded",{loaded:this.sessions.size,skipped:s.length-this.sessions.size});}catch(t){this.log.error("failed to load sessions",{error:t});}}serializeAll(){let e=[];for(let t of this.sessions.values())e.push(this.serialize(t));return e}serialize(e){return {id:e.id,channel:e.channel,backendSessionId:e.backendSessionId,subscribers:[...e.subscribers],state:e.state,createdAt:e.createdAt,lastActiveAt:e.lastActiveAt}}getSessionFilePath(){return L__namespace.join(this.config.storagePath,"sessions.json")}async ensureStorageDir(){p__namespace.existsSync(this.config.storagePath)||await p__namespace.promises.mkdir(this.config.storagePath,{recursive:true});}startAutoSave(){this.autoSaveTimer||(this.autoSaveTimer=setInterval(async()=>{this.isDirty&&await this.save();},this.config.autoSaveInterval));}stopAutoSave(){this.autoSaveTimer&&(clearInterval(this.autoSaveTimer),this.autoSaveTimer=null);}async cleanup(){let e=Date.now(),t=this.config.maxRetentionDays*24*60*60*1e3,s=[];for(let n of this.sessions.values())e-n.lastActiveAt>t&&s.push(n.id);for(let n of s)await this.delete(n);return s.length>0&&(this.log.info("cleaned up expired sessions",{count:s.length}),await this.save()),s.length}getStats(){let e={total:this.sessions.size,byStatus:{active:0,idle:0,closed:0},byPlatform:{}};for(let t of this.sessions.values()){e.byStatus[t.state.status]++;let s=t.channel.platform;e.byPlatform[s]=(e.byPlatform[s]||0)+1;}return e}};var v=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:session"}));chunkGY3SWWW3_cjs.a(this,"sessionStore");chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"initialized",false);this.config={enablePersistence:e.enablePersistence??true,storagePath:e.storagePath??"./data/gateway-sessions",autoSaveInterval:e.autoSaveInterval??6e4,maxRetentionDays:e.maxRetentionDays??7},this.sessionStore=new b({storagePath:this.config.storagePath,autoSaveInterval:this.config.autoSaveInterval,maxRetentionDays:this.config.maxRetentionDays,enablePersistence:this.config.enablePersistence});}async initialize(){this.initialized||(await this.sessionStore.initialize(),this.initialized=true,this.log.info("session manager initialized",{sessionCount:this.sessionStore.size(),persistenceEnabled:this.config.enablePersistence}));}async close(){await this.sessionStore.close(),this.log.info("session manager closed");}async getOrCreate(e,t={}){let s=T(e),n=this.sessionStore.get(s);return n?(n.lastActiveAt=Date.now(),this.log.debug("session found",{sessionId:s}),n):(n={id:s,channel:e,backendSessionId:null,subscribers:new Set,state:R(),createdAt:Date.now(),lastActiveAt:Date.now()},await this.sessionStore.set(n),this.log.info("session created",{sessionId:s,platform:e.platform,channelId:e.channelId,chatId:e.chatId,userId:e.userId}),n)}get(e){return this.sessionStore.get(e)}async closeSession(e){let t=this.sessionStore.get(e);if(!t){this.log.warn("session not found",{sessionId:e});return}t.state.status="closed",await this.sessionStore.delete(e),this.log.info("session closed",{sessionId:e});}async closeById(e){return this.closeSession(e)}async bindBackendSession(e,t){let s=this.sessionStore.get(e);if(!s)throw new Error(`Session not found: ${e}`);s.backendSessionId=t,await this.sessionStore.set(s),this.log.info("backend session bound",{gatewaySessionId:e,backendSessionId:t});}async unbindBackendSession(e){let t=this.sessionStore.get(e);if(!t){this.log.warn("session not found",{sessionId:e});return}t.backendSessionId=null,await this.sessionStore.set(t),this.log.info("backend session unbound",{gatewaySessionId:e});}getState(e){return this.sessionStore.get(e)?.state}async incrementMessageCountOrCreate(e,t){let s=this.sessionStore.get(e);if(!s){let n=t??{platform:"api",channelId:"default"};s=await this.getOrCreate(n);}return s.state.messageCount++,s.state.lastMessageAt=Date.now(),s.lastActiveAt=Date.now(),await this.sessionStore.set(s),s}async incrementMessageCount(e){let t=this.sessionStore.get(e);t&&(t.state.messageCount++,t.state.lastMessageAt=Date.now(),t.lastActiveAt=Date.now(),await this.sessionStore.set(t));}async addSubscriber(e,t){let s=this.sessionStore.get(e);if(!s){this.log.warn("session not found",{sessionId:e});return}s.subscribers.add(t),await this.sessionStore.set(s),this.log.debug("subscriber added to session",{sessionId:e,subscriberId:t});}async removeSubscriber(e,t){let s=this.sessionStore.get(e);s&&(s.subscribers.delete(t),await this.sessionStore.set(s),this.log.debug("subscriber removed from session",{sessionId:e,subscriberId:t}));}query(e){let t=this.sessionStore.getAll();return e.platform&&(t=t.filter(s=>s.channel.platform===e.platform)),e.channelId&&(t=t.filter(s=>s.channel.channelId===e.channelId)),e.userId&&(t=t.filter(s=>s.channel.userId===e.userId)),e.status&&(t=t.filter(s=>s.state.status===e.status)),t}getSessionsByChannel(e){return this.sessionStore.getByChannel(e)}getAllSessions(){return this.sessionStore.getAll()}getSessionCount(){return this.sessionStore.size()}async cleanupIdleSessions(e){let t=Date.now(),s=[];for(let n of this.sessionStore.getAll())n.state.status==="idle"&&t-n.lastActiveAt>e&&s.push(n.id);for(let n of s)await this.closeSession(n);return s.length>0&&this.log.info("cleaned up idle sessions",{count:s.length}),s.length}async save(){this.log.debug("session save triggered");}getStats(){return this.sessionStore.getStats()}};var C=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:plugin-loader"}));chunkGY3SWWW3_cjs.a(this,"plugins",new Map);chunkGY3SWWW3_cjs.a(this,"messageHandler",null);chunkGY3SWWW3_cjs.a(this,"healthCheckTimer",null);chunkGY3SWWW3_cjs.a(this,"config");this.config={pluginDir:e.pluginDir??"./plugins",autoDiscover:e.autoDiscover??false,healthCheckInterval:e.healthCheckInterval??6e4,healthCheckTimeout:e.healthCheckTimeout??5e3};}setMessageHandler(e){this.messageHandler=e;}async register(e,t){if(this.plugins.has(e.id)){this.log.warn("plugin already registered",{id:e.id});return}this.plugins.set(e.id,{plugin:e,config:t,started:false}),this.log.info("plugin registered",{id:e.id,platform:e.platform,name:e.name});}async unregister(e){let t=this.plugins.get(e);if(!t){this.log.warn("plugin not found",{id:e});return}t.started&&await this.stop(e),this.plugins.delete(e),this.log.info("plugin unregistered",{id:e});}async start(e){let t=this.plugins.get(e);if(!t)throw new Error(`Plugin not found: ${e}`);if(t.started){this.log.warn("plugin already started",{id:e});return}if(!this.messageHandler)throw new Error("Message handler not set");if(!t.config.enabled){this.log.warn("plugin is disabled",{id:e});return}this.log.info("starting plugin",{id:e});try{await t.plugin.start(t.config,this.messageHandler),t.started=!0,this.log.info("plugin started",{id:e});}catch(s){throw this.log.error("failed to start plugin",{id:e,error:String(s)}),s}}async stop(e){let t=this.plugins.get(e);if(!t){this.log.warn("plugin not found",{id:e});return}if(!t.started){this.log.warn("plugin not started",{id:e});return}this.log.info("stopping plugin",{id:e});try{await t.plugin.stop(),t.started=!1,this.log.info("plugin stopped",{id:e});}catch(s){throw this.log.error("failed to stop plugin",{id:e,error:String(s)}),s}}async startAll(){let e=[];for(let[t,s]of this.plugins){if(!s.config.enabled){this.log.debug("skipping disabled plugin",{id:t});continue}try{await this.start(t);}catch(n){e.push({id:t,error:n instanceof Error?n:new Error(String(n))});}}e.length>0&&this.log.warn("some plugins failed to start",{count:e.length,errors:e.map(t=>({id:t.id,error:t.error.message}))});}async stopAll(){let e=[];for(let[t,s]of this.plugins)if(s.started)try{await this.stop(t);}catch(n){e.push({id:t,error:n instanceof Error?n:new Error(String(n))});}e.length>0&&this.log.warn("some plugins failed to stop",{count:e.length,errors:e.map(t=>({id:t.id,error:t.error.message}))});}getPlugin(e){return this.plugins.get(e)?.plugin}getAllPlugins(){return [...this.plugins.values()].map(e=>e.plugin)}getRunningPlugins(){return [...this.plugins.values()].filter(e=>e.started).map(e=>e.plugin)}isRunning(e){return this.plugins.get(e)?.started??false}async healthCheck(e){let t=this.plugins.get(e);if(!t)return {healthy:false,lastCheck:Date.now(),error:"Plugin not found"};try{let s=await Promise.race([t.plugin.healthCheck(),new Promise((n,r)=>setTimeout(()=>r(new Error("Health check timeout")),this.config.healthCheckTimeout))]);return t.lastHealthCheck=s,s}catch(s){let n={healthy:false,lastCheck:Date.now(),error:s instanceof Error?s.message:String(s)};return t.lastHealthCheck=n,n}}async healthCheckAll(){let e=new Map;for(let[t]of this.plugins)e.set(t,await this.healthCheck(t));return e}startHealthCheckTimer(){if(this.healthCheckTimer){this.log.warn("health check timer already running");return}this.healthCheckTimer=setInterval(async()=>{let e=await this.healthCheckAll();for(let[t,s]of e)s.healthy||this.log.warn("plugin health check failed",{id:t,error:s.error});},this.config.healthCheckInterval),this.log.info("health check timer started",{interval:this.config.healthCheckInterval});}stopHealthCheckTimer(){this.healthCheckTimer&&(clearInterval(this.healthCheckTimer),this.healthCheckTimer=null,this.log.info("health check timer stopped"));}async sendToPlugin(e,t){let s=this.plugins.get(e);if(!s)throw new Error(`Plugin not found: ${e}`);if(!s.started)throw new Error(`Plugin not started: ${e}`);return s.plugin.send(t)}};var o=chunk5ETPYYQR_cjs.a.create({service:"gateway:websocket-server"});function V(){return `ws_${Date.now()}_${Math.random().toString(36).substring(2,9)}`}function P(c,e){for(let[t,s]of e.entries())if(s===c)return t;return null}var w=class{constructor(e={}){chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"server",null);chunkGY3SWWW3_cjs.a(this,"wss",null);chunkGY3SWWW3_cjs.a(this,"httpsServer",null);chunkGY3SWWW3_cjs.a(this,"httpRedirectServer",null);chunkGY3SWWW3_cjs.a(this,"connections",new Map);chunkGY3SWWW3_cjs.a(this,"subscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"sessionSubscriptions",new Map);chunkGY3SWWW3_cjs.a(this,"sessionLastActivity",new Map);chunkGY3SWWW3_cjs.a(this,"cleanupInterval",null);chunkGY3SWWW3_cjs.a(this,"heartbeatInterval",null);chunkGY3SWWW3_cjs.a(this,"startTime",0);chunkGY3SWWW3_cjs.a(this,"running",false);this.config={...E,...e};}async start(){if(this.running){o.warn("WebSocket server is already running");return}o.debug("WebSocket server start: checking configuration",{port:this.config.port,hostname:this.config.hostname,path:this.config.path,https:this.config.https?.enabled??false}),this.startTime=Date.now();let e={onOpen:(n,r)=>{this.handleConnection(r);},onMessage:async(n,r)=>{try{let a=JSON.parse(n.data.toString());await this.handleMessage(r,a);}catch(a){o.error("Error processing WebSocket message",{error:a.message}),this.sendError(r,a.message);}},onClose:(n,r)=>{this.handleDisconnection(r);},onError:(n,r)=>{o.error("WebSocket error",{error:n.type});}},t=this.config.https?.enabled??false;o.debug("WebSocket server start: choosing server type",{https:t});try{t?(o.debug("WebSocket server start: starting HTTPS server"),await this.startHTTPSServer(e)):(o.debug("WebSocket server start: starting HTTP server"),await this.startHTTPServer(e)),o.debug("WebSocket server start: HTTP/HTTPS server started successfully");}catch(n){throw o.error("WebSocket server start: HTTP/HTTPS server failed",{error:n instanceof Error?n.message:String(n)}),n}o.debug("WebSocket server start: server initialization complete"),this.cleanupInterval=setInterval(()=>this.cleanupExpiredSessions(),3e4),this.heartbeatInterval=setInterval(()=>this.checkConnectionTimeout(),this.config.heartbeatInterval),this.running=true;let s=t?"wss":"ws";o.info("WebSocket server started",{url:`${s}://${this.config.hostname}:${this.config.port}${this.config.path}`});}async startHTTPServer(e){o.debug("HTTP server start: creating server",{port:this.config.port,hostname:this.config.hostname}),this.server=utils.createServer({port:this.config.port,hostname:this.config.hostname,routes:this.createRoutes(),timeout:this.config.connectionTimeout,cors:true}),o.debug("HTTP server start: server created, checking raw server");let t=this.server.raw;t?(o.debug("HTTP server start: raw server available, setting up WebSocket upgrade"),this.setupWebSocketUpgrade(t,e),o.debug("HTTP server start: WebSocket upgrade configured")):o.warn("HTTP server start: raw server is undefined");}async startHTTPSServer(e){let t=this.config.https,s=this.loadSSLCertificates(t);if(!s){o.error("Failed to load SSL certificates, falling back to HTTP"),await this.startHTTPServer(e);return}this.server=utils.createServer({port:this.config.port,hostname:this.config.hostname,routes:this.createRoutes(),timeout:this.config.connectionTimeout,cors:true});let n=this.server.raw;this.httpsServer=j__default.default.createServer(s,n.listeners("request")[0]),this.setupWebSocketUpgrade(this.httpsServer,e),await new Promise(r=>{this.httpsServer.listen(this.config.port,this.config.hostname,()=>{o.info("HTTPS server started",{port:this.config.port,hostname:this.config.hostname}),r();});}),t.forceRedirect&&t.httpPort&&this.startHTTPRedirectServer(t.httpPort);}loadSSLCertificates(e){try{return e.cert&&e.key?{cert:e.cert,key:e.key}:e.certPath&&e.keyPath?{cert:p__namespace.default.readFileSync(e.certPath),key:p__namespace.default.readFileSync(e.keyPath)}:null}catch(t){return o.error("Failed to load SSL certificates",{error:t.message}),null}}setupWebSocketUpgrade(e,t){this.wss=new ws.WebSocketServer({noServer:true}),e.on("upgrade",(s,n,r)=>{try{let g=new URL(s.url||"/",`http://${s.headers.host}`).pathname;(g===this.config.path||this.config.path==="/"&&(g==="/"||g===""))&&this.wss.handleUpgrade(s,n,r,l=>{t.onOpen?.({type:"open"},l),l.on("message",h=>{t.onMessage?.({data:h},l);}),l.on("close",(h,D)=>{t.onClose?.({code:h,reason:D},l);}),l.on("error",h=>{t.onError?.({type:"error",message:h.message},l);});});}catch{}});}startHTTPRedirectServer(e){this.httpRedirectServer=Q__default.default.createServer((t,s)=>{let r=`https://${t.headers.host?.split(":")[0]||this.config.hostname}:${this.config.port}${t.url}`;o.debug("Redirecting HTTP to HTTPS",{from:t.url,to:r}),s.writeHead(301,{Location:r,"Content-Type":"text/plain"}),s.end(`Redirecting to ${r}`);}),this.httpRedirectServer.listen(e,this.config.hostname,()=>{o.info("HTTP redirect server started",{port:e,hostname:this.config.hostname});});}async stop(){if(!this.running){o.warn("WebSocket server is not running");return}o.info("Stopping WebSocket server"),this.cleanupInterval&&(clearInterval(this.cleanupInterval),this.cleanupInterval=null),this.heartbeatInterval&&(clearInterval(this.heartbeatInterval),this.heartbeatInterval=null);for(let[e,t]of this.connections.entries())try{t.close(1e3,"Server shutting down");}catch(s){o.debug("Error closing connection",{clientId:e,error:String(s)});}this.connections.clear(),this.subscriptions.clear(),this.sessionSubscriptions.clear(),this.sessionLastActivity.clear(),this.wss&&(this.wss.close(),this.wss=null),this.httpsServer&&(await new Promise(e=>{this.httpsServer.close(()=>e());}),this.httpsServer=null),this.httpRedirectServer&&(await new Promise(e=>{this.httpRedirectServer.close(()=>e());}),this.httpRedirectServer=null),this.server&&(await this.server.stop(),this.server=null),this.running=false,o.info("WebSocket server stopped");}broadcastToSession(e,t){this.touchSession(e);let s=this.sessionSubscriptions.get(e);if(!s||s.size===0){o.debug("No clients subscribed to session",{sessionId:e});return}let n=t?.properties?.sessionID;if(typeof n=="string"&&n&&n!==e)return;let r=JSON.stringify({type:"event",sessionId:e,payload:t});for(let a of s){let g=this.subscriptions.get(a);if(g&&g.ws.readyState===g.ws.OPEN&&g.subscribedSessions.has(e)){if(g.subscribedEventTypes.size>0&&!g.subscribedEventTypes.has(t.type)){o.debug("Event type not subscribed, skipping",{clientId:a,sessionId:e,eventType:t.type,subscribedTypes:Array.from(g.subscribedEventTypes)});continue}g.ws.send(r),o.debug("Event pushed to client",{clientId:a,sessionId:e,eventType:t.type});}}}getConnectionCount(){return this.connections.size}getSubscriptionCount(){return this.sessionSubscriptions.size}getConnectionInfos(){let e=[];for(let[t,s]of this.subscriptions.entries())e.push({clientId:t,state:s.ws.readyState===s.ws.OPEN?"connected":"disconnected",connectedAt:s.connectedAt,lastActivityAt:s.lastActivityAt,authInfo:s.authInfo});return e}isRunning(){return this.running}handleConnection(e){if(this.connections.size>=(this.config.maxConnections??y.maxConnections)){o.warn("Maximum connections reached, rejecting new connection"),e.close(1013,"Maximum connections reached");return}let t=V();this.connections.set(t,e),this.subscriptions.set(t,{clientId:t,ws:e,subscribedSessions:new Set,subscribedEventTypes:new Set,connectedAt:Date.now(),lastActivityAt:Date.now()}),o.info("WebSocket client connected",{clientId:t,totalConnections:this.connections.size}),this.sendMessage(e,{type:"connected",clientId:t,payload:{version:"1.0.0",capabilities:["subscribe","message","interrupt","releaseSubscription"]}});}handleDisconnection(e){let t=P(e,this.connections);if(!t)return;this.connections.delete(t);let s=this.subscriptions.get(t);if(s){for(let n of s.subscribedSessions){let r=this.sessionSubscriptions.get(n);r&&(r.delete(t),r.size===0&&this.sessionSubscriptions.delete(n));}this.subscriptions.delete(t);}o.info("WebSocket client disconnected",{clientId:t,totalConnections:this.connections.size});}async handleMessage(e,t){let s=P(e,this.connections);if(!s){this.sendError(e,"Client not found");return}let n=this.subscriptions.get(s);switch(n&&(n.lastActivityAt=Date.now()),t.type){case "ping":this.sendMessage(e,{type:"pong"});break;case "initialize":await this.handleInitialize(e,t);break;case "subscribe":await this.handleSubscribe(e,t);break;case "unsubscribe":await this.handleUnsubscribe(e,t);break;case "message":await this.handleMessageRequest(e,t);break;case "interrupt":await this.handleInterrupt(e,t);break;case "releaseSubscription":await this.handleReleaseSubscription(e,t);break;default:this.sendError(e,`Unknown message type: ${t.type}`);}}async handleInitialize(e,t){this.sendMessage(e,{type:"initialize_response",success:true,result:{capabilities:["subscribe","message","interrupt","releaseSubscription"]}});}async handleSubscribe(e,t){if(t.type!=="subscribe"||!t.sessionId){this.sendError(e,"Invalid subscribe request");return}let s=P(e,this.connections);if(!s){this.sendError(e,"Client not found");return}let n=this.subscriptions.get(s);if(!n){this.sendError(e,"Subscription not found");return}if(n.subscribedSessions.add(t.sessionId),this.sessionSubscriptions.has(t.sessionId)||this.sessionSubscriptions.set(t.sessionId,new Set),this.sessionSubscriptions.get(t.sessionId).add(s),this.touchSession(t.sessionId),t.payload?.eventTypes)for(let r of t.payload.eventTypes)n.subscribedEventTypes.add(r);o.info("Client subscribed to session",{clientId:s,sessionId:t.sessionId}),this.sendMessage(e,{type:"subscribed",sessionId:t.sessionId});}async handleUnsubscribe(e,t){if(t.type!=="unsubscribe"||!t.sessionId){this.sendError(e,"Invalid unsubscribe request");return}let s=P(e,this.connections);if(!s)return;let n=this.subscriptions.get(s);if(!n)return;n.subscribedSessions.delete(t.sessionId);let r=this.sessionSubscriptions.get(t.sessionId);r&&(r.delete(s),r.size===0&&this.sessionSubscriptions.delete(t.sessionId)),o.info("Client unsubscribed from session",{clientId:s,sessionId:t.sessionId}),this.sendMessage(e,{type:"unsubscribed",sessionId:t.sessionId});}async handleMessageRequest(e,t){if(t.type!=="message"||!t.sessionId||!t.payload){this.sendError(e,"Invalid message request");return}o.debug("Message received",{sessionId:t.sessionId}),this.sendMessage(e,{type:"response",sessionId:t.sessionId,success:true,result:{received:true}});}async handleInterrupt(e,t){if(t.type!=="interrupt"||!t.sessionId){this.sendError(e,"Invalid interrupt request");return}o.info("Interrupt requested via WebSocket",{sessionId:t.sessionId}),this.sendMessage(e,{type:"interrupted",sessionId:t.sessionId,success:true});}async handleReleaseSubscription(e,t){if(t.type!=="releaseSubscription"||!t.sessionId){this.sendError(e,"Invalid releaseSubscription request");return}o.info("Release subscription requested via WebSocket",{sessionId:t.sessionId}),this.releaseSessionResources(t.sessionId),this.sendMessage(e,{type:"subscriptionReleased",sessionId:t.sessionId});}touchSession(e){this.sessionLastActivity.set(e,Date.now());}checkConnectionTimeout(){let e=Date.now(),t=this.config.connectionTimeout??y.connectionTimeout,s=[];for(let[n,r]of this.subscriptions.entries())e-r.lastActivityAt>t&&s.push(n);for(let n of s){let r=this.subscriptions.get(n);if(r){o.info("Closing inactive connection",{clientId:n,lastActivityAt:r.lastActivityAt,timeoutMs:t});try{r.ws.close(1001,"Connection timeout");}catch(a){o.debug("Error closing timed out connection",{clientId:n,error:String(a)});}}}s.length>0&&o.debug("Closed timed out connections",{count:s.length,remainingConnections:this.connections.size-s.length});}cleanupExpiredSessions(){let e=Date.now(),t=[],s=this.config.sessionExpireMs??y.sessionExpireMs;for(let[n,r]of this.sessionLastActivity.entries())e-r>s&&t.push(n);for(let n of t)o.info("Session expired due to inactivity, releasing resources",{sessionId:n,lastActivity:this.sessionLastActivity.get(n),expireMs:s}),this.releaseSessionResources(n),this.sessionLastActivity.delete(n);t.length>0&&o.debug("Cleaned up expired sessions",{count:t.length,remainingActive:this.sessionLastActivity.size});}releaseSessionResources(e){let t=this.sessionSubscriptions.get(e);if(t){for(let s of t){let n=this.subscriptions.get(s);n&&n.ws.readyState===n.ws.OPEN&&this.sendMessage(n.ws,{type:"session_closed",sessionId:e,payload:{reason:"Session resources released due to expiration or explicit release"}});}for(let s of t){let n=this.subscriptions.get(s);n&&n.subscribedSessions.delete(e);}this.sessionSubscriptions.delete(e),o.info("Session resources released",{sessionId:e,clientCount:t.size});}}sendMessage(e,t){e.readyState===e.OPEN&&e.send(JSON.stringify(t));}sendError(e,t,s){this.sendMessage(e,{type:"error",message:t,code:s});}createRoutes(){return [{method:"get",path:"/health",handler:e=>{let t={status:"healthy",timestamp:Date.now(),version:"1.0.0"};return e.json(t)}},{method:"get",path:"/status",handler:e=>{let t={status:this.running?"running":"stopped",connections:this.connections.size,sessions:this.sessionSubscriptions.size,subscriptions:this.subscriptions.size,uptime:this.running?Date.now()-this.startTime:0,version:"1.0.0"};return e.json(t)}},{method:"post",path:"/initialize",handler:async e=>{let t=await e.req.json().catch(()=>({}));return o.debug("ACP initialize request received",{body:t}),e.json({success:true,result:{capabilities:["subscribe","message","interrupt","releaseSubscription"],version:"1.0.0"}})}},{method:"post",path:"/session",handler:async e=>{await e.req.json().catch(()=>({}));let n={id:`session_${Date.now()}_${Math.random().toString(36).substring(2,9)}`,createdAt:Date.now()};return e.json(n)}},{method:"post",path:"/session/:sessionId/prompt",handler:async e=>{let t=e.req.param("sessionId");await e.req.json().catch(()=>({}));if(!t)return e.json({error:"No sessionId provided"},400);let r={messageId:`msg_${Date.now()}_${Math.random().toString(36).substring(2,9)}`,sessionId:t,createdAt:Date.now()};return e.status(202),e.json(r)}},{method:"get",path:"/subscriptions",handler:e=>{let t=Array.from(this.sessionSubscriptions.entries()).map(([n,r])=>({sessionId:n,subscriberCount:r.size,subscribers:Array.from(r)})),s={subscriptions:t,total:t.length};return e.json(s)}}]}};function K(c){let e={platform:c.channel?.platform??"api",channelId:c.channel?.channelId??"unknown",chatId:c.channel?.chatId,userId:c.channel?.userId};return {id:c.messageId??`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:c.sessionId,type:c.type,content:Array.isArray(c.content)?c.content:[{type:"text",text:String(c.content??"")}],metadata:{channel:e,replyToMessageId:c.replyToMessageId},timestamp:c.timestamp??Date.now()}}var O=class{constructor(e){chunkGY3SWWW3_cjs.a(this,"log",chunk5ETPYYQR_cjs.a.create({service:"gateway:server"}));chunkGY3SWWW3_cjs.a(this,"config");chunkGY3SWWW3_cjs.a(this,"router");chunkGY3SWWW3_cjs.a(this,"sessionManager");chunkGY3SWWW3_cjs.a(this,"pluginLoader");chunkGY3SWWW3_cjs.a(this,"messageStore");chunkGY3SWWW3_cjs.a(this,"wsServer",null);chunkGY3SWWW3_cjs.a(this,"running",false);chunkGY3SWWW3_cjs.a(this,"connections",0);this.config=e,this.router=new x,this.sessionManager=new v,this.pluginLoader=new C({healthCheckInterval:6e4});let t=chunkEK6MKXZH_cjs.S(),s=chunkEK6MKXZH_cjs.P();this.messageStore=new S({storagePath:`${t.Path.data}/gateway-messages`}),s&&chunkEK6MKXZH_cjs.Q().setMessageStorage({store:async r=>{let a=K(r);await this.messageStore.store(a);},updateStatus:async(r,a,g)=>{await this.messageStore.updateStatus(r,a,g);}}),this.pluginLoader.setMessageHandler({onMessage:n=>this.onMessage(n),onEvent:n=>this.onEvent(n)});}async start(){if(this.running){this.log.warn("gateway server is already running");return}this.log.info("starting gateway server",{port:this.config.port,hostname:this.config.hostname}),await this.sessionManager.initialize(),this.log.debug("starting WebSocket server"),this.wsServer=new w({port:this.config.port,hostname:this.config.hostname,path:String(this.config.path??"/")}),await this.wsServer.start(),this.log.debug("WebSocket server started"),await this.pluginLoader.startAll(),this.pluginLoader.startHealthCheckTimer(),this.running=true,this.log.info("gateway server started");}async stop(){if(!this.running){this.log.warn("gateway server is not running");return}this.log.info("stopping gateway server"),this.wsServer&&(this.log.debug("stopping WebSocket server"),await this.wsServer.stop(),this.wsServer=null,this.log.debug("WebSocket server stopped")),this.pluginLoader.stopHealthCheckTimer(),await this.pluginLoader.stopAll(),await this.router.shutdown(),await this.sessionManager.close(),this.running=false,this.log.info("gateway server stopped");}getStatus(){return {running:this.running,port:this.config.port,hostname:this.config.hostname,connections:this.connections}}async onMessage(e){if(this.log.debug("message received",{id:e.id,sessionId:e.sessionId,type:e.type}),this.messageStore.isProcessed(e.id)){this.log.warn("message already processed, skipping",{id:e.id});return}await this.storeMessage(e),await this.sessionManager.incrementMessageCountOrCreate(e.sessionId,e.metadata.channel),(await this.router.route(e)).broadcastCount===0&&e.type==="input"&&this.log.debug("no subscribers, message pending",{id:e.id});}async storeMessage(e){if(await this.messageStore.store(e),chunkEK6MKXZH_cjs.P())try{let t=chunkEK6MKXZH_cjs.Q(),s={name:plugin.HookEvent.GatewayMessageReceive,data:{messageId:e.id,sessionId:e.sessionId,type:e.type,content:e.content,channel:e.metadata.channel,replyToMessageId:e.metadata.replyToMessageId,timestamp:e.timestamp||Date.now()}};await t.hookRegistry.emit(s);}catch{this.log.debug("hook trigger skipped (no context)");}}async processMessage(e,t,s,n={}){if(await this.messageStore.updateStatus(t,"processing",{processedBy:s}),chunkEK6MKXZH_cjs.P())try{await chunkEK6MKXZH_cjs.Q().hookRegistry.emit({name:plugin.HookEvent.GatewayMessageProcess,data:{messageId:t,sessionId:e,processorId:s,processType:n.processType||"direct",agent:n.agent,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}}async completeMessage(e,t,s,n={}){let{success:r=true,error:a,responseMessageId:g,tokens:l,duration:h}=n;if(await this.messageStore.updateStatus(t,r?"completed":"failed",{processedBy:s,error:a}),chunkEK6MKXZH_cjs.P())try{await chunkEK6MKXZH_cjs.Q().hookRegistry.emit({name:plugin.HookEvent.GatewayMessageComplete,data:{messageId:t,sessionId:e,processorId:s,status:r?"completed":"failed",error:a,responseMessageId:g,tokens:l,duration:h,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}await this.router.completeMessage(e,t,s,r?"completed":"failed");}async onEvent(e){switch(this.log.debug("channel event",{type:e.type,channelId:e.channelId}),e.type){case "connected":this.connections++;break;case "disconnected":this.connections--;break;case "error":this.log.error("channel error",{channelId:e.channelId,data:e.data});break}}async registerChannel(e,t){let s=t??{enabled:true,platform:{},session:{autoCreate:true,resetPolicy:"onNewConversation"},routing:{broadcast:true,dispatchToSubAgent:false}};await this.pluginLoader.register(e,s),this.running&&await this.pluginLoader.start(e.id);}async unregisterChannel(e){await this.pluginLoader.unregister(e);}async subscribe(e,t){await this.router.subscribe(e,t),await this.sessionManager.addSubscriber(e,t.id);}async unsubscribe(e){await this.router.unsubscribe(e);}async send(e){await this.onMessage(e);}async sendToChannel(e,t,s={}){let{proactive:n=false,replyToMessageId:r}=s;if(chunkEK6MKXZH_cjs.P())try{await chunkEK6MKXZH_cjs.Q().hookRegistry.emit({name:plugin.HookEvent.GatewayMessageSend,data:{messageId:t.id,sessionId:t.sessionId,type:t.type,channel:t.metadata.channel,proactive:n,replyToMessageId:r,timestamp:Date.now()}});}catch{this.log.debug("hook trigger skipped (no context)");}return this.pluginLoader.sendToPlugin(e,t)}async sendProactiveMessage(e,t,s,n){let r={id:`msg_${Date.now()}_${Math.random().toString(36).slice(2,9)}`,sessionId:t,type:"output",content:s,metadata:{channel:n},timestamp:Date.now()};return this.sendToChannel(e,r,{proactive:true})}getSessionManager(){return this.sessionManager}getRouter(){return this.router}getPluginLoader(){return this.pluginLoader}getMessageStore(){return this.messageStore}async getMessageHistory(e,t){return this.messageStore.getSessionHistory(e,t)}};
2
2
  exports.a=M;exports.b=I;exports.c=H;exports.d=N;exports.e=T;exports.f=R;exports.g=k;exports.h=E;exports.i=B;exports.j=q;exports.k=F;exports.l=U;exports.m=x;exports.n=S;exports.o=b;exports.p=v;exports.q=C;exports.r=w;exports.s=O;
@@ -1 +1 @@
1
- 'use strict';var chunkY7UBDKER_cjs=require('./chunk-Y7UBDKER.cjs');require('./chunk-GY3SWWW3.cjs');var promises=require('fs/promises'),o=require('path'),xdgBasedir=require('xdg-basedir'),w=require('os');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var o__default=/*#__PURE__*/_interopDefault(o);var w__default=/*#__PURE__*/_interopDefault(w);var s="easbot",c=o__default.default.join(xdgBasedir.xdgData,s),x=o__default.default.join(xdgBasedir.xdgCache,s),H=o__default.default.join(xdgBasedir.xdgConfig,s),N=o__default.default.join(xdgBasedir.xdgState,s);exports.Global=void 0;(e=>{e.Path={get home(){return process.env.EASBOT_TEST_HOME||w__default.default.homedir()},data:c,bin:o__default.default.join(c,"bin"),log:o__default.default.join(c,"log"),cache:x,config:H,state:N};let t=null,n=null;async function P(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{let r=await import('./package-ICSXKMHH.cjs');t=r.default?.version||r.version||"local";}catch{t="local";}return t||"local"}function A(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{t=chunkY7UBDKER_cjs.a.version||"local";}catch{t="local";}return t||"local"}e.getVersion=A;function l(){if(n)return n;if(process.env.EASBOT_CHANNEL)return n=process.env.EASBOT_CHANNEL,n;let r=A();return r==="local"||r.includes("dev")?n="local":r.includes("beta")||r.includes("alpha")?n="preview":n="latest",n||"local"}e.getChannel=l;function y(){return l()==="local"}e.isLocal=y;function C(){return l()==="preview"}e.isPreview=C;let u=false,a=null;async function V(){if(!u)return a||(a=(async()=>{await P(),await Promise.all([promises.mkdir(e.Path.data,{recursive:true}),promises.mkdir(e.Path.config,{recursive:true}),promises.mkdir(e.Path.state,{recursive:true}),promises.mkdir(e.Path.log,{recursive:true}),promises.mkdir(e.Path.bin,{recursive:true}),promises.mkdir(e.Path.cache,{recursive:true})]).then(()=>{process.env.EASBOT_DATA_PATH=e.Path.data,process.env.EASBOT_CONFIG_PATH=e.Path.config,process.env.EASBOT_STATE_PATH=e.Path.state,process.env.EASBOT_LOG_PATH=e.Path.log,process.env.EASBOT_BIN_PATH=e.Path.bin,process.env.EASBOT_CACHE_PATH=e.Path.cache;});let r="1",h=o__default.default.join(e.Path.cache,"version");if(await promises.readFile(h,"utf-8").catch(()=>"0")!==r){try{let p=await promises.readdir(e.Path.cache);await Promise.all(p.map(E=>promises.rm(o__default.default.join(e.Path.cache,E),{recursive:!0,force:!0})));}catch{}await promises.writeFile(h,r);}u=true;})(),a)}e.init=V;})(exports.Global||(exports.Global={}));
1
+ 'use strict';var chunkPBDA3HG4_cjs=require('./chunk-PBDA3HG4.cjs');require('./chunk-GY3SWWW3.cjs');var promises=require('fs/promises'),o=require('path'),xdgBasedir=require('xdg-basedir'),w=require('os');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var o__default=/*#__PURE__*/_interopDefault(o);var w__default=/*#__PURE__*/_interopDefault(w);var s="easbot",c=o__default.default.join(xdgBasedir.xdgData,s),x=o__default.default.join(xdgBasedir.xdgCache,s),H=o__default.default.join(xdgBasedir.xdgConfig,s),N=o__default.default.join(xdgBasedir.xdgState,s);exports.Global=void 0;(e=>{e.Path={get home(){return process.env.EASBOT_TEST_HOME||w__default.default.homedir()},data:c,bin:o__default.default.join(c,"bin"),log:o__default.default.join(c,"log"),cache:x,config:H,state:N};let t=null,n=null;async function P(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{let r=await import('./package-RO6UIPH6.cjs');t=r.default?.version||r.version||"local";}catch{t="local";}return t||"local"}function A(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{t=chunkPBDA3HG4_cjs.a.version||"local";}catch{t="local";}return t||"local"}e.getVersion=A;function l(){if(n)return n;if(process.env.EASBOT_CHANNEL)return n=process.env.EASBOT_CHANNEL,n;let r=A();return r==="local"||r.includes("dev")?n="local":r.includes("beta")||r.includes("alpha")?n="preview":n="latest",n||"local"}e.getChannel=l;function y(){return l()==="local"}e.isLocal=y;function C(){return l()==="preview"}e.isPreview=C;let u=false,a=null;async function V(){if(!u)return a||(a=(async()=>{await P(),await Promise.all([promises.mkdir(e.Path.data,{recursive:true}),promises.mkdir(e.Path.config,{recursive:true}),promises.mkdir(e.Path.state,{recursive:true}),promises.mkdir(e.Path.log,{recursive:true}),promises.mkdir(e.Path.bin,{recursive:true}),promises.mkdir(e.Path.cache,{recursive:true})]).then(()=>{process.env.EASBOT_DATA_PATH=e.Path.data,process.env.EASBOT_CONFIG_PATH=e.Path.config,process.env.EASBOT_STATE_PATH=e.Path.state,process.env.EASBOT_LOG_PATH=e.Path.log,process.env.EASBOT_BIN_PATH=e.Path.bin,process.env.EASBOT_CACHE_PATH=e.Path.cache;});let r="1",h=o__default.default.join(e.Path.cache,"version");if(await promises.readFile(h,"utf-8").catch(()=>"0")!==r){try{let p=await promises.readdir(e.Path.cache);await Promise.all(p.map(E=>promises.rm(o__default.default.join(e.Path.cache,E),{recursive:!0,force:!0})));}catch{}await promises.writeFile(h,r);}u=true;})(),a)}e.init=V;})(exports.Global||(exports.Global={}));
@@ -1 +1 @@
1
- import {a}from'./chunk-3UUTBBIX.mjs';import'./chunk-HAMGVOQD.mjs';import {mkdir,readFile,readdir,rm,writeFile}from'fs/promises';import o from'path';import {xdgData,xdgCache,xdgConfig,xdgState}from'xdg-basedir';import w from'os';var s="easbot",c=o.join(xdgData,s),x=o.join(xdgCache,s),H=o.join(xdgConfig,s),N=o.join(xdgState,s),f;(e=>{e.Path={get home(){return process.env.EASBOT_TEST_HOME||w.homedir()},data:c,bin:o.join(c,"bin"),log:o.join(c,"log"),cache:x,config:H,state:N};let t=null,n=null;async function P(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{let r=await import('./package-DLCORTTP.mjs');t=r.default?.version||r.version||"local";}catch{t="local";}return t||"local"}function A(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{t=a.version||"local";}catch{t="local";}return t||"local"}e.getVersion=A;function l(){if(n)return n;if(process.env.EASBOT_CHANNEL)return n=process.env.EASBOT_CHANNEL,n;let r=A();return r==="local"||r.includes("dev")?n="local":r.includes("beta")||r.includes("alpha")?n="preview":n="latest",n||"local"}e.getChannel=l;function y(){return l()==="local"}e.isLocal=y;function C(){return l()==="preview"}e.isPreview=C;let u=false,a$1=null;async function V(){if(!u)return a$1||(a$1=(async()=>{await P(),await Promise.all([mkdir(e.Path.data,{recursive:true}),mkdir(e.Path.config,{recursive:true}),mkdir(e.Path.state,{recursive:true}),mkdir(e.Path.log,{recursive:true}),mkdir(e.Path.bin,{recursive:true}),mkdir(e.Path.cache,{recursive:true})]).then(()=>{process.env.EASBOT_DATA_PATH=e.Path.data,process.env.EASBOT_CONFIG_PATH=e.Path.config,process.env.EASBOT_STATE_PATH=e.Path.state,process.env.EASBOT_LOG_PATH=e.Path.log,process.env.EASBOT_BIN_PATH=e.Path.bin,process.env.EASBOT_CACHE_PATH=e.Path.cache;});let r="1",h=o.join(e.Path.cache,"version");if(await readFile(h,"utf-8").catch(()=>"0")!==r){try{let p=await readdir(e.Path.cache);await Promise.all(p.map(E=>rm(o.join(e.Path.cache,E),{recursive:!0,force:!0})));}catch{}await writeFile(h,r);}u=true;})(),a$1)}e.init=V;})(f||(f={}));export{f as Global};
1
+ import {a}from'./chunk-D2ZQ47GV.mjs';import'./chunk-HAMGVOQD.mjs';import {mkdir,readFile,readdir,rm,writeFile}from'fs/promises';import o from'path';import {xdgData,xdgCache,xdgConfig,xdgState}from'xdg-basedir';import w from'os';var s="easbot",c=o.join(xdgData,s),x=o.join(xdgCache,s),H=o.join(xdgConfig,s),N=o.join(xdgState,s),f;(e=>{e.Path={get home(){return process.env.EASBOT_TEST_HOME||w.homedir()},data:c,bin:o.join(c,"bin"),log:o.join(c,"log"),cache:x,config:H,state:N};let t=null,n=null;async function P(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{let r=await import('./package-JLVEIYME.mjs');t=r.default?.version||r.version||"local";}catch{t="local";}return t||"local"}function A(){if(t)return t;if(process.env.EASBOT_VERSION)return t=process.env.EASBOT_VERSION,t;try{t=a.version||"local";}catch{t="local";}return t||"local"}e.getVersion=A;function l(){if(n)return n;if(process.env.EASBOT_CHANNEL)return n=process.env.EASBOT_CHANNEL,n;let r=A();return r==="local"||r.includes("dev")?n="local":r.includes("beta")||r.includes("alpha")?n="preview":n="latest",n||"local"}e.getChannel=l;function y(){return l()==="local"}e.isLocal=y;function C(){return l()==="preview"}e.isPreview=C;let u=false,a$1=null;async function V(){if(!u)return a$1||(a$1=(async()=>{await P(),await Promise.all([mkdir(e.Path.data,{recursive:true}),mkdir(e.Path.config,{recursive:true}),mkdir(e.Path.state,{recursive:true}),mkdir(e.Path.log,{recursive:true}),mkdir(e.Path.bin,{recursive:true}),mkdir(e.Path.cache,{recursive:true})]).then(()=>{process.env.EASBOT_DATA_PATH=e.Path.data,process.env.EASBOT_CONFIG_PATH=e.Path.config,process.env.EASBOT_STATE_PATH=e.Path.state,process.env.EASBOT_LOG_PATH=e.Path.log,process.env.EASBOT_BIN_PATH=e.Path.bin,process.env.EASBOT_CACHE_PATH=e.Path.cache;});let r="1",h=o.join(e.Path.cache,"version");if(await readFile(h,"utf-8").catch(()=>"0")!==r){try{let p=await readdir(e.Path.cache);await Promise.all(p.map(E=>rm(o.join(e.Path.cache,E),{recursive:!0,force:!0})));}catch{}await writeFile(h,r);}u=true;})(),a$1)}e.init=V;})(f||(f={}));export{f as Global};
@@ -0,0 +1 @@
1
+ export{a as default}from'./chunk-D2ZQ47GV.mjs';import'./chunk-HAMGVOQD.mjs';
@@ -0,0 +1 @@
1
+ 'use strict';var chunkPBDA3HG4_cjs=require('./chunk-PBDA3HG4.cjs');require('./chunk-GY3SWWW3.cjs');module.exports=chunkPBDA3HG4_cjs.a;