@kelceyp/caw-server 1.0.220 → 1.0.221

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.
Files changed (2) hide show
  1. package/dist/main.js +1 -1
  2. package/package.json +1 -1
package/dist/main.js CHANGED
@@ -1245,4 +1245,4 @@ Errors:
1245
1245
  `;for(let Z of W.slice(0,10))i+=` - ${Z.entityId}: ${Z.error}
1246
1246
  `;if(W.length>10)i+=` ... and ${W.length-10} more errors
1247
1247
  `}return{content:[{type:"text",text:i}]}}catch(Y){return B2(Y)}},BK8={name:"backfill_resolved_fts",title:"Backfill Resolved FTS",description:"Re-resolve FTS content for published documents/templates with <include> directives. If confirm=false, returns preview of affected entities. If confirm=true, resolves includes and updates FTS content.",schema:ia6,execute:Za6};var UK8=[yK8,BK8];var Oa6=C.object({store:C.string().optional().describe('Store code (default: "A")'),limit:C.coerce.number().optional().describe("Maximum number of events to return (default: 50)"),offset:C.coerce.number().optional().describe("Pagination offset (default: 0)")}),pa6=async({core:_,projectId:v},{store:u,limit:K,offset:q})=>{try{let Y=await _.listDeletedEvents(v,{store:u??"A",limit:K??50,offset:q??0});return{content:[{type:"text",text:JSON.stringify(Y,null,2)}]}}catch(Y){return B2(Y)}},FK8={name:"list_deleted_events",title:"List Deletion Events",description:"List deletion events. Each event represents a single delete action (e.g. deleting a folder and all its contents), identified by a root entity ID. Returns events with root entity ID, root entity info, item count, deletedAt timestamp (for display), and days until purge.",schema:Oa6,execute:pa6};var Ha6=C.object({root_entity_id:C.string().describe('Root entity ID of the deletion event (e.g. "Ad131")'),store:C.string().optional().describe('Store code (default: "A")'),limit:C.coerce.number().optional().describe("Maximum number of items to return (default: 50)"),offset:C.coerce.number().optional().describe("Pagination offset (default: 0)")}),Xa6=async({core:_,projectId:v},{root_entity_id:u,store:K,limit:q,offset:Y})=>{try{let c=await _.getDeletedEvent(v,{rootEntityId:u,store:K??"A",limit:q??50,offset:Y??0});return{content:[{type:"text",text:JSON.stringify(c,null,2)}]}}catch(c){return B2(c)}},oK8={name:"get_deleted_event",title:"Get Deletion Event",description:"Get the items in a specific deletion event identified by its root entity ID. Returns root entity info and a paginated list of all entities deleted in that event.",schema:Ha6,execute:Xa6};var Ga6=C.object({root_entity_id:C.string().describe('Root entity ID of the deletion event (e.g. "Ad131")'),store:C.string().optional().describe('Store code (default: "A")')}),fa6=async({core:_,projectId:v},{root_entity_id:u,store:K})=>{try{let q=await _.restoreDeletedEvent(v,{rootEntityId:u,store:K??"A"});return{content:[{type:"text",text:JSON.stringify(q,null,2)}]}}catch(q){return B2(q)}},QK8={name:"restore_deleted_event",title:"Restore Deletion Event",description:'Restore all entities in a deletion event (identified by its root entity ID). All entities deleted in the same action are restored together. If the root entity has a name collision, it is renamed with a "-restored" suffix.',schema:Ga6,execute:fa6};var dK8=[FK8,oK8,QK8];var Aa6=C.object({}),Ta6=async({core:_,projectId:v})=>{try{let u=await _.getSystemSettings(v);return{content:[{type:"text",text:JSON.stringify(u,null,2)}]}}catch(u){return B2(u)}},rK8={name:"get_system_settings",title:"Get System Settings",description:"Returns all system settings for the project (autoArchiveDelayDays, maxConcurrentJobs, trashRetentionDays, maxResumeAttempts, defaultResumePromptRef, scratchDocDelayDays, jobDelayDays, indexingMode, batchIntervalMinutes, claudeStrategy)",schema:Aa6,execute:Ta6};var Pa6=C.object({settings:VK(C.object({autoArchiveDelayDays:C.number().int().optional(),maxConcurrentJobs:C.number().int().optional(),maxJobDepth:C.number().int().optional(),trashRetentionDays:C.number().int().optional(),maxResumeAttempts:C.number().int().optional(),defaultResumePromptRef:C.string().nullable().optional(),scratchDocDelayDays:C.number().int().optional(),jobDelayDays:C.number().int().optional(),indexingMode:C.enum(["batch","constant"]).optional(),batchIntervalMinutes:C.number().int().positive().optional(),claudeStrategy:C.enum(["pty","headless"]).optional()})).describe("Partial settings object. Only provided keys are updated.")}),ja6=async({core:_,projectId:v},{settings:u})=>{try{let K=await _.updateSystemSettings(v,{settings:u});return{content:[{type:"text",text:JSON.stringify(K,null,2)}]}}catch(K){return B2(K)}},sK8={name:"update_system_settings",title:"Update System Settings",description:'Update system settings for the project. Accepts partial updates — only provided keys are changed. Settings: autoArchiveDelayDays, maxConcurrentJobs, trashRetentionDays, maxResumeAttempts, defaultResumePromptRef, scratchDocDelayDays, jobDelayDays, indexingMode, batchIntervalMinutes, claudeStrategy. Validates values: integer fields must be positive integers; defaultResumePromptRef must be a template ID string or null (to clear); indexingMode must be "batch" or "constant"; batchIntervalMinutes must be a positive integer; claudeStrategy must be "pty" or "headless".',schema:Pa6,execute:ja6};var na6=C.object({}),Va6=async({core:_})=>{try{let v=await _.getIntegrations();return{content:[{type:"text",text:JSON.stringify(v,null,2)}]}}catch(v){return B2(v)}},tK8={name:"get_integrations",title:"Get Integrations",description:"Returns detection state for all integrations (name, displayName, available, path, version, reason, action, category, detectedAt). No parameters required.",schema:na6,execute:Va6};var eK8=[rK8,sK8,tK8];var _q8=O8.create({namespace:"caw:mcp"}),Ma6=({core:_,projectId:v,buildVersion:u})=>{let K=new r$2({name:"caw-server",version:u||"0.0.0-dev"},{capabilities:{tools:{}}}),q=[...Vu8,...Nu8,...Su8,...tu8,...uK8,...pK8,...TK8,...jK8,...SK8,...lK8,...UK8,...dK8,...eK8];for(let Y of q)K.registerTool(Y.name,{title:Y.title,description:Y.description,inputSchema:Y.schema},async(c,z)=>{try{let w=z?.sessionId||z?.requestInfo?.headers?.["mcp-session-id"];return await Y.execute({core:_,projectId:v,sessionId:w,extra:z},c)}catch(w){if(w.isDomainError)_q8.debug("Tool %s domain error: %o",Y.name,w);else _q8.error(`Tool ${Y.name} failed:`,w);return B2(w)}});return K},$q8=Object.freeze({createMcpServer:Ma6});var Mg=O8.create({namespace:"caw:mcp"}),Y7=O8.create({namespace:"caw:mcp:session"}),ka6=({core:_,buildVersion:v})=>{let u=new Map;return Object.freeze({handleRequest:async(z,w,{projectId:J})=>{let W=z.headers["mcp-session-id"];if(W){let H=u.get(W);if(!H)return Y7.debug("Session not found: %s",W),w.status(404).json({jsonrpc:"2.0",error:{code:-32001,message:"Session not found. Server may have been restarted. If so, ask user to reconnect"},id:null});if(H.projectId!==J)return Y7.debug("Session %s belongs to project %s, not %s",W,H.projectId,J),w.status(404).json({jsonrpc:"2.0",error:{code:-32001,message:"Session not found. Server may have been restarted. If so, ask user to reconnect"},id:null});Y7.debug("Routing request to existing session: %s",W),H.lastActivity=Date.now();try{return await H.transport.handleRequest(z,w,z.body)}catch(X){Mg.error("MCP transport request failed:",X)}}Y7.debug("Creating new MCP session for project: %s",J);let i=null,Z=new Q_2({sessionIdGenerator:()=>{return i=xa6(),Y7.debug("Generated session ID: %s",i),i},onsessioninitialized:(H)=>{Y7.debug("Session initialized: %s",H),w.setHeader("Mcp-Session-Id",H),p.sessionId=H,u.set(H,p),Y7.debug("Session stored: %s",H)}});Y7.debug("Creating new server for session");let O=$q8.createMcpServer({core:_,projectId:J,buildVersion:v});try{await O.connect(Z)}catch(H){return Mg.error("MCP server connect failed:",H),Z.close?.(),O.close?.(),w.status(500).json({jsonrpc:"2.0",error:{code:-32603,message:"Internal server error"},id:null})}let p={sessionId:null,transport:Z,server:O,projectId:J,createdAt:Date.now(),lastActivity:Date.now()};w.on("finish",()=>{if(w.statusCode!==200&&i&&u.has(i))Y7.debug("Request failed (status %d), cleaning up session: %s",w.statusCode,i),u.delete(i),Z.close?.(),O.close?.()});try{await Z.handleRequest(z,w,z.body)}catch(H){Mg.error("MCP transport request failed:",H)}},getSession:(z)=>{return u.get(z)},closeSession:(z)=>{let w=u.get(z);if(!w){Y7.debug("Cannot close session - not found: %s",z);return}Y7.debug("Closing session: %s",z),w.transport.close?.(),w.server.close?.(),u.delete(z)},cleanupIdle:(z)=>{let w=Date.now(),J=0;for(let[W,i]of u.entries())if(w-i.lastActivity>z)Y7.debug("Cleaning up idle session: %s (idle for %dms)",W,w-i.lastActivity),i.transport.close?.(),i.server.close?.(),u.delete(W),J++;if(J>0)Mg.debug("Cleaned up %d idle sessions",J)}})},vq8=Object.freeze({create:ka6});var Na6=86400000,Da6=300000,ma6=({mcpSessionManager:_})=>{let{jobFns:v,controlFns:u}=P$.create({name:"cleanup-idle-mcp-sessions",interval:Da6,execute:async()=>{_.cleanupIdle(Na6)}});return{job:Object.freeze(v),control:u}},uq8=Object.freeze({create:ma6});var lg=O8.create({namespace:"caw:server"}),Hb6=async({port:_,envDir:v,databaseFilepath:u,assetsDir:K,lazilyCreateDb:q,enableRest:Y,enableSse:c,enableStatic:z,enableCrons:w,enableMcp:J=!0,enableCli:W=!0,staticPath:i,kyselyFactory:Z,killProcess:O,platform:p,processEnv:H,buildVersion:X,cliMcpUrl:G})=>{let T=await Ke2.create({envDir:v,databaseFilepath:u,assetsDir:K,lazilyCreateDb:q,kyselyFactory:Z,killProcess:O,platform:p,processEnv:H,cawPort:_}),f=J68.create(),j=Ov2.default();j.use(Ov2.default.json({limit:"5mb"}));let P=new Set(["127.0.0.1","::1","::ffff:127.0.0.1"]);if(j.get("/health",(n,N)=>{let h={status:"ok"};if(P.has(n.socket.remoteAddress))Object.assign(h,T.getSafeToStopStatus());N.json(h)}),j.get("/api/info",(n,N)=>{N.json({version:X,hookFailureCount:AD2()})}),Y)j.use("/api",c68),j.use("/api",z68),ce2.create(j,{core:T}),we2.create(j,{core:T}),Je2.create(j,{core:T}),We2.create(j,{core:T}),Ze2.create(j,{core:T}),pe2.create(j,{core:T}),He2.create(j,{core:T}),Xe2.create(j,{core:T}),Ge2.create(j,{core:T}),fe2.create(j,{core:T}),Q88.create(j,{core:T}),d88.create(j,{core:T}),r88.create(j,{core:T}),s88.create(j,{core:T}),t88.create(j,{core:T}),e88.create(j,{core:T}),_68.create(j,{core:T}),v68.create(j,{core:T}),K68.create(j,{core:T});if(c)q68.create(j,{core:T}),Y68.create(j,{core:T});if(J){lg.debug("Initializing MCP subsystem...");let n=vq8.create({core:T,buildVersion:X});T.registerCronJob(uq8,{mcpSessionManager:n}),i68.create(j,{core:T,mcpSessionManager:n}),lg.debug("MCP subsystem initialized")}if(W){lg.debug("Initializing CLI Adapter subsystem...");let n=(await Promise.resolve().then(() => (Dq8(),Nq8))).default,N=(await Promise.resolve().then(() => (hq8(),Cq8))).default,h=(await Promise.resolve().then(() => (aq8(),Sq8))).default,I=(await Promise.resolve().then(() => (Iq8(),bq8))).default,S=h.create({mcpClientFactory:n.create({serverUrl:G}),aliasGenerator:N.create(),core:T});await S.init(),T.subscribeServerEvents({onProjectCreated:(d)=>S.addProject({objectId:d.id,name:d.name,path:d.path})}),I.create(j,{cliSessionManager:S}),lg.debug("CLI Adapter subsystem initialized")}if(z&&i)Ye2.create(j,{staticPath:i});if(w)T.startAllCrons();j.use((n,N,h,I)=>{h.status(n.status||n.statusCode||500).json({error:n.message,type:n.type})});let D=null,V=new Set;return Object.freeze({app:j,core:T,start:async()=>{return new Promise((n,N)=>{D=j.listen(_,(h)=>{if(h){N(h);return}let I=D.address();if(!I){N(Error("Server failed to bind"));return}f.start(),n(I.port)}),D.on("connection",(h)=>{V.add(h),h.on("close",()=>{V.delete(h)})})})},stop:async()=>{if(f.stop(),await T.shutdownEmbeddingPipeline(),T.stopAllCrons(),D){for(let n of V)n.destroy();return V.clear(),new Promise((n)=>{D.close(()=>{n()})})}}})},lq8=Object.freeze({create:Hb6});H6();var yg=O8.create({namespace:"caw:main"}),Gb6=2229,fb6=Bg.join(Bg.dirname(Xb6(import.meta.url)),"public_html"),Ab6=!0,Tb6=!0,Pb6=!0,jb6=!0,nb6=!1,Vb6=!0,Mb6=!0,xb6=!0,kb6="debug",sc=(_,v)=>_!==void 0?_==="true":v,Nb6=()=>{let _=process.env.CAW_ENV_DIR;if(!_)yg.error("ERROR: CAW_ENV_DIR environment variable is required"),yg.error("Set it to the path of your environment folder, e.g.:"),yg.error(' export CAW_ENV_DIR="$HOME/.caw/prod"'),process.exit(1);return{envDir:_,databaseFilepath:Bg.join(_,"caw.db"),assetsDir:Bg.join(_,"assets")}},Db6=async(_)=>{let v=await lq8.create(_),u=await v.start();process.stdout.write(`Server listening on port ${u}
1248
- `),process.on("SIGINT",()=>v.stop("SIGINT")),process.on("SIGTERM",()=>v.stop("SIGTERM"))},mb6=()=>{let{envDir:_,databaseFilepath:v,assetsDir:u}=Nb6(),K=process.env.CAW_PORT?Number(process.env.CAW_PORT):Gb6,q=process.env.STATIC_PATH||fb6,Y=sc(process.env.LAZY_CREATE_DB,Ab6),c=sc(process.env.ENABLE_REST,Tb6),z=sc(process.env.ENABLE_MCP,Pb6),w=sc(process.env.ENABLE_CLI,jb6),J=sc(process.env.ENABLE_WEBLLM_WSS,nb6),W=sc(process.env.ENABLE_SSE,Vb6),i=sc(process.env.ENABLE_STATIC,Mb6),Z=sc(process.env.ENABLE_CRONS,xb6),O=process.env.LOG_LEVEL||kb6,p=(f,j)=>process.kill(f,j),H=process.platform,X={...process.env},G="1.0.219",T=process.env.CAW_CLI_MCP_URL;return Object.freeze({port:K,staticPath:q,envDir:_,databaseFilepath:v,assetsDir:u,lazilyCreateDb:Y,enableRest:c,enableMcp:z,enableCli:w,enableWebllmWss:J,enableSse:W,enableStatic:i,enableCrons:Z,logLevel:O,killProcess:p,platform:H,processEnv:X,buildVersion:"1.0.219",cliMcpUrl:T})},yq8=mb6();O8.setLevel(yq8.logLevel);await Db6(yq8).catch((_)=>{yg.error("ERROR: Failed to start server:",_),process.exit(1)});
1248
+ `),process.on("SIGINT",()=>v.stop("SIGINT")),process.on("SIGTERM",()=>v.stop("SIGTERM"))},mb6=()=>{let{envDir:_,databaseFilepath:v,assetsDir:u}=Nb6(),K=process.env.CAW_PORT?Number(process.env.CAW_PORT):Gb6,q=process.env.STATIC_PATH||fb6,Y=sc(process.env.LAZY_CREATE_DB,Ab6),c=sc(process.env.ENABLE_REST,Tb6),z=sc(process.env.ENABLE_MCP,Pb6),w=sc(process.env.ENABLE_CLI,jb6),J=sc(process.env.ENABLE_WEBLLM_WSS,nb6),W=sc(process.env.ENABLE_SSE,Vb6),i=sc(process.env.ENABLE_STATIC,Mb6),Z=sc(process.env.ENABLE_CRONS,xb6),O=process.env.LOG_LEVEL||kb6,p=(f,j)=>process.kill(f,j),H=process.platform,X={...process.env},G="1.0.221",T=process.env.CAW_CLI_MCP_URL;return Object.freeze({port:K,staticPath:q,envDir:_,databaseFilepath:v,assetsDir:u,lazilyCreateDb:Y,enableRest:c,enableMcp:z,enableCli:w,enableWebllmWss:J,enableSse:W,enableStatic:i,enableCrons:Z,logLevel:O,killProcess:p,platform:H,processEnv:X,buildVersion:"1.0.221",cliMcpUrl:T})},yq8=mb6();O8.setLevel(yq8.logLevel);await Db6(yq8).catch((_)=>{yg.error("ERROR: Failed to start server:",_),process.exit(1)});
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@kelceyp/caw-server",
3
- "version": "1.0.220",
3
+ "version": "1.0.221",
4
4
  "description": "CAW server - REST API, MCP server and core features",
5
5
  "type": "module",
6
6
  "main": "dist/main.js",