trueline-mcp 2.5.6 → 2.5.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/server.js CHANGED
@@ -42,7 +42,7 @@ Set the \`cycles\` parameter to \`"ref"\` to resolve cyclical schemas with defs.
42
42
  `:`[${J[z]}\r
43
43
  ]`;continue}if(Y+=J[z],J[z]==="\\")G=!0;else if(W&&J[z]==="]")W=!1;else if(!W&&J[z]==="[")W=!0}try{new RegExp(Y)}catch{return console.warn(`Could not convert regex pattern at ${X.currentPath.join("/")} to a flag-independent form! Falling back to the flag-ignorant source`),Q.source}return Y}function SX(Q,X){if(X.target==="openAi")console.warn("Warning: OpenAI may not support records in schemas! Try an array of key-value pairs instead.");if(X.target==="openApi3"&&Q.keyType?._def.typeName===R.ZodEnum)return{type:"object",required:Q.keyType._def.values,properties:Q.keyType._def.values.reduce((J,Y)=>({...J,[Y]:c(Q.valueType._def,{...X,currentPath:[...X.currentPath,"properties",Y]})??R0(X)}),{}),additionalProperties:X.rejectedAdditionalProperties};let $={type:"object",additionalProperties:c(Q.valueType._def,{...X,currentPath:[...X.currentPath,"additionalProperties"]})??X.allowedAdditionalProperties};if(X.target==="openApi3")return $;if(Q.keyType?._def.typeName===R.ZodString&&Q.keyType._def.checks?.length){let{type:J,...Y}=RX(Q.keyType._def,X);return{...$,propertyNames:Y}}else if(Q.keyType?._def.typeName===R.ZodEnum)return{...$,propertyNames:{enum:Q.keyType._def.values}};else if(Q.keyType?._def.typeName===R.ZodBranded&&Q.keyType._def.type._def.typeName===R.ZodString&&Q.keyType._def.type._def.checks?.length){let{type:J,...Y}=MX(Q.keyType._def,X);return{...$,propertyNames:Y}}return $}function zW(Q,X){if(X.mapStrategy==="record")return SX(Q,X);let $=c(Q.keyType._def,{...X,currentPath:[...X.currentPath,"items","items","0"]})||R0(X),J=c(Q.valueType._def,{...X,currentPath:[...X.currentPath,"items","items","1"]})||R0(X);return{type:"array",maxItems:125,items:{type:"array",items:[$,J],minItems:2,maxItems:2}}}function KW(Q){let X=Q.values,J=Object.keys(Q.values).filter((G)=>{return typeof X[X[G]]!=="number"}).map((G)=>X[G]),Y=Array.from(new Set(J.map((G)=>typeof G)));return{type:Y.length===1?Y[0]==="string"?"string":"number":["string","number"],enum:J}}function qW(Q){return Q.target==="openAi"?void 0:{not:R0({...Q,currentPath:[...Q.currentPath,"not"]})}}function BW(Q){return Q.target==="openApi3"?{enum:["null"],nullable:!0}:{type:"null"}}var X8={ZodString:"string",ZodNumber:"number",ZodBigInt:"integer",ZodBoolean:"boolean",ZodNull:"null"};function UW(Q,X){if(X.target==="openApi3")return VW(Q,X);let $=Q.options instanceof Map?Array.from(Q.options.values()):Q.options;if($.every((J)=>(J._def.typeName in X8)&&(!J._def.checks||!J._def.checks.length))){let J=$.reduce((Y,G)=>{let W=X8[G._def.typeName];return W&&!Y.includes(W)?[...Y,W]:Y},[]);return{type:J.length>1?J:J[0]}}else if($.every((J)=>J._def.typeName==="ZodLiteral"&&!J.description)){let J=$.reduce((Y,G)=>{let W=typeof G._def.value;switch(W){case"string":case"number":case"boolean":return[...Y,W];case"bigint":return[...Y,"integer"];case"object":if(G._def.value===null)return[...Y,"null"];case"symbol":case"undefined":case"function":default:return Y}},[]);if(J.length===$.length){let Y=J.filter((G,W,H)=>H.indexOf(G)===W);return{type:Y.length>1?Y:Y[0],enum:$.reduce((G,W)=>{return G.includes(W._def.value)?G:[...G,W._def.value]},[])}}}else if($.every((J)=>J._def.typeName==="ZodEnum"))return{type:"string",enum:$.reduce((J,Y)=>[...J,...Y._def.values.filter((G)=>!J.includes(G))],[])};return VW(Q,X)}var VW=(Q,X)=>{let $=(Q.options instanceof Map?Array.from(Q.options.values()):Q.options).map((J,Y)=>c(J._def,{...X,currentPath:[...X.currentPath,"anyOf",`${Y}`]})).filter((J)=>!!J&&(!X.strictUnions||typeof J==="object"&&Object.keys(J).length>0));return $.length?{anyOf:$}:void 0};function AW(Q,X){if(["ZodString","ZodNumber","ZodBigInt","ZodBoolean","ZodNull"].includes(Q.innerType._def.typeName)&&(!Q.innerType._def.checks||!Q.innerType._def.checks.length)){if(X.target==="openApi3")return{type:X8[Q.innerType._def.typeName],nullable:!0};return{type:[X8[Q.innerType._def.typeName],"null"]}}if(X.target==="openApi3"){let J=c(Q.innerType._def,{...X,currentPath:[...X.currentPath]});if(J&&"$ref"in J)return{allOf:[J],nullable:!0};return J&&{...J,nullable:!0}}let $=c(Q.innerType._def,{...X,currentPath:[...X.currentPath,"anyOf","0"]});return $&&{anyOf:[$,{type:"null"}]}}function DW(Q,X){let $={type:"number"};if(!Q.checks)return $;for(let J of Q.checks)switch(J.kind){case"int":$.type="integer",sJ($,"type",J.message,X);break;case"min":if(X.target==="jsonSchema7")if(J.inclusive)$0($,"minimum",J.value,J.message,X);else $0($,"exclusiveMinimum",J.value,J.message,X);else{if(!J.inclusive)$.exclusiveMinimum=!0;$0($,"minimum",J.value,J.message,X)}break;case"max":if(X.target==="jsonSchema7")if(J.inclusive)$0($,"maximum",J.value,J.message,X);else $0($,"exclusiveMaximum",J.value,J.message,X);else{if(!J.inclusive)$.exclusiveMaximum=!0;$0($,"maximum",J.value,J.message,X)}break;case"multipleOf":$0($,"multipleOf",J.value,J.message,X);break}return $}function FW(Q,X){let $=X.target==="openAi",J={type:"object",properties:{}},Y=[],G=Q.shape();for(let H in G){let z=G[H];if(z===void 0||z._def===void 0)continue;let K=HD(z);if(K&&$){if(z._def.typeName==="ZodOptional")z=z._def.innerType;if(!z.isNullable())z=z.nullable();K=!1}let q=c(z._def,{...X,currentPath:[...X.currentPath,"properties",H],propertyPath:[...X.currentPath,"properties",H]});if(q===void 0)continue;if(J.properties[H]=q,!K)Y.push(H)}if(Y.length)J.required=Y;let W=WD(Q,X);if(W!==void 0)J.additionalProperties=W;return J}function WD(Q,X){if(Q.catchall._def.typeName!=="ZodNever")return c(Q.catchall._def,{...X,currentPath:[...X.currentPath,"additionalProperties"]});switch(Q.unknownKeys){case"passthrough":return X.allowedAdditionalProperties;case"strict":return X.rejectedAdditionalProperties;case"strip":return X.removeAdditionalStrategy==="strict"?X.allowedAdditionalProperties:X.rejectedAdditionalProperties}}function HD(Q){try{return Q.isOptional()}catch{return!0}}var OW=(Q,X)=>{if(X.currentPath.toString()===X.propertyPath?.toString())return c(Q.innerType._def,X);let $=c(Q.innerType._def,{...X,currentPath:[...X.currentPath,"anyOf","1"]});return $?{anyOf:[{not:R0(X)},$]}:R0(X)};var LW=(Q,X)=>{if(X.pipeStrategy==="input")return c(Q.in._def,X);else if(X.pipeStrategy==="output")return c(Q.out._def,X);let $=c(Q.in._def,{...X,currentPath:[...X.currentPath,"allOf","0"]}),J=c(Q.out._def,{...X,currentPath:[...X.currentPath,"allOf",$?"1":"0"]});return{allOf:[$,J].filter((Y)=>Y!==void 0)}};function wW(Q,X){return c(Q.type._def,X)}function jW(Q,X){let J={type:"array",uniqueItems:!0,items:c(Q.valueType._def,{...X,currentPath:[...X.currentPath,"items"]})};if(Q.minSize)$0(J,"minItems",Q.minSize.value,Q.minSize.message,X);if(Q.maxSize)$0(J,"maxItems",Q.maxSize.value,Q.maxSize.message,X);return J}function NW(Q,X){if(Q.rest)return{type:"array",minItems:Q.items.length,items:Q.items.map(($,J)=>c($._def,{...X,currentPath:[...X.currentPath,"items",`${J}`]})).reduce(($,J)=>J===void 0?$:[...$,J],[]),additionalItems:c(Q.rest._def,{...X,currentPath:[...X.currentPath,"additionalItems"]})};else return{type:"array",minItems:Q.items.length,maxItems:Q.items.length,items:Q.items.map(($,J)=>c($._def,{...X,currentPath:[...X.currentPath,"items",`${J}`]})).reduce(($,J)=>J===void 0?$:[...$,J],[])}}function EW(Q){return{not:R0(Q)}}function MW(Q){return R0(Q)}var RW=(Q,X)=>{return c(Q.innerType._def,X)};var SW=(Q,X,$)=>{switch(X){case R.ZodString:return RX(Q,$);case R.ZodNumber:return DW(Q,$);case R.ZodObject:return FW(Q,$);case R.ZodBigInt:return eG(Q,$);case R.ZodBoolean:return QW();case R.ZodDate:return eJ(Q,$);case R.ZodUndefined:return EW($);case R.ZodNull:return BW($);case R.ZodArray:return sG(Q,$);case R.ZodUnion:case R.ZodDiscriminatedUnion:return UW(Q,$);case R.ZodIntersection:return GW(Q,$);case R.ZodTuple:return NW(Q,$);case R.ZodRecord:return SX(Q,$);case R.ZodLiteral:return WW(Q,$);case R.ZodEnum:return YW(Q);case R.ZodNativeEnum:return KW(Q);case R.ZodNullable:return AW(Q,$);case R.ZodOptional:return OW(Q,$);case R.ZodMap:return zW(Q,$);case R.ZodSet:return jW(Q,$);case R.ZodLazy:return()=>Q.getter()._def;case R.ZodPromise:return wW(Q,$);case R.ZodNaN:case R.ZodNever:return qW($);case R.ZodEffects:return JW(Q,$);case R.ZodAny:return R0($);case R.ZodUnknown:return MW($);case R.ZodDefault:return $W(Q,$);case R.ZodBranded:return MX(Q,$);case R.ZodReadonly:return RW(Q,$);case R.ZodCatch:return XW(Q,$);case R.ZodPipeline:return LW(Q,$);case R.ZodFunction:case R.ZodVoid:case R.ZodSymbol:return;default:return((J)=>{return})(X)}};function c(Q,X,$=!1){let J=X.seen.get(Q);if(X.override){let H=X.override?.(Q,X,J,$);if(H!==rG)return H}if(J&&!$){let H=zD(J,X);if(H!==void 0)return H}let Y={def:Q,path:X.currentPath,jsonSchema:void 0};X.seen.set(Q,Y);let G=SW(Q,Q.typeName,X),W=typeof G==="function"?c(G(),X):G;if(W)KD(Q,X,W);if(X.postProcess){let H=X.postProcess(W,Q,X);return Y.jsonSchema=W,H}return Y.jsonSchema=W,W}var zD=(Q,X)=>{switch(X.$refStrategy){case"root":return{$ref:Q.path.join("/")};case"relative":return{$ref:EX(X.currentPath,Q.path)};case"none":case"seen":{if(Q.path.length<X.currentPath.length&&Q.path.every(($,J)=>X.currentPath[J]===$))return console.warn(`Recursive reference detected at ${X.currentPath.join("/")}! Defaulting to any`),R0(X);return X.$refStrategy==="seen"?R0(X):void 0}}},KD=(Q,X,$)=>{if(Q.description){if($.description=Q.description,X.markdownDescription)$.markdownDescription=Q.description}return $};var $7=(Q,X)=>{let $=aG(X),J=typeof X==="object"&&X.definitions?Object.entries(X.definitions).reduce((z,[K,q])=>({...z,[K]:c(q._def,{...$,currentPath:[...$.basePath,$.definitionPath,K]},!0)??R0($)}),{}):void 0,Y=typeof X==="string"?X:X?.nameStrategy==="title"?void 0:X?.name,G=c(Q._def,Y===void 0?$:{...$,currentPath:[...$.basePath,$.definitionPath,Y]},!1)??R0($),W=typeof X==="object"&&X.name!==void 0&&X.nameStrategy==="title"?X.name:void 0;if(W!==void 0)G.title=W;if($.flags.hasReferencedOpenAiAnyType){if(!J)J={};if(!J[$.openAiAnyTypeName])J[$.openAiAnyTypeName]={type:["string","number","integer","boolean","array","null"],items:{$ref:$.$refStrategy==="relative"?"1":[...$.basePath,$.definitionPath,$.openAiAnyTypeName].join("/")}}}let H=Y===void 0?J?{...G,[$.definitionPath]:J}:G:{$ref:[...$.$refStrategy==="relative"?[]:$.basePath,$.definitionPath,Y].join("/"),[$.definitionPath]:{...J,[Y]:G}};if($.target==="jsonSchema7")H.$schema="http://json-schema.org/draft-07/schema#";else if($.target==="jsonSchema2019-09"||$.target==="openAi")H.$schema="https://json-schema.org/draft/2019-09/schema#";if($.target==="openAi"&&(("anyOf"in H)||("oneOf"in H)||("allOf"in H)||("type"in H)&&Array.isArray(H.type)))console.warn("Warning: OpenAI may not support schemas with unions as roots! Try wrapping it in an object property.");return H};function qD(Q){if(!Q)return"draft-7";if(Q==="jsonSchema7"||Q==="draft-7")return"draft-7";if(Q==="jsonSchema2019-09"||Q==="draft-2020-12")return"draft-2020-12";return"draft-7"}function J7(Q,X){if(z6(Q))return SJ(Q,{target:qD(X?.target),io:X?.pipeStrategy??"input"});return $7(Q,{strictUnions:X?.strictUnions??!0,pipeStrategy:X?.pipeStrategy??"input"})}function Y7(Q){let $=J9(Q)?.method;if(!$)throw Error("Schema is missing a method literal");let J=s8($);if(typeof J!=="string")throw Error("Schema method literal must be a string");return J}function G7(Q,X){let $=$9(Q,X);if(!$.success)throw $.error;return $.data}var BD=60000;class W7{constructor(Q){if(this._options=Q,this._requestMessageId=0,this._requestHandlers=new Map,this._requestHandlerAbortControllers=new Map,this._notificationHandlers=new Map,this._responseHandlers=new Map,this._progressHandlers=new Map,this._timeoutInfo=new Map,this._pendingDebouncedNotifications=new Set,this._taskProgressTokens=new Map,this._requestResolvers=new Map,this.setNotificationHandler(GX,(X)=>{this._oncancel(X)}),this.setNotificationHandler(HX,(X)=>{this._onprogress(X)}),this.setRequestHandler(WX,(X)=>({})),this._taskStore=Q?.taskStore,this._taskMessageQueue=Q?.taskMessageQueue,this._taskStore)this.setRequestHandler(zX,async(X,$)=>{let J=await this._taskStore.getTask(X.params.taskId,$.sessionId);if(!J)throw new x(h.InvalidParams,"Failed to retrieve task: Task not found");return{...J}}),this.setRequestHandler(qX,async(X,$)=>{let J=async()=>{let Y=X.params.taskId;if(this._taskMessageQueue){let W;while(W=await this._taskMessageQueue.dequeue(Y,$.sessionId)){if(W.type==="response"||W.type==="error"){let H=W.message,z=H.id,K=this._requestResolvers.get(z);if(K)if(this._requestResolvers.delete(z),W.type==="response")K(H);else{let q=H,B=new x(q.error.code,q.error.message,q.error.data);K(B)}else{let q=W.type==="response"?"Response":"Error";this._onerror(Error(`${q} handler missing for request ${z}`))}continue}await this._transport?.send(W.message,{relatedRequestId:$.requestId})}}let G=await this._taskStore.getTask(Y,$.sessionId);if(!G)throw new x(h.InvalidParams,`Task not found: ${Y}`);if(!G9(G.status))return await this._waitForTaskUpdate(Y,$.signal),await J();if(G9(G.status)){let W=await this._taskStore.getTaskResult(Y,$.sessionId);return this._clearTaskQueue(Y),{...W,_meta:{...W._meta,[Y9]:{taskId:Y}}}}return await J()};return await J()}),this.setRequestHandler(BX,async(X,$)=>{try{let{tasks:J,nextCursor:Y}=await this._taskStore.listTasks(X.params?.cursor,$.sessionId);return{tasks:J,nextCursor:Y,_meta:{}}}catch(J){throw new x(h.InvalidParams,`Failed to list tasks: ${J instanceof Error?J.message:String(J)}`)}}),this.setRequestHandler(UX,async(X,$)=>{try{let J=await this._taskStore.getTask(X.params.taskId,$.sessionId);if(!J)throw new x(h.InvalidParams,`Task not found: ${X.params.taskId}`);if(G9(J.status))throw new x(h.InvalidParams,`Cannot cancel task in terminal status: ${J.status}`);await this._taskStore.updateTaskStatus(X.params.taskId,"cancelled","Client cancelled task execution.",$.sessionId),this._clearTaskQueue(X.params.taskId);let Y=await this._taskStore.getTask(X.params.taskId,$.sessionId);if(!Y)throw new x(h.InvalidParams,`Task not found after cancellation: ${X.params.taskId}`);return{_meta:{},...Y}}catch(J){if(J instanceof x)throw J;throw new x(h.InvalidRequest,`Failed to cancel task: ${J instanceof Error?J.message:String(J)}`)}})}async _oncancel(Q){if(!Q.params.requestId)return;this._requestHandlerAbortControllers.get(Q.params.requestId)?.abort(Q.params.reason)}_setupTimeout(Q,X,$,J,Y=!1){this._timeoutInfo.set(Q,{timeoutId:setTimeout(J,X),startTime:Date.now(),timeout:X,maxTotalTimeout:$,resetTimeoutOnProgress:Y,onTimeout:J})}_resetTimeout(Q){let X=this._timeoutInfo.get(Q);if(!X)return!1;let $=Date.now()-X.startTime;if(X.maxTotalTimeout&&$>=X.maxTotalTimeout)throw this._timeoutInfo.delete(Q),x.fromError(h.RequestTimeout,"Maximum total timeout exceeded",{maxTotalTimeout:X.maxTotalTimeout,totalElapsed:$});return clearTimeout(X.timeoutId),X.timeoutId=setTimeout(X.onTimeout,X.timeout),!0}_cleanupTimeout(Q){let X=this._timeoutInfo.get(Q);if(X)clearTimeout(X.timeoutId),this._timeoutInfo.delete(Q)}async connect(Q){if(this._transport)throw Error("Already connected to a transport. Call close() before connecting to a new transport, or use a separate Protocol instance per connection.");this._transport=Q;let X=this.transport?.onclose;this._transport.onclose=()=>{X?.(),this._onclose()};let $=this.transport?.onerror;this._transport.onerror=(Y)=>{$?.(Y),this._onerror(Y)};let J=this._transport?.onmessage;this._transport.onmessage=(Y,G)=>{if(J?.(Y,G),nQ(Y)||gG(Y))this._onresponse(Y);else if(hJ(Y))this._onrequest(Y,G);else if(xG(Y))this._onnotification(Y);else this._onerror(Error(`Unknown message type: ${JSON.stringify(Y)}`))},await this._transport.start()}_onclose(){let Q=this._responseHandlers;this._responseHandlers=new Map,this._progressHandlers.clear(),this._taskProgressTokens.clear(),this._pendingDebouncedNotifications.clear();for(let $ of this._requestHandlerAbortControllers.values())$.abort();this._requestHandlerAbortControllers.clear();let X=x.fromError(h.ConnectionClosed,"Connection closed");this._transport=void 0,this.onclose?.();for(let $ of Q.values())$(X)}_onerror(Q){this.onerror?.(Q)}_onnotification(Q){let X=this._notificationHandlers.get(Q.method)??this.fallbackNotificationHandler;if(X===void 0)return;Promise.resolve().then(()=>X(Q)).catch(($)=>this._onerror(Error(`Uncaught error in notification handler: ${$}`)))}_onrequest(Q,X){let $=this._requestHandlers.get(Q.method)??this.fallbackRequestHandler,J=this._transport,Y=Q.params?._meta?.[Y9]?.taskId;if($===void 0){let K={jsonrpc:"2.0",id:Q.id,error:{code:h.MethodNotFound,message:"Method not found"}};if(Y&&this._taskMessageQueue)this._enqueueTaskMessage(Y,{type:"error",message:K,timestamp:Date.now()},J?.sessionId).catch((q)=>this._onerror(Error(`Failed to enqueue error response: ${q}`)));else J?.send(K).catch((q)=>this._onerror(Error(`Failed to send an error response: ${q}`)));return}let G=new AbortController;this._requestHandlerAbortControllers.set(Q.id,G);let W=kG(Q.params)?Q.params.task:void 0,H=this._taskStore?this.requestTaskStore(Q,J?.sessionId):void 0,z={signal:G.signal,sessionId:J?.sessionId,_meta:Q.params?._meta,sendNotification:async(K)=>{if(G.signal.aborted)return;let q={relatedRequestId:Q.id};if(Y)q.relatedTask={taskId:Y};await this.notification(K,q)},sendRequest:async(K,q,B)=>{if(G.signal.aborted)throw new x(h.ConnectionClosed,"Request was cancelled");let A={...B,relatedRequestId:Q.id};if(Y&&!A.relatedTask)A.relatedTask={taskId:Y};let U=A.relatedTask?.taskId??Y;if(U&&H)await H.updateTaskStatus(U,"input_required");return await this.request(K,q,A)},authInfo:X?.authInfo,requestId:Q.id,requestInfo:X?.requestInfo,taskId:Y,taskStore:H,taskRequestedTtl:W?.ttl,closeSSEStream:X?.closeSSEStream,closeStandaloneSSEStream:X?.closeStandaloneSSEStream};Promise.resolve().then(()=>{if(W)this.assertTaskHandlerCapability(Q.method)}).then(()=>$(Q,z)).then(async(K)=>{if(G.signal.aborted)return;let q={result:K,jsonrpc:"2.0",id:Q.id};if(Y&&this._taskMessageQueue)await this._enqueueTaskMessage(Y,{type:"response",message:q,timestamp:Date.now()},J?.sessionId);else await J?.send(q)},async(K)=>{if(G.signal.aborted)return;let q={jsonrpc:"2.0",id:Q.id,error:{code:Number.isSafeInteger(K.code)?K.code:h.InternalError,message:K.message??"Internal error",...K.data!==void 0&&{data:K.data}}};if(Y&&this._taskMessageQueue)await this._enqueueTaskMessage(Y,{type:"error",message:q,timestamp:Date.now()},J?.sessionId);else await J?.send(q)}).catch((K)=>this._onerror(Error(`Failed to send response: ${K}`))).finally(()=>{this._requestHandlerAbortControllers.delete(Q.id)})}_onprogress(Q){let{progressToken:X,...$}=Q.params,J=Number(X),Y=this._progressHandlers.get(J);if(!Y){this._onerror(Error(`Received a progress notification for an unknown token: ${JSON.stringify(Q)}`));return}let G=this._responseHandlers.get(J),W=this._timeoutInfo.get(J);if(W&&G&&W.resetTimeoutOnProgress)try{this._resetTimeout(J)}catch(H){this._responseHandlers.delete(J),this._progressHandlers.delete(J),this._cleanupTimeout(J),G(H);return}Y($)}_onresponse(Q){let X=Number(Q.id),$=this._requestResolvers.get(X);if($){if(this._requestResolvers.delete(X),nQ(Q))$(Q);else{let G=new x(Q.error.code,Q.error.message,Q.error.data);$(G)}return}let J=this._responseHandlers.get(X);if(J===void 0){this._onerror(Error(`Received a response for an unknown message ID: ${JSON.stringify(Q)}`));return}this._responseHandlers.delete(X),this._cleanupTimeout(X);let Y=!1;if(nQ(Q)&&Q.result&&typeof Q.result==="object"){let G=Q.result;if(G.task&&typeof G.task==="object"){let W=G.task;if(typeof W.taskId==="string")Y=!0,this._taskProgressTokens.set(W.taskId,X)}}if(!Y)this._progressHandlers.delete(X);if(nQ(Q))J(Q);else{let G=x.fromError(Q.error.code,Q.error.message,Q.error.data);J(G)}}get transport(){return this._transport}async close(){await this._transport?.close()}async*requestStream(Q,X,$){let{task:J}=$??{};if(!J){try{yield{type:"result",result:await this.request(Q,X,$)}}catch(G){yield{type:"error",error:G instanceof x?G:new x(h.InternalError,String(G))}}return}let Y;try{let G=await this.request(Q,$Q,$);if(G.task)Y=G.task.taskId,yield{type:"taskCreated",task:G.task};else throw new x(h.InternalError,"Task creation did not return a task");while(!0){let W=await this.getTask({taskId:Y},$);if(yield{type:"taskStatus",task:W},G9(W.status)){if(W.status==="completed")yield{type:"result",result:await this.getTaskResult({taskId:Y},X,$)};else if(W.status==="failed")yield{type:"error",error:new x(h.InternalError,`Task ${Y} failed`)};else if(W.status==="cancelled")yield{type:"error",error:new x(h.InternalError,`Task ${Y} was cancelled`)};return}if(W.status==="input_required"){yield{type:"result",result:await this.getTaskResult({taskId:Y},X,$)};return}let H=W.pollInterval??this._options?.defaultTaskPollInterval??1000;await new Promise((z)=>setTimeout(z,H)),$?.signal?.throwIfAborted()}}catch(G){yield{type:"error",error:G instanceof x?G:new x(h.InternalError,String(G))}}}request(Q,X,$){let{relatedRequestId:J,resumptionToken:Y,onresumptiontoken:G,task:W,relatedTask:H}=$??{};return new Promise((z,K)=>{let q=(O)=>{K(O)};if(!this._transport){q(Error("Not connected"));return}if(this._options?.enforceStrictCapabilities===!0)try{if(this.assertCapabilityForMethod(Q.method),W)this.assertTaskCapability(Q.method)}catch(O){q(O);return}$?.signal?.throwIfAborted();let B=this._requestMessageId++,A={...Q,jsonrpc:"2.0",id:B};if($?.onprogress)this._progressHandlers.set(B,$.onprogress),A.params={...Q.params,_meta:{...Q.params?._meta||{},progressToken:B}};if(W)A.params={...A.params,task:W};if(H)A.params={...A.params,_meta:{...A.params?._meta||{},[Y9]:H}};let U=(O)=>{this._responseHandlers.delete(B),this._progressHandlers.delete(B),this._cleanupTimeout(B),this._transport?.send({jsonrpc:"2.0",method:"notifications/cancelled",params:{requestId:B,reason:String(O)}},{relatedRequestId:J,resumptionToken:Y,onresumptiontoken:G}).catch((E)=>this._onerror(Error(`Failed to send cancellation: ${E}`)));let w=O instanceof x?O:new x(h.RequestTimeout,String(O));K(w)};this._responseHandlers.set(B,(O)=>{if($?.signal?.aborted)return;if(O instanceof Error)return K(O);try{let w=$9(X,O.result);if(!w.success)K(w.error);else z(w.data)}catch(w){K(w)}}),$?.signal?.addEventListener("abort",()=>{U($?.signal?.reason)});let D=$?.timeout??BD,F=()=>U(x.fromError(h.RequestTimeout,"Request timed out",{timeout:D}));this._setupTimeout(B,D,$?.maxTotalTimeout,F,$?.resetTimeoutOnProgress??!1);let L=H?.taskId;if(L){let O=(w)=>{let E=this._responseHandlers.get(B);if(E)E(w);else this._onerror(Error(`Response handler missing for side-channeled request ${B}`))};this._requestResolvers.set(B,O),this._enqueueTaskMessage(L,{type:"request",message:A,timestamp:Date.now()}).catch((w)=>{this._cleanupTimeout(B),K(w)})}else this._transport.send(A,{relatedRequestId:J,resumptionToken:Y,onresumptiontoken:G}).catch((O)=>{this._cleanupTimeout(B),K(O)})})}async getTask(Q,X){return this.request({method:"tasks/get",params:Q},KX,X)}async getTaskResult(Q,X,$){return this.request({method:"tasks/result",params:Q},X,$)}async listTasks(Q,X){return this.request({method:"tasks/list",params:Q},VX,X)}async cancelTask(Q,X){return this.request({method:"tasks/cancel",params:Q},fG,X)}async notification(Q,X){if(!this._transport)throw Error("Not connected");this.assertNotificationCapability(Q.method);let $=X?.relatedTask?.taskId;if($){let W={...Q,jsonrpc:"2.0",params:{...Q.params,_meta:{...Q.params?._meta||{},[Y9]:X.relatedTask}}};await this._enqueueTaskMessage($,{type:"notification",message:W,timestamp:Date.now()});return}if((this._options?.debouncedNotificationMethods??[]).includes(Q.method)&&!Q.params&&!X?.relatedRequestId&&!X?.relatedTask){if(this._pendingDebouncedNotifications.has(Q.method))return;this._pendingDebouncedNotifications.add(Q.method),Promise.resolve().then(()=>{if(this._pendingDebouncedNotifications.delete(Q.method),!this._transport)return;let W={...Q,jsonrpc:"2.0"};if(X?.relatedTask)W={...W,params:{...W.params,_meta:{...W.params?._meta||{},[Y9]:X.relatedTask}}};this._transport?.send(W,X).catch((H)=>this._onerror(H))});return}let G={...Q,jsonrpc:"2.0"};if(X?.relatedTask)G={...G,params:{...G.params,_meta:{...G.params?._meta||{},[Y9]:X.relatedTask}}};await this._transport.send(G,X)}setRequestHandler(Q,X){let $=Y7(Q);this.assertRequestHandlerCapability($),this._requestHandlers.set($,(J,Y)=>{let G=G7(Q,J);return Promise.resolve(X(G,Y))})}removeRequestHandler(Q){this._requestHandlers.delete(Q)}assertCanSetRequestHandler(Q){if(this._requestHandlers.has(Q))throw Error(`A request handler for ${Q} already exists, which would be overridden`)}setNotificationHandler(Q,X){let $=Y7(Q);this._notificationHandlers.set($,(J)=>{let Y=G7(Q,J);return Promise.resolve(X(Y))})}removeNotificationHandler(Q){this._notificationHandlers.delete(Q)}_cleanupTaskProgressHandler(Q){let X=this._taskProgressTokens.get(Q);if(X!==void 0)this._progressHandlers.delete(X),this._taskProgressTokens.delete(Q)}async _enqueueTaskMessage(Q,X,$){if(!this._taskStore||!this._taskMessageQueue)throw Error("Cannot enqueue task message: taskStore and taskMessageQueue are not configured");let J=this._options?.maxTaskQueueSize;await this._taskMessageQueue.enqueue(Q,X,$,J)}async _clearTaskQueue(Q,X){if(this._taskMessageQueue){let $=await this._taskMessageQueue.dequeueAll(Q,X);for(let J of $)if(J.type==="request"&&hJ(J.message)){let Y=J.message.id,G=this._requestResolvers.get(Y);if(G)G(new x(h.InternalError,"Task cancelled or completed")),this._requestResolvers.delete(Y);else this._onerror(Error(`Resolver missing for request ${Y} during task ${Q} cleanup`))}}}async _waitForTaskUpdate(Q,X){let $=this._options?.defaultTaskPollInterval??1000;try{let J=await this._taskStore?.getTask(Q);if(J?.pollInterval)$=J.pollInterval}catch{}return new Promise((J,Y)=>{if(X.aborted){Y(new x(h.InvalidRequest,"Request cancelled"));return}let G=setTimeout(J,$);X.addEventListener("abort",()=>{clearTimeout(G),Y(new x(h.InvalidRequest,"Request cancelled"))},{once:!0})})}requestTaskStore(Q,X){let $=this._taskStore;if(!$)throw Error("No task store configured");return{createTask:async(J)=>{if(!Q)throw Error("No request provided");return await $.createTask(J,Q.id,{method:Q.method,params:Q.params},X)},getTask:async(J)=>{let Y=await $.getTask(J,X);if(!Y)throw new x(h.InvalidParams,"Failed to retrieve task: Task not found");return Y},storeTaskResult:async(J,Y,G)=>{await $.storeTaskResult(J,Y,G,X);let W=await $.getTask(J,X);if(W){let H=aQ.parse({method:"notifications/tasks/status",params:W});if(await this.notification(H),G9(W.status))this._cleanupTaskProgressHandler(J)}},getTaskResult:(J)=>{return $.getTaskResult(J,X)},updateTaskStatus:async(J,Y,G)=>{let W=await $.getTask(J,X);if(!W)throw new x(h.InvalidParams,`Task "${J}" not found - it may have been cleaned up`);if(G9(W.status))throw new x(h.InvalidParams,`Cannot update task "${J}" from terminal status "${W.status}" to "${Y}". Terminal states (completed, failed, cancelled) cannot transition to other states.`);await $.updateTaskStatus(J,Y,G,X);let H=await $.getTask(J,X);if(H){let z=aQ.parse({method:"notifications/tasks/status",params:H});if(await this.notification(z),G9(H.status))this._cleanupTaskProgressHandler(J)}},listTasks:(J)=>{return $.listTasks(J,X)}}}}function PW(Q){return Q!==null&&typeof Q==="object"&&!Array.isArray(Q)}function bW(Q,X){let $={...Q};for(let J in X){let Y=J,G=X[Y];if(G===void 0)continue;let W=$[Y];if(PW(W)&&PW(G))$[Y]={...W,...G};else $[Y]=G}return $}var Y1=N4(QY(),1),G1=N4(J1(),1);function NM(){let Q=new Y1.default({strict:!1,validateFormats:!0,validateSchema:!1,allErrors:!0});return G1.default(Q),Q}class qY{constructor(Q){this._ajv=Q??NM()}getValidator(Q){let X="$id"in Q&&typeof Q.$id==="string"?this._ajv.getSchema(Q.$id)??this._ajv.compile(Q):this._ajv.compile(Q);return($)=>{if(X($))return{valid:!0,data:$,errorMessage:void 0};else return{valid:!1,data:void 0,errorMessage:this._ajv.errorsText(X.errors)}}}}class BY{constructor(Q){this._server=Q}requestStream(Q,X,$){return this._server.requestStream(Q,X,$)}createMessageStream(Q,X){let $=this._server.getClientCapabilities();if((Q.tools||Q.toolChoice)&&!$?.sampling?.tools)throw Error("Client does not support sampling tools capability.");if(Q.messages.length>0){let J=Q.messages[Q.messages.length-1],Y=Array.isArray(J.content)?J.content:[J.content],G=Y.some((K)=>K.type==="tool_result"),W=Q.messages.length>1?Q.messages[Q.messages.length-2]:void 0,H=W?Array.isArray(W.content)?W.content:[W.content]:[],z=H.some((K)=>K.type==="tool_use");if(G){if(Y.some((K)=>K.type!=="tool_result"))throw Error("The last message must contain only tool_result content if any is present");if(!z)throw Error("tool_result blocks are not matching any tool_use from the previous message")}if(z){let K=new Set(H.filter((B)=>B.type==="tool_use").map((B)=>B.id)),q=new Set(Y.filter((B)=>B.type==="tool_result").map((B)=>B.toolUseId));if(K.size!==q.size||![...K].every((B)=>q.has(B)))throw Error("ids of tool_result blocks and tool_use blocks from previous message do not match")}}return this.requestStream({method:"sampling/createMessage",params:Q},Q8,X)}elicitInputStream(Q,X){let $=this._server.getClientCapabilities(),J=Q.mode??"form";switch(J){case"url":{if(!$?.elicitation?.url)throw Error("Client does not support url elicitation.");break}case"form":{if(!$?.elicitation?.form)throw Error("Client does not support form elicitation.");break}}let Y=J==="form"&&Q.mode===void 0?{...Q,mode:"form"}:Q;return this.requestStream({method:"elicitation/create",params:Y},GQ,X)}async getTask(Q,X){return this._server.getTask({taskId:Q},X)}async getTaskResult(Q,X,$){return this._server.getTaskResult({taskId:Q},X,$)}async listTasks(Q,X){return this._server.listTasks(Q?{cursor:Q}:void 0,X)}async cancelTask(Q,X){return this._server.cancelTask({taskId:Q},X)}}function W1(Q,X,$){if(!Q)throw Error(`${$} does not support task creation (required for ${X})`);switch(X){case"tools/call":if(!Q.tools?.call)throw Error(`${$} does not support task creation for tools/call (required for ${X})`);break;default:break}}function H1(Q,X,$){if(!Q)throw Error(`${$} does not support task creation (required for ${X})`);switch(X){case"sampling/createMessage":if(!Q.sampling?.createMessage)throw Error(`${$} does not support task creation for sampling/createMessage (required for ${X})`);break;case"elicitation/create":if(!Q.elicitation?.create)throw Error(`${$} does not support task creation for elicitation/create (required for ${X})`);break;default:break}}class VY extends W7{constructor(Q,X){super(X);if(this._serverInfo=Q,this._loggingLevels=new Map,this.LOG_LEVEL_SEVERITY=new Map(eQ.options.map(($,J)=>[$,J])),this.isMessageIgnored=($,J)=>{let Y=this._loggingLevels.get(J);return Y?this.LOG_LEVEL_SEVERITY.get($)<this.LOG_LEVEL_SEVERITY.get(Y):!1},this._capabilities=X?.capabilities??{},this._instructions=X?.instructions,this._jsonSchemaValidator=X?.jsonSchemaValidator??new qY,this.setRequestHandler(mJ,($)=>this._oninitialize($)),this.setNotificationHandler(lJ,()=>this.oninitialized?.()),this._capabilities.logging)this.setRequestHandler(rJ,async($,J)=>{let Y=J.sessionId||J.requestInfo?.headers["mcp-session-id"]||void 0,{level:G}=$.params,W=eQ.safeParse(G);if(W.success)this._loggingLevels.set(Y,W.data);return{}})}get experimental(){if(!this._experimental)this._experimental={tasks:new BY(this)};return this._experimental}registerCapabilities(Q){if(this.transport)throw Error("Cannot register capabilities after connecting to transport");this._capabilities=bW(this._capabilities,Q)}setRequestHandler(Q,X){let J=J9(Q)?.method;if(!J)throw Error("Schema is missing a method literal");let Y;if(z6(J)){let W=J;Y=W._zod?.def?.value??W.value}else{let W=J;Y=W._def?.value??W.value}if(typeof Y!=="string")throw Error("Schema method literal must be a string");if(Y==="tools/call"){let W=async(H,z)=>{let K=$9(YQ,H);if(!K.success){let U=K.error instanceof Error?K.error.message:String(K.error);throw new x(h.InvalidParams,`Invalid tools/call request: ${U}`)}let{params:q}=K.data,B=await Promise.resolve(X(H,z));if(q.task){let U=$9($Q,B);if(!U.success){let D=U.error instanceof Error?U.error.message:String(U.error);throw new x(h.InvalidParams,`Invalid task creation result: ${D}`)}return U.data}let A=$9(jX,B);if(!A.success){let U=A.error instanceof Error?A.error.message:String(A.error);throw new x(h.InvalidParams,`Invalid tools/call result: ${U}`)}return A.data};return super.setRequestHandler(Q,W)}return super.setRequestHandler(Q,X)}assertCapabilityForMethod(Q){switch(Q){case"sampling/createMessage":if(!this._clientCapabilities?.sampling)throw Error(`Client does not support sampling (required for ${Q})`);break;case"elicitation/create":if(!this._clientCapabilities?.elicitation)throw Error(`Client does not support elicitation (required for ${Q})`);break;case"roots/list":if(!this._clientCapabilities?.roots)throw Error(`Client does not support listing roots (required for ${Q})`);break;case"ping":break}}assertNotificationCapability(Q){switch(Q){case"notifications/message":if(!this._capabilities.logging)throw Error(`Server does not support logging (required for ${Q})`);break;case"notifications/resources/updated":case"notifications/resources/list_changed":if(!this._capabilities.resources)throw Error(`Server does not support notifying about resources (required for ${Q})`);break;case"notifications/tools/list_changed":if(!this._capabilities.tools)throw Error(`Server does not support notifying of tool list changes (required for ${Q})`);break;case"notifications/prompts/list_changed":if(!this._capabilities.prompts)throw Error(`Server does not support notifying of prompt list changes (required for ${Q})`);break;case"notifications/elicitation/complete":if(!this._clientCapabilities?.elicitation?.url)throw Error(`Client does not support URL elicitation (required for ${Q})`);break;case"notifications/cancelled":break;case"notifications/progress":break}}assertRequestHandlerCapability(Q){if(!this._capabilities)return;switch(Q){case"completion/complete":if(!this._capabilities.completions)throw Error(`Server does not support completions (required for ${Q})`);break;case"logging/setLevel":if(!this._capabilities.logging)throw Error(`Server does not support logging (required for ${Q})`);break;case"prompts/get":case"prompts/list":if(!this._capabilities.prompts)throw Error(`Server does not support prompts (required for ${Q})`);break;case"resources/list":case"resources/templates/list":case"resources/read":if(!this._capabilities.resources)throw Error(`Server does not support resources (required for ${Q})`);break;case"tools/call":case"tools/list":if(!this._capabilities.tools)throw Error(`Server does not support tools (required for ${Q})`);break;case"tasks/get":case"tasks/list":case"tasks/result":case"tasks/cancel":if(!this._capabilities.tasks)throw Error(`Server does not support tasks capability (required for ${Q})`);break;case"ping":case"initialize":break}}assertTaskCapability(Q){H1(this._clientCapabilities?.tasks?.requests,Q,"Client")}assertTaskHandlerCapability(Q){if(!this._capabilities)return;W1(this._capabilities.tasks?.requests,Q,"Server")}async _oninitialize(Q){let X=Q.params.protocolVersion;return this._clientCapabilities=Q.params.capabilities,this._clientVersion=Q.params.clientInfo,{protocolVersion:ZG.includes(X)?X:gJ,capabilities:this.getCapabilities(),serverInfo:this._serverInfo,...this._instructions&&{instructions:this._instructions}}}getClientCapabilities(){return this._clientCapabilities}getClientVersion(){return this._clientVersion}getCapabilities(){return this._capabilities}async ping(){return this.request({method:"ping"},YX)}async createMessage(Q,X){if(Q.tools||Q.toolChoice){if(!this._clientCapabilities?.sampling?.tools)throw Error("Client does not support sampling tools capability.")}if(Q.messages.length>0){let $=Q.messages[Q.messages.length-1],J=Array.isArray($.content)?$.content:[$.content],Y=J.some((z)=>z.type==="tool_result"),G=Q.messages.length>1?Q.messages[Q.messages.length-2]:void 0,W=G?Array.isArray(G.content)?G.content:[G.content]:[],H=W.some((z)=>z.type==="tool_use");if(Y){if(J.some((z)=>z.type!=="tool_result"))throw Error("The last message must contain only tool_result content if any is present");if(!H)throw Error("tool_result blocks are not matching any tool_use from the previous message")}if(H){let z=new Set(W.filter((q)=>q.type==="tool_use").map((q)=>q.id)),K=new Set(J.filter((q)=>q.type==="tool_result").map((q)=>q.toolUseId));if(z.size!==K.size||![...z].every((q)=>K.has(q)))throw Error("ids of tool_result blocks and tool_use blocks from previous message do not match")}}if(Q.tools)return this.request({method:"sampling/createMessage",params:Q},tJ,X);return this.request({method:"sampling/createMessage",params:Q},Q8,X)}async elicitInput(Q,X){switch(Q.mode??"form"){case"url":{if(!this._clientCapabilities?.elicitation?.url)throw Error("Client does not support url elicitation.");let J=Q;return this.request({method:"elicitation/create",params:J},GQ,X)}case"form":{if(!this._clientCapabilities?.elicitation?.form)throw Error("Client does not support form elicitation.");let J=Q.mode==="form"?Q:{...Q,mode:"form"},Y=await this.request({method:"elicitation/create",params:J},GQ,X);if(Y.action==="accept"&&Y.content&&J.requestedSchema)try{let W=this._jsonSchemaValidator.getValidator(J.requestedSchema)(Y.content);if(!W.valid)throw new x(h.InvalidParams,`Elicitation response content does not match requested schema: ${W.errorMessage}`)}catch(G){if(G instanceof x)throw G;throw new x(h.InternalError,`Error validating elicitation response: ${G instanceof Error?G.message:String(G)}`)}return Y}}}createElicitationCompletionNotifier(Q,X){if(!this._clientCapabilities?.elicitation?.url)throw Error("Client does not support URL elicitation (required for notifications/elicitation/complete)");return()=>this.notification({method:"notifications/elicitation/complete",params:{elicitationId:Q}},X)}async listRoots(Q,X){return this.request({method:"roots/list",params:Q},aJ,X)}async sendLoggingMessage(Q,X){if(this._capabilities.logging){if(!this.isMessageIgnored(Q.level,X))return this.notification({method:"notifications/message",params:Q})}}async sendResourceUpdated(Q){return this.notification({method:"notifications/resources/updated",params:Q})}async sendResourceListChanged(){return this.notification({method:"notifications/resources/list_changed"})}async sendToolListChanged(){return this.notification({method:"notifications/tools/list_changed"})}async sendPromptListChanged(){return this.notification({method:"notifications/prompts/list_changed"})}}var K1=Symbol.for("mcp.completable");function UY(Q){return!!Q&&typeof Q==="object"&&K1 in Q}function q1(Q){return Q[K1]?.complete}var z1;(function(Q){Q.Completable="McpCompletable"})(z1||(z1={}));var EM=/^[A-Za-z0-9._-]{1,128}$/;function MM(Q){let X=[];if(Q.length===0)return{isValid:!1,warnings:["Tool name cannot be empty"]};if(Q.length>128)return{isValid:!1,warnings:[`Tool name exceeds maximum length of 128 characters (current: ${Q.length})`]};if(Q.includes(" "))X.push("Tool name contains spaces, which may cause parsing issues");if(Q.includes(","))X.push("Tool name contains commas, which may cause parsing issues");if(Q.startsWith("-")||Q.endsWith("-"))X.push("Tool name starts or ends with a dash, which may cause parsing issues in some contexts");if(Q.startsWith(".")||Q.endsWith("."))X.push("Tool name starts or ends with a dot, which may cause parsing issues in some contexts");if(!EM.test(Q)){let $=Q.split("").filter((J)=>!/[A-Za-z0-9._-]/.test(J)).filter((J,Y,G)=>G.indexOf(J)===Y);return X.push(`Tool name contains invalid characters: ${$.map((J)=>`"${J}"`).join(", ")}`,"Allowed characters are: A-Z, a-z, 0-9, underscore (_), dash (-), and dot (.)"),{isValid:!1,warnings:X}}return{isValid:!0,warnings:X}}function RM(Q,X){if(X.length>0){console.warn(`Tool name validation warning for "${Q}":`);for(let $ of X)console.warn(` - ${$}`);console.warn("Tool registration will proceed, but this may cause compatibility issues."),console.warn("Consider updating the tool name to conform to the MCP tool naming standard."),console.warn("See SEP: Specify Format for Tool Names (https://github.com/modelcontextprotocol/modelcontextprotocol/issues/986) for more details.")}}function AY(Q){let X=MM(Q);return RM(Q,X.warnings),X.isValid}class DY{constructor(Q){this._mcpServer=Q}registerToolTask(Q,X,$){let J={taskSupport:"required",...X.execution};if(J.taskSupport==="forbidden")throw Error(`Cannot register task-based tool '${Q}' with taskSupport 'forbidden'. Use registerTool() instead.`);return this._mcpServer._createRegisteredTool(Q,X.title,X.description,X.inputSchema,X.outputSchema,X.annotations,J,X._meta,$)}}class OY{constructor(Q,X){this._registeredResources={},this._registeredResourceTemplates={},this._registeredTools={},this._registeredPrompts={},this._toolHandlersInitialized=!1,this._completionHandlerInitialized=!1,this._resourceHandlersInitialized=!1,this._promptHandlersInitialized=!1,this.server=new VY(Q,X)}get experimental(){if(!this._experimental)this._experimental={tasks:new DY(this)};return this._experimental}async connect(Q){return await this.server.connect(Q)}async close(){await this.server.close()}setToolRequestHandlers(){if(this._toolHandlersInitialized)return;this.server.assertCanSetRequestHandler(U9(wX)),this.server.assertCanSetRequestHandler(U9(YQ)),this.server.registerCapabilities({tools:{listChanged:!0}}),this.server.setRequestHandler(wX,()=>({tools:Object.entries(this._registeredTools).filter(([,Q])=>Q.enabled).map(([Q,X])=>{let $={name:Q,title:X.title,description:X.description,inputSchema:(()=>{let J=QQ(X.inputSchema);return J?J7(J,{strictUnions:!0,pipeStrategy:"input"}):SM})(),annotations:X.annotations,execution:X.execution,_meta:X._meta};if(X.outputSchema){let J=QQ(X.outputSchema);if(J)$.outputSchema=J7(J,{strictUnions:!0,pipeStrategy:"output"})}return $})})),this.server.setRequestHandler(YQ,async(Q,X)=>{try{let $=this._registeredTools[Q.params.name];if(!$)throw new x(h.InvalidParams,`Tool ${Q.params.name} not found`);if(!$.enabled)throw new x(h.InvalidParams,`Tool ${Q.params.name} disabled`);let J=!!Q.params.task,Y=$.execution?.taskSupport,G="createTask"in $.handler;if((Y==="required"||Y==="optional")&&!G)throw new x(h.InternalError,`Tool ${Q.params.name} has taskSupport '${Y}' but was not registered with registerToolTask`);if(Y==="required"&&!J)throw new x(h.MethodNotFound,`Tool ${Q.params.name} requires task augmentation (taskSupport: 'required')`);if(Y==="optional"&&!J&&G)return await this.handleAutomaticTaskPolling($,Q,X);let W=await this.validateToolInput($,Q.params.arguments,Q.params.name),H=await this.executeToolHandler($,W,X);if(J)return H;return await this.validateToolOutput($,H,Q.params.name),H}catch($){if($ instanceof x){if($.code===h.UrlElicitationRequired)throw $}return this.createToolError($ instanceof Error?$.message:String($))}}),this._toolHandlersInitialized=!0}createToolError(Q){return{content:[{type:"text",text:Q}],isError:!0}}async validateToolInput(Q,X,$){if(!Q.inputSchema)return;let Y=QQ(Q.inputSchema)??Q.inputSchema,G=await t8(Y,X);if(!G.success){let W="error"in G?G.error:"Unknown error",H=a8(W);throw new x(h.InvalidParams,`Input validation error: Invalid arguments for tool ${$}: ${H}`)}return G.data}async validateToolOutput(Q,X,$){if(!Q.outputSchema)return;if(!("content"in X))return;if(X.isError)return;if(!X.structuredContent)throw new x(h.InvalidParams,`Output validation error: Tool ${$} has an output schema but no structured content was provided`);let J=QQ(Q.outputSchema),Y=await t8(J,X.structuredContent);if(!Y.success){let G="error"in Y?Y.error:"Unknown error",W=a8(G);throw new x(h.InvalidParams,`Output validation error: Invalid structured content for tool ${$}: ${W}`)}}async executeToolHandler(Q,X,$){let J=Q.handler;if("createTask"in J){if(!$.taskStore)throw Error("No task store provided.");let G={...$,taskStore:$.taskStore};if(Q.inputSchema)return await Promise.resolve(J.createTask(X,G));else return await Promise.resolve(J.createTask(G))}if(Q.inputSchema)return await Promise.resolve(J(X,$));else return await Promise.resolve(J($))}async handleAutomaticTaskPolling(Q,X,$){if(!$.taskStore)throw Error("No task store provided for task-capable tool.");let J=await this.validateToolInput(Q,X.params.arguments,X.params.name),Y=Q.handler,G={...$,taskStore:$.taskStore},W=J?await Promise.resolve(Y.createTask(J,G)):await Promise.resolve(Y.createTask(G)),H=W.task.taskId,z=W.task,K=z.pollInterval??5000;while(z.status!=="completed"&&z.status!=="failed"&&z.status!=="cancelled"){await new Promise((B)=>setTimeout(B,K));let q=await $.taskStore.getTask(H);if(!q)throw new x(h.InternalError,`Task ${H} not found during polling`);z=q}return await $.taskStore.getTaskResult(H)}setCompletionRequestHandler(){if(this._completionHandlerInitialized)return;this.server.assertCanSetRequestHandler(U9(NX)),this.server.registerCapabilities({completions:{}}),this.server.setRequestHandler(NX,async(Q)=>{switch(Q.params.ref.type){case"ref/prompt":return iG(Q),this.handlePromptCompletion(Q,Q.params.ref);case"ref/resource":return nG(Q),this.handleResourceCompletion(Q,Q.params.ref);default:throw new x(h.InvalidParams,`Invalid completion reference: ${Q.params.ref}`)}}),this._completionHandlerInitialized=!0}async handlePromptCompletion(Q,X){let $=this._registeredPrompts[X.name];if(!$)throw new x(h.InvalidParams,`Prompt ${X.name} not found`);if(!$.enabled)throw new x(h.InvalidParams,`Prompt ${X.name} disabled`);if(!$.argsSchema)return S8;let Y=J9($.argsSchema)?.[Q.params.argument.name];if(!UY(Y))return S8;let G=q1(Y);if(!G)return S8;let W=await G(Q.params.argument.value,Q.params.context);return V1(W)}async handleResourceCompletion(Q,X){let $=Object.values(this._registeredResourceTemplates).find((G)=>G.resourceTemplate.uriTemplate.toString()===X.uri);if(!$){if(this._registeredResources[X.uri])return S8;throw new x(h.InvalidParams,`Resource template ${Q.params.ref.uri} not found`)}let J=$.resourceTemplate.completeCallback(Q.params.argument.name);if(!J)return S8;let Y=await J(Q.params.argument.value,Q.params.context);return V1(Y)}setResourceRequestHandlers(){if(this._resourceHandlersInitialized)return;this.server.assertCanSetRequestHandler(U9(AX)),this.server.assertCanSetRequestHandler(U9(DX)),this.server.assertCanSetRequestHandler(U9(FX)),this.server.registerCapabilities({resources:{listChanged:!0}}),this.server.setRequestHandler(AX,async(Q,X)=>{let $=Object.entries(this._registeredResources).filter(([Y,G])=>G.enabled).map(([Y,G])=>({uri:Y,name:G.name,...G.metadata})),J=[];for(let Y of Object.values(this._registeredResourceTemplates)){if(!Y.resourceTemplate.listCallback)continue;let G=await Y.resourceTemplate.listCallback(X);for(let W of G.resources)J.push({...Y.metadata,...W})}return{resources:[...$,...J]}}),this.server.setRequestHandler(DX,async()=>{return{resourceTemplates:Object.entries(this._registeredResourceTemplates).map(([X,$])=>({name:X,uriTemplate:$.resourceTemplate.uriTemplate.toString(),...$.metadata}))}}),this.server.setRequestHandler(FX,async(Q,X)=>{let $=new URL(Q.params.uri),J=this._registeredResources[$.toString()];if(J){if(!J.enabled)throw new x(h.InvalidParams,`Resource ${$} disabled`);return J.readCallback($,X)}for(let Y of Object.values(this._registeredResourceTemplates)){let G=Y.resourceTemplate.uriTemplate.match($.toString());if(G)return Y.readCallback($,G,X)}throw new x(h.InvalidParams,`Resource ${$} not found`)}),this._resourceHandlersInitialized=!0}setPromptRequestHandlers(){if(this._promptHandlersInitialized)return;this.server.assertCanSetRequestHandler(U9(OX)),this.server.assertCanSetRequestHandler(U9(LX)),this.server.registerCapabilities({prompts:{listChanged:!0}}),this.server.setRequestHandler(OX,()=>({prompts:Object.entries(this._registeredPrompts).filter(([,Q])=>Q.enabled).map(([Q,X])=>{return{name:Q,title:X.title,description:X.description,arguments:X.argsSchema?bM(X.argsSchema):void 0}})})),this.server.setRequestHandler(LX,async(Q,X)=>{let $=this._registeredPrompts[Q.params.name];if(!$)throw new x(h.InvalidParams,`Prompt ${Q.params.name} not found`);if(!$.enabled)throw new x(h.InvalidParams,`Prompt ${Q.params.name} disabled`);if($.argsSchema){let J=QQ($.argsSchema),Y=await t8(J,Q.params.arguments);if(!Y.success){let H="error"in Y?Y.error:"Unknown error",z=a8(H);throw new x(h.InvalidParams,`Invalid arguments for prompt ${Q.params.name}: ${z}`)}let G=Y.data,W=$.callback;return await Promise.resolve(W(G,X))}else{let J=$.callback;return await Promise.resolve(J(X))}}),this._promptHandlersInitialized=!0}resource(Q,X,...$){let J;if(typeof $[0]==="object")J=$.shift();let Y=$[0];if(typeof X==="string"){if(this._registeredResources[X])throw Error(`Resource ${X} is already registered`);let G=this._createRegisteredResource(Q,void 0,X,J,Y);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),G}else{if(this._registeredResourceTemplates[Q])throw Error(`Resource template ${Q} is already registered`);let G=this._createRegisteredResourceTemplate(Q,void 0,X,J,Y);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),G}}registerResource(Q,X,$,J){if(typeof X==="string"){if(this._registeredResources[X])throw Error(`Resource ${X} is already registered`);let Y=this._createRegisteredResource(Q,$.title,X,$,J);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),Y}else{if(this._registeredResourceTemplates[Q])throw Error(`Resource template ${Q} is already registered`);let Y=this._createRegisteredResourceTemplate(Q,$.title,X,$,J);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),Y}}_createRegisteredResource(Q,X,$,J,Y){let G={name:Q,title:X,metadata:J,readCallback:Y,enabled:!0,disable:()=>G.update({enabled:!1}),enable:()=>G.update({enabled:!0}),remove:()=>G.update({uri:null}),update:(W)=>{if(typeof W.uri<"u"&&W.uri!==$){if(delete this._registeredResources[$],W.uri)this._registeredResources[W.uri]=G}if(typeof W.name<"u")G.name=W.name;if(typeof W.title<"u")G.title=W.title;if(typeof W.metadata<"u")G.metadata=W.metadata;if(typeof W.callback<"u")G.readCallback=W.callback;if(typeof W.enabled<"u")G.enabled=W.enabled;this.sendResourceListChanged()}};return this._registeredResources[$]=G,G}_createRegisteredResourceTemplate(Q,X,$,J,Y){let G={resourceTemplate:$,title:X,metadata:J,readCallback:Y,enabled:!0,disable:()=>G.update({enabled:!1}),enable:()=>G.update({enabled:!0}),remove:()=>G.update({name:null}),update:(z)=>{if(typeof z.name<"u"&&z.name!==Q){if(delete this._registeredResourceTemplates[Q],z.name)this._registeredResourceTemplates[z.name]=G}if(typeof z.title<"u")G.title=z.title;if(typeof z.template<"u")G.resourceTemplate=z.template;if(typeof z.metadata<"u")G.metadata=z.metadata;if(typeof z.callback<"u")G.readCallback=z.callback;if(typeof z.enabled<"u")G.enabled=z.enabled;this.sendResourceListChanged()}};this._registeredResourceTemplates[Q]=G;let W=$.uriTemplate.variableNames;if(Array.isArray(W)&&W.some((z)=>!!$.completeCallback(z)))this.setCompletionRequestHandler();return G}_createRegisteredPrompt(Q,X,$,J,Y){let G={title:X,description:$,argsSchema:J===void 0?void 0:I9(J),callback:Y,enabled:!0,disable:()=>G.update({enabled:!1}),enable:()=>G.update({enabled:!0}),remove:()=>G.update({name:null}),update:(W)=>{if(typeof W.name<"u"&&W.name!==Q){if(delete this._registeredPrompts[Q],W.name)this._registeredPrompts[W.name]=G}if(typeof W.title<"u")G.title=W.title;if(typeof W.description<"u")G.description=W.description;if(typeof W.argsSchema<"u")G.argsSchema=I9(W.argsSchema);if(typeof W.callback<"u")G.callback=W.callback;if(typeof W.enabled<"u")G.enabled=W.enabled;this.sendPromptListChanged()}};if(this._registeredPrompts[Q]=G,J){if(Object.values(J).some((H)=>{let z=H instanceof $6?H._def?.innerType:H;return UY(z)}))this.setCompletionRequestHandler()}return G}_createRegisteredTool(Q,X,$,J,Y,G,W,H,z){AY(Q);let K={title:X,description:$,inputSchema:B1(J),outputSchema:B1(Y),annotations:G,execution:W,_meta:H,handler:z,enabled:!0,disable:()=>K.update({enabled:!1}),enable:()=>K.update({enabled:!0}),remove:()=>K.update({name:null}),update:(q)=>{if(typeof q.name<"u"&&q.name!==Q){if(typeof q.name==="string")AY(q.name);if(delete this._registeredTools[Q],q.name)this._registeredTools[q.name]=K}if(typeof q.title<"u")K.title=q.title;if(typeof q.description<"u")K.description=q.description;if(typeof q.paramsSchema<"u")K.inputSchema=I9(q.paramsSchema);if(typeof q.outputSchema<"u")K.outputSchema=I9(q.outputSchema);if(typeof q.callback<"u")K.handler=q.callback;if(typeof q.annotations<"u")K.annotations=q.annotations;if(typeof q._meta<"u")K._meta=q._meta;if(typeof q.enabled<"u")K.enabled=q.enabled;this.sendToolListChanged()}};return this._registeredTools[Q]=K,this.setToolRequestHandlers(),this.sendToolListChanged(),K}tool(Q,...X){if(this._registeredTools[Q])throw Error(`Tool ${Q} is already registered`);let $,J,Y,G;if(typeof X[0]==="string")$=X.shift();if(X.length>1){let H=X[0];if(FY(H)){if(J=X.shift(),X.length>1&&typeof X[0]==="object"&&X[0]!==null&&!FY(X[0]))G=X.shift()}else if(typeof H==="object"&&H!==null)G=X.shift()}let W=X[0];return this._createRegisteredTool(Q,void 0,$,J,Y,G,{taskSupport:"forbidden"},void 0,W)}registerTool(Q,X,$){if(this._registeredTools[Q])throw Error(`Tool ${Q} is already registered`);let{title:J,description:Y,inputSchema:G,outputSchema:W,annotations:H,_meta:z}=X;return this._createRegisteredTool(Q,J,Y,G,W,H,{taskSupport:"forbidden"},z,$)}prompt(Q,...X){if(this._registeredPrompts[Q])throw Error(`Prompt ${Q} is already registered`);let $;if(typeof X[0]==="string")$=X.shift();let J;if(X.length>1)J=X.shift();let Y=X[0],G=this._createRegisteredPrompt(Q,void 0,$,J,Y);return this.setPromptRequestHandlers(),this.sendPromptListChanged(),G}registerPrompt(Q,X,$){if(this._registeredPrompts[Q])throw Error(`Prompt ${Q} is already registered`);let{title:J,description:Y,argsSchema:G}=X,W=this._createRegisteredPrompt(Q,J,Y,G,$);return this.setPromptRequestHandlers(),this.sendPromptListChanged(),W}isConnected(){return this.server.transport!==void 0}async sendLoggingMessage(Q,X){return this.server.sendLoggingMessage(Q,X)}sendResourceListChanged(){if(this.isConnected())this.server.sendResourceListChanged()}sendToolListChanged(){if(this.isConnected())this.server.sendToolListChanged()}sendPromptListChanged(){if(this.isConnected())this.server.sendPromptListChanged()}}var SM={type:"object",properties:{}};function U1(Q){return Q!==null&&typeof Q==="object"&&"parse"in Q&&typeof Q.parse==="function"&&"safeParse"in Q&&typeof Q.safeParse==="function"}function PM(Q){return"_def"in Q||"_zod"in Q||U1(Q)}function FY(Q){if(typeof Q!=="object"||Q===null)return!1;if(PM(Q))return!1;if(Object.keys(Q).length===0)return!0;return Object.values(Q).some(U1)}function B1(Q){if(!Q)return;if(FY(Q))return I9(Q);return Q}function bM(Q){let X=J9(Q);if(!X)return[];return Object.entries(X).map(([$,J])=>{let Y=WG(J),G=HG(J);return{name:$,description:Y,required:!G}})}function U9(Q){let $=J9(Q)?.method;if(!$)throw Error("Schema is missing a method literal");let J=s8($);if(typeof J==="string")return J;throw Error("Schema method literal must be a string")}function V1(Q){return{completion:{values:Q.slice(0,100),total:Q.length,hasMore:Q.length>100}}}var S8={completion:{values:[],hasMore:!1}};import D1 from"node:process";class LY{append(Q){this._buffer=this._buffer?Buffer.concat([this._buffer,Q]):Q}readMessage(){if(!this._buffer)return null;let Q=this._buffer.indexOf(`
44
44
  `);if(Q===-1)return null;let X=this._buffer.toString("utf8",0,Q).replace(/\r$/,"");return this._buffer=this._buffer.subarray(Q+1),IM(X)}clear(){this._buffer=void 0}}function IM(Q){return yG.parse(JSON.parse(Q))}function A1(Q){return JSON.stringify(Q)+`
45
- `}class wY{constructor(Q=D1.stdin,X=D1.stdout){this._stdin=Q,this._stdout=X,this._readBuffer=new LY,this._started=!1,this._ondata=($)=>{this._readBuffer.append($),this.processReadBuffer()},this._onerror=($)=>{this.onerror?.($)}}async start(){if(this._started)throw Error("StdioServerTransport already started! If using Server class, note that connect() calls start() automatically.");this._started=!0,this._stdin.on("data",this._ondata),this._stdin.on("error",this._onerror)}processReadBuffer(){while(!0)try{let Q=this._readBuffer.readMessage();if(Q===null)break;this.onmessage?.(Q)}catch(Q){this.onerror?.(Q)}}async close(){if(this._stdin.off("data",this._ondata),this._stdin.off("error",this._onerror),this._stdin.listenerCount("data")===0)this._stdin.pause();this._readBuffer.clear(),this.onclose?.()}send(Q){return new Promise((X)=>{let $=A1(Q);if(this._stdout.write($))X();else this._stdout.once("drain",X)})}}var F1={name:"trueline-mcp",version:"2.5.6",type:"module",description:"Truth-verified file editing for AI coding agents via MCP",license:"Apache-2.0",repository:{type:"git",url:"https://github.com/rjkaes/trueline-mcp.git"},homepage:"https://github.com/rjkaes/trueline-mcp#readme",bugs:{url:"https://github.com/rjkaes/trueline-mcp/issues"},keywords:["mcp","trueline","editing","verification","claude-code","gemini-cli","copilot"],bin:{"trueline-mcp":"./bin/trueline-mcp.js","trueline-hook":"./bin/trueline-hook.js"},files:["dist/","bin/","scripts/resolve-binary.cjs","hooks/","configs/","src/security.js","README.md","INSTALL.md","LICENSE"],scripts:{dev:"bun run src/server.ts",build:"bun build src/server.ts --target=node --minify --outfile dist/server.js",prepublishOnly:"bun run build",typecheck:"bun x tsc --noEmit",test:"bun test",lint:"bunx biome ci .","lint:fix":"bunx biome check --write .",benchmark:"bun run benchmarks/token-benchmark.ts && bun run benchmarks/perf-benchmark.ts"},dependencies:{"@modelcontextprotocol/sdk":"^1.26.0","tree-sitter-wasms":"0.1.13","web-tree-sitter":"0.24.7",zod:"^3.25.0"},devDependencies:{"@biomejs/biome":"2.4.5","@types/node":"^22.19.0","@xmldom/xmldom":"^0.8.11","bun-types":"^1.3.10",lefthook:"^2.1.2",typescript:"^5.7.0"}};function I0(Q){return{content:[{type:"text",text:Q}]}}function Y0(Q){return{content:[{type:"text",text:Q}],isError:!0}}import{execSync as WR}from"node:child_process";import{readFile as HR}from"node:fs/promises";import{extname as zR,relative as p1,resolve as KR}from"node:path";var M1={grammar:"typescript",outline:new Set(["function_declaration","class_declaration","interface_declaration","type_alias_declaration","enum_declaration","lexical_declaration","variable_declaration","export_statement","method_definition","public_field_definition"]),skip:new Set(["import_statement"]),recurse:new Set(["class_body"]),topLevelOnly:new Set(["expression_statement"])},CM={...M1,grammar:"tsx"},K4={grammar:"javascript",outline:new Set(["function_declaration","class_declaration","lexical_declaration","variable_declaration","export_statement","expression_statement","method_definition","field_definition"]),skip:new Set(["import_statement"]),recurse:new Set(["class_body"])},O1={grammar:"python",outline:new Set(["function_definition","class_definition","decorated_definition","expression_statement"]),skip:new Set(["import_statement","import_from_statement"]),recurse:new Set(["block"]),whitespaceMode:"preserve-indent"},TM={grammar:"go",outline:new Set(["function_declaration","method_declaration","type_declaration","const_declaration","var_declaration"]),skip:new Set(["package_clause","import_declaration"]),recurse:new Set([])},kM={grammar:"rust",outline:new Set(["function_item","struct_item","enum_item","trait_item","impl_item","const_item","static_item","mod_item","type_item","macro_definition"]),skip:new Set(["use_declaration"]),recurse:new Set(["declaration_list"])},vM={grammar:"java",outline:new Set(["class_declaration","interface_declaration","enum_declaration","method_declaration","constructor_declaration","field_declaration"]),skip:new Set(["package_declaration","import_declaration"]),recurse:new Set(["class_body"])},_M={grammar:"ruby",outline:new Set(["method","class","module","assignment","call"]),skip:new Set([]),recurse:new Set(["body_statement"])},P8={grammar:"cpp",outline:new Set(["function_definition","class_specifier","struct_specifier","enum_specifier","namespace_definition","declaration","template_declaration"]),skip:new Set(["preproc_include"]),recurse:new Set(["declaration_list","field_declaration_list"])},L1={grammar:"c",outline:new Set(["function_definition","struct_specifier","enum_specifier","declaration","type_definition"]),skip:new Set(["preproc_include"]),recurse:new Set([])},xM={grammar:"c_sharp",outline:new Set(["class_declaration","interface_declaration","struct_declaration","enum_declaration","method_declaration","constructor_declaration","property_declaration","namespace_declaration"]),skip:new Set(["using_directive"]),recurse:new Set(["declaration_list"])},w1={grammar:"kotlin",outline:new Set(["function_declaration","class_declaration","object_declaration","property_declaration"]),skip:new Set(["import_list","package_header"]),recurse:new Set(["class_body"])},gM={grammar:"swift",outline:new Set(["function_declaration","class_declaration","struct_declaration","enum_declaration","protocol_declaration","extension_declaration","property_declaration"]),skip:new Set(["import_declaration"]),recurse:new Set(["class_body"])},yM={grammar:"php",outline:new Set(["function_definition","class_declaration","interface_declaration","trait_declaration","method_declaration","property_declaration"]),skip:new Set(["namespace_use_declaration"]),recurse:new Set(["declaration_list"])},j1={grammar:"scala",outline:new Set(["function_definition","class_definition","object_definition","trait_definition","val_definition","var_definition","type_definition"]),skip:new Set(["import_declaration"]),recurse:new Set(["template_body"])},N1={grammar:"elixir",outline:new Set(["call"]),skip:new Set([]),recurse:new Set([])},hM={grammar:"lua",outline:new Set(["function_declaration","local_function","variable_declaration","local_variable_declaration","assignment_statement"]),skip:new Set([]),recurse:new Set([])},fM={grammar:"dart",outline:new Set(["function_signature","class_definition","enum_declaration","mixin_declaration","extension_declaration","type_alias"]),skip:new Set(["import_or_export"]),recurse:new Set(["class_body"])},uM={grammar:"zig",outline:new Set(["TopLevelDecl","VarDecl","FnProto"]),skip:new Set([]),recurse:new Set([])},E1={grammar:"bash",outline:new Set(["function_definition","variable_assignment"]),skip:new Set([]),recurse:new Set([])},mM={".ts":M1,".tsx":CM,".js":K4,".jsx":K4,".mjs":K4,".cjs":K4,".py":O1,".pyi":O1,".go":TM,".rs":kM,".java":vM,".c":L1,".h":L1,".cpp":P8,".cc":P8,".cxx":P8,".hpp":P8,".hh":P8,".cs":xM,".rb":_M,".php":yM,".kt":w1,".kts":w1,".swift":gM,".scala":j1,".sc":j1,".ex":N1,".exs":N1,".lua":hM,".dart":fM,".zig":uM,".sh":E1,".bash":E1};function LQ(Q){return mM[Q]}var B4=N4(S1(),1);import{createRequire as pM}from"node:module";import{resolve as I1,dirname as Z1}from"node:path";var C1=pM(import.meta.url),P1=!1,b1=new Map;function iM(){let Q=C1.resolve("web-tree-sitter/package.json");return I1(Z1(Q),"tree-sitter.wasm")}async function T1(){if(P1)return;let Q=new Promise((X,$)=>setTimeout(()=>$(Error("tree-sitter WASM init timed out after 10 s")),1e4));await Promise.race([B4.default.init({locateFile:()=>iM()}),Q]),P1=!0}function nM(Q){let X=C1.resolve("tree-sitter-wasms/package.json");return I1(Z1(X),"out",`tree-sitter-${Q}.wasm`)}async function dM(Q){let X=b1.get(Q);if(X)return X;await T1();let $=await B4.default.Language.load(nM(Q));return b1.set(Q,$),$}async function k1(Q){await T1();let X=new B4.default,$=await dM(Q);return X.setLanguage($),X}async function V4(Q,X,$=1/0){let Y=(await k1(X.grammar)).parse(Q),G=Q.split(`
45
+ `}class wY{constructor(Q=D1.stdin,X=D1.stdout){this._stdin=Q,this._stdout=X,this._readBuffer=new LY,this._started=!1,this._ondata=($)=>{this._readBuffer.append($),this.processReadBuffer()},this._onerror=($)=>{this.onerror?.($)}}async start(){if(this._started)throw Error("StdioServerTransport already started! If using Server class, note that connect() calls start() automatically.");this._started=!0,this._stdin.on("data",this._ondata),this._stdin.on("error",this._onerror)}processReadBuffer(){while(!0)try{let Q=this._readBuffer.readMessage();if(Q===null)break;this.onmessage?.(Q)}catch(Q){this.onerror?.(Q)}}async close(){if(this._stdin.off("data",this._ondata),this._stdin.off("error",this._onerror),this._stdin.listenerCount("data")===0)this._stdin.pause();this._readBuffer.clear(),this.onclose?.()}send(Q){return new Promise((X)=>{let $=A1(Q);if(this._stdout.write($))X();else this._stdout.once("drain",X)})}}var F1={name:"trueline-mcp",version:"2.5.8",type:"module",description:"Truth-verified file editing for AI coding agents via MCP",license:"Apache-2.0",repository:{type:"git",url:"https://github.com/rjkaes/trueline-mcp.git"},homepage:"https://github.com/rjkaes/trueline-mcp#readme",bugs:{url:"https://github.com/rjkaes/trueline-mcp/issues"},keywords:["mcp","trueline","editing","verification","claude-code","gemini-cli","copilot"],bin:{"trueline-mcp":"./bin/trueline-mcp.js","trueline-hook":"./bin/trueline-hook.js"},files:["dist/","bin/","scripts/resolve-binary.cjs","hooks/","configs/","src/security.js","README.md","INSTALL.md","LICENSE"],scripts:{dev:"bun run src/server.ts",build:"bun build src/server.ts --target=node --minify --outfile dist/server.js",prepublishOnly:"bun run build",typecheck:"bun x tsc --noEmit",test:"bun test",lint:"bunx biome ci .","lint:fix":"bunx biome check --write .",benchmark:"bun run benchmarks/token-benchmark.ts && bun run benchmarks/perf-benchmark.ts"},dependencies:{"@modelcontextprotocol/sdk":"^1.26.0","tree-sitter-wasms":"0.1.13","web-tree-sitter":"0.24.7",zod:"^3.25.0"},devDependencies:{"@biomejs/biome":"2.4.5","@types/node":"^22.19.0","@xmldom/xmldom":"^0.8.11","bun-types":"^1.3.10",lefthook:"^2.1.2",typescript:"^5.7.0"}};function I0(Q){return{content:[{type:"text",text:Q}]}}function Y0(Q){return{content:[{type:"text",text:Q}],isError:!0}}import{execSync as WR}from"node:child_process";import{readFile as HR}from"node:fs/promises";import{extname as zR,relative as p1,resolve as KR}from"node:path";var M1={grammar:"typescript",outline:new Set(["function_declaration","class_declaration","interface_declaration","type_alias_declaration","enum_declaration","lexical_declaration","variable_declaration","export_statement","method_definition","public_field_definition"]),skip:new Set(["import_statement"]),recurse:new Set(["class_body"]),topLevelOnly:new Set(["expression_statement"])},CM={...M1,grammar:"tsx"},K4={grammar:"javascript",outline:new Set(["function_declaration","class_declaration","lexical_declaration","variable_declaration","export_statement","expression_statement","method_definition","field_definition"]),skip:new Set(["import_statement"]),recurse:new Set(["class_body"])},O1={grammar:"python",outline:new Set(["function_definition","class_definition","decorated_definition","expression_statement"]),skip:new Set(["import_statement","import_from_statement"]),recurse:new Set(["block"]),whitespaceMode:"preserve-indent"},TM={grammar:"go",outline:new Set(["function_declaration","method_declaration","type_declaration","const_declaration","var_declaration"]),skip:new Set(["package_clause","import_declaration"]),recurse:new Set([])},kM={grammar:"rust",outline:new Set(["function_item","struct_item","enum_item","trait_item","impl_item","const_item","static_item","mod_item","type_item","macro_definition"]),skip:new Set(["use_declaration"]),recurse:new Set(["declaration_list"])},vM={grammar:"java",outline:new Set(["class_declaration","interface_declaration","enum_declaration","method_declaration","constructor_declaration","field_declaration"]),skip:new Set(["package_declaration","import_declaration"]),recurse:new Set(["class_body"])},_M={grammar:"ruby",outline:new Set(["method","class","module","assignment","call"]),skip:new Set([]),recurse:new Set(["body_statement"])},P8={grammar:"cpp",outline:new Set(["function_definition","class_specifier","struct_specifier","enum_specifier","namespace_definition","declaration","template_declaration"]),skip:new Set(["preproc_include"]),recurse:new Set(["declaration_list","field_declaration_list"])},L1={grammar:"c",outline:new Set(["function_definition","struct_specifier","enum_specifier","declaration","type_definition"]),skip:new Set(["preproc_include"]),recurse:new Set([])},xM={grammar:"c_sharp",outline:new Set(["class_declaration","interface_declaration","struct_declaration","enum_declaration","method_declaration","constructor_declaration","property_declaration","namespace_declaration"]),skip:new Set(["using_directive"]),recurse:new Set(["declaration_list"])},w1={grammar:"kotlin",outline:new Set(["function_declaration","class_declaration","object_declaration","property_declaration"]),skip:new Set(["import_list","package_header"]),recurse:new Set(["class_body"])},gM={grammar:"swift",outline:new Set(["function_declaration","class_declaration","struct_declaration","enum_declaration","protocol_declaration","extension_declaration","property_declaration"]),skip:new Set(["import_declaration"]),recurse:new Set(["class_body"])},yM={grammar:"php",outline:new Set(["function_definition","class_declaration","interface_declaration","trait_declaration","method_declaration","property_declaration"]),skip:new Set(["namespace_use_declaration"]),recurse:new Set(["declaration_list"])},j1={grammar:"scala",outline:new Set(["function_definition","class_definition","object_definition","trait_definition","val_definition","var_definition","type_definition"]),skip:new Set(["import_declaration"]),recurse:new Set(["template_body"])},N1={grammar:"elixir",outline:new Set(["call"]),skip:new Set([]),recurse:new Set([])},hM={grammar:"lua",outline:new Set(["function_declaration","local_function","variable_declaration","local_variable_declaration","assignment_statement"]),skip:new Set([]),recurse:new Set([])},fM={grammar:"dart",outline:new Set(["function_signature","class_definition","enum_declaration","mixin_declaration","extension_declaration","type_alias"]),skip:new Set(["import_or_export"]),recurse:new Set(["class_body"])},uM={grammar:"zig",outline:new Set(["TopLevelDecl","VarDecl","FnProto"]),skip:new Set([]),recurse:new Set([])},E1={grammar:"bash",outline:new Set(["function_definition","variable_assignment"]),skip:new Set([]),recurse:new Set([])},mM={".ts":M1,".tsx":CM,".js":K4,".jsx":K4,".mjs":K4,".cjs":K4,".py":O1,".pyi":O1,".go":TM,".rs":kM,".java":vM,".c":L1,".h":L1,".cpp":P8,".cc":P8,".cxx":P8,".hpp":P8,".hh":P8,".cs":xM,".rb":_M,".php":yM,".kt":w1,".kts":w1,".swift":gM,".scala":j1,".sc":j1,".ex":N1,".exs":N1,".lua":hM,".dart":fM,".zig":uM,".sh":E1,".bash":E1};function LQ(Q){return mM[Q]}var B4=N4(S1(),1);import{createRequire as pM}from"node:module";import{resolve as I1,dirname as Z1}from"node:path";var C1=pM(import.meta.url),P1=!1,b1=new Map;function iM(){let Q=C1.resolve("web-tree-sitter/package.json");return I1(Z1(Q),"tree-sitter.wasm")}async function T1(){if(P1)return;let Q=new Promise((X,$)=>setTimeout(()=>$(Error("tree-sitter WASM init timed out after 10 s")),1e4));await Promise.race([B4.default.init({locateFile:()=>iM()}),Q]),P1=!0}function nM(Q){let X=C1.resolve("tree-sitter-wasms/package.json");return I1(Z1(X),"out",`tree-sitter-${Q}.wasm`)}async function dM(Q){let X=b1.get(Q);if(X)return X;await T1();let $=await B4.default.Language.load(nM(Q));return b1.set(Q,$),$}async function k1(Q){await T1();let X=new B4.default,$=await dM(Q);return X.setLanguage($),X}async function V4(Q,X,$=1/0){let Y=(await k1(X.grammar)).parse(Q),G=Q.split(`
46
46
  `),W=[];function H(F){let L=F.startPosition.row,O=F.endPosition.row,w=G[L]?.trimEnd()??"";if(L===O||w.includes("{"))return w.length>200?`${w.slice(0,197)}...`:w;let E=[w.trimEnd()];for(let I=L+1;I<=Math.min(O,L+20);I++){let f=(G[I]??"").trimEnd();if(E.push(f.trim()),f.includes("{"))break}let S=E.join(" ").replace(/\s+/g," ").replace(/\(\s+/g,"(").replace(/,\s*\)/g,")").replace(/\s*\{\s*$/,"");if(S.length>200)S=`${S.slice(0,197)}...`;return S}let z=-1,K=-1,q=0,B="";function A(){if(q===0)return;let F=q===1?`1 ${B}`:`${q} ${B}s`;W.push({startLine:z,endLine:K,depth:0,nodeType:"_skipped",text:`(${F})`}),z=-1,K=-1,q=0,B=""}function U(F){let L=F.startPosition.row+1,O=F.endPosition.row+1,w=F.type.replace(/_/g," ").replace(/ statement$/,"").replace(/ declaration$/,"");if(q===0||w===B){if(q===0)z=L,B=w;K=O,q++}else A(),z=L,K=O,q=1,B=w}function D(F,L,O){if(O&&X.skip.has(F.type)){U(F);return}if(X.skip.has(F.type))return;let w=X.outline.has(F.type),E=X.topLevelOnly?.has(F.type)??!1;if(E&&!O)return;if(w||E&&O){if(O)A();if(W.push({startLine:F.startPosition.row+1,endLine:F.endPosition.row+1,depth:L,nodeType:F.type,text:H(F)}),L+1<=$)for(let S of F.children){if(!S.isNamed)continue;if(X.recurse.has(S.type))for(let I of S.children){if(!I.isNamed)continue;D(I,L+1,!1)}}return}}for(let F of Y.rootNode.children)D(F,0,!0);return A(),W}function v1(Q,X){let $=[];for(let J of Q){let Y=" ".repeat(J.depth);$.push(`${Y}${J.startLine}-${J.endLine}: ${J.text}`)}return $.push(""),$.push(`(${Q.length} symbols, ${X} source lines)`),$.join(`
47
47
  `)}var U6=2166136261;var U4="0-0:00000000";function x1(Q){let X=2166136261;for(let $=0;$<Q.length;$++){let J=Q.charCodeAt($);if(J>=55296&&J<=56319&&$+1<Q.length){let Y=Q.charCodeAt($+1);if(Y>=56320&&Y<=57343)J=(J-55296<<10)+(Y-56320)+65536,$++}if(J<128)X=Math.imul(X^J,16777619)>>>0;else if(J<2048)X=Math.imul(X^(192|J>>6),16777619)>>>0,X=Math.imul(X^(128|J&63),16777619)>>>0;else if(J<65536)X=Math.imul(X^(224|J>>12),16777619)>>>0,X=Math.imul(X^(128|J>>6&63),16777619)>>>0,X=Math.imul(X^(128|J&63),16777619)>>>0;else X=Math.imul(X^(240|J>>18),16777619)>>>0,X=Math.imul(X^(128|J>>12&63),16777619)>>>0,X=Math.imul(X^(128|J>>6&63),16777619)>>>0,X=Math.imul(X^(128|J&63),16777619)>>>0}return X>>>0}function d6(Q,X,$){let J=2166136261;for(let Y=X;Y<$;Y++)J=Math.imul(J^Q[Y],16777619)>>>0;return J}function C6(Q,X){return Q=Math.imul(Q^X&255,16777619)>>>0,Q=Math.imul(Q^X>>>8&255,16777619)>>>0,Q=Math.imul(Q^X>>>16&255,16777619)>>>0,Q=Math.imul(Q^X>>>24&255,16777619)>>>0,Q}function T6(Q,X,$){return`${Q}-${X}:${$.toString(16).padStart(8,"0")}`}var _1="abcdefghijklmnopqrstuvwxyz234567",oM=(()=>{let Q=Array(1024);for(let X=0;X<32;X++)for(let $=0;$<32;$++)Q[X*32+$]=_1[X]+_1[$];return Q})();function jQ(Q){return oM[(Q&31)<<5|Q>>>8&31]}function rM(Q,X="collapse"){let $=Q.split(`
48
48
  `);if(X==="preserve-indent")return $.map((J)=>J.replace(/\s+$/,"").replace(/(\S)\s{2,}(\S)/g,"$1 $2")).join(`
@@ -51,7 +51,7 @@ Set the \`cycles\` parameter to \`"ref"\` to resolve cyclical schemas with defs.
51
51
  `))X=C6(X,x1($));return X}async function NY(Q,X){let $=LQ(X.startsWith(".")?X:`.${X}`);if(!$)return[];let J=await V4(Q,$),Y=Q.split(`
52
52
  `),G=$.whitespaceMode??"collapse";return J.filter((W)=>W.nodeType!=="_skipped").map((W)=>{let H=Y.slice(W.startLine-1,W.endLine),z=H.join(`
53
53
  `),K=H.length>1?H.slice(1):H,q=rM(K.join(`
54
- `),G);return{name:aM(W.text,W.nodeType),signature:W.text,bodyHash:tM(q),nodeType:W.nodeType,bodyText:z}})}function aM(Q,X){let $=Q.match(/(?:function|class|interface|type|enum|const|let|var|def|fn|func|fun|pub\s+fn|async\s+function)\s+(\w+)/);if($)return $[1];let J=Q.match(/^\s*(?:(?:public|private|protected|static|async|export|abstract)\s+)*(\w+)\s*[(<]/);if(J)return J[1];let Y=Q.match(/(\w+)/);return Y?Y[1]:Q.slice(0,40)}function g1(Q,X){let $={added:[],removed:[],renamed:[],signatureChanged:[],logicChanged:[]},J=new Map(Q.map((B)=>[B.name,B])),Y=new Map(X.map((B)=>[B.name,B])),G=[],W=[],H=[];for(let B of Q){let A=Y.get(B.name);if(A)G.push({old:B,new:A});else W.push(B)}for(let B of X)if(!J.has(B.name))H.push(B);let z=new Map;for(let B of W){let A=z.get(B.bodyHash)??[];A.push(B),z.set(B.bodyHash,A)}let K=new Set,q=new Set;for(let B of H){let A=z.get(B.bodyHash);if(A&&A.length>0){let U=A.shift();$.renamed.push({oldName:U.name,newName:B.name,signature:B.signature}),K.add(U.name),q.add(B.name)}}for(let B of W)if(!K.has(B.name))$.removed.push(B);for(let B of H)if(!q.has(B.name))$.added.push(B);for(let{old:B,new:A}of G)if(B.signature!==A.signature&&B.bodyHash!==A.bodyHash)$.signatureChanged.push({name:B.name,oldSig:B.signature,newSig:A.signature}),$.logicChanged.push({name:B.name,signature:A.signature,oldBody:B.bodyText,newBody:A.bodyText});else if(B.signature!==A.signature)$.signatureChanged.push({name:B.name,oldSig:B.signature,newSig:A.signature});else if(B.bodyHash!==A.bodyHash)$.logicChanged.push({name:B.name,signature:A.signature,oldBody:B.bodyText,newBody:A.bodyText});return $}import{realpath as PY,stat as $R}from"node:fs/promises";import{resolve as JR,sep as YR}from"node:path";var MY=/^\d+$/;function EY(Q){let X=Q.indexOf(":");if(X===-1)throw Error(`Invalid line:hash reference "${Q}" — missing colon`);let $=Q.slice(0,X),J=Q.slice(X+1);if(!MY.test($))throw Error(`Invalid line number in "${Q}" — must be a non-negative integer`);let Y=Number($);if(Y===0&&J!=="")throw Error(`Invalid line:hash reference "${Q}" — line 0 must have empty hash`);if(Y>0&&!/^[a-z2-7]{2}$/.test(J))throw Error(`Invalid hash in "${Q}" — must be exactly 2 characters from a-z, 2-7`);return{line:Y,hash:J}}function y1(Q){let X=!1,$=Q;if($.startsWith("+"))X=!0,$=$.slice(1);let J=$.indexOf("..");if(X&&J!==-1)throw Error(`Invalid range "${Q}" — insert-after (+) cannot be used with a multi-line range`);if(J===-1){let W=EY($);return{start:W,end:{...W},insertAfter:X}}let Y=EY($.slice(0,J)),G=EY($.slice(J+2));if(Y.line>G.line)throw Error(`Invalid range "${Q}" — start line ${Y.line} must be ≤ end line ${G.line}`);return{start:Y,end:G,insertAfter:X}}function A4(Q){let X=Q.indexOf("-");if(X===-1)throw Error(`Invalid checksum "${Q}" — expected format "startLine-endLine:hex"`);let $=Q.indexOf(":",X);if($===-1)throw Error(`Invalid checksum "${Q}" — expected format "startLine-endLine:hex"`);let J=Q.slice(0,X),Y=Q.slice(X+1,$),G=Q.slice($+1);if(!MY.test(J))throw Error(`Invalid checksum "${Q}" — start line must be a decimal integer`);if(!MY.test(Y))throw Error(`Invalid checksum "${Q}" — end line must be a decimal integer`);if(!/^[0-9a-f]{8}$/.test(G))throw Error(`Invalid checksum "${Q}" — hash must be 8 hex chars, got "${G}"`);let W=Number(J),H=Number(Y);if(W===0&&H!==0)throw Error(`Invalid checksum "${Q}" — startLine 0 requires endLine 0`);if(W>H)throw Error(`Invalid checksum "${Q}" — start ${W} must be ≤ end ${H}`);if(W===0&&H===0&&G!=="00000000")throw Error(`Invalid checksum "${Q}" — empty-file sentinel must have hash 00000000`);return{startLine:W,endLine:H,hash:G}}function h1(Q){if(!Q||Q.length===0)return[{start:1,end:1/0}];let X=Q.map(($)=>{let J=$.start??1,Y=$.end??1/0;if(J<1)throw Error(`Invalid range: start ${J} must be >= 1`);if(J>Y)throw Error(`Invalid range: start ${J} must be <= end ${Y}`);return{start:J,end:Y}});X.sort(($,J)=>$.start-J.start);for(let $=1;$<X.length;$++){let J=X[$-1],Y=X[$];if(J.end===1/0||Y.start<=J.end+1)J.end=Math.max(J.end,Y.end),X.splice($,1),$--}return X}import{readFile as sM,stat as eM}from"node:fs/promises";import{homedir as QR}from"node:os";import{resolve as RY}from"node:path";var SY=new Map,f1=new Map;function XR(Q){let X=Q.match(/^(\w+)\((.+)\)$/);return X?{tool:X[1],glob:X[2]}:null}function u1(Q,X=!1){let $=`${Q}:${X}`,J=f1.get($);if(J)return J;Q=Q.replace(/(\*\*\/)+/g,"**/");let Y=Q.replace(/\*\*\/|\*\*|\*|\?|[^*?]+/g,(W,H)=>{let z=H===0||Q[H-1]==="/";switch(W){case"**/":return z?"(.*/)?":"[^/]*/";case"**":return z?".*":"[^/]*";case"*":return"[^/]*";case"?":return"[^/]";default:return W.replace(/[.+^${}()|[\]\\/-]/g,"\\$&")}}),G=new RegExp(`^${Y}$`,X?"i":"");return f1.set($,G),G}async function m1(Q,X,$){let J=async(W)=>{let H=`${W}:${Q}`,z;try{z=(await eM(W)).mtimeMs}catch{return null}let K=SY.get(H);if(K&&K.mtime===z)return K.globs;let q;try{q=JSON.parse(await sM(W,"utf-8"))}catch{return SY.set(H,{mtime:z,globs:null}),null}let B=typeof q==="object"&&q!==null?q:void 0,U=(typeof B?.permissions==="object"&&B.permissions!==null?B.permissions:void 0)?.deny,D=[];if(Array.isArray(U))for(let F of U){if(typeof F!=="string")continue;let L=XR(F);if(L?.tool===Q)D.push(L.glob)}return SY.set(H,{mtime:z,globs:D}),D},Y=[];if(X)Y.push(RY(X,".claude","settings.local.json")),Y.push(RY(X,".claude","settings.json"));return Y.push($??RY(QR(),".claude","settings.json")),(await Promise.all(Y.map(J))).filter((W)=>W!==null)}function l1(Q,X,$=process.platform==="win32"){let J=Q.replace(/\\/g,"/"),Y=J.split("/").pop()??J,G=(H)=>{let z=u1(H,$);if(z.test(J))return!0;if(!H.includes("/"))return z.test(Y);if(!H.startsWith("/")&&!H.startsWith("*"))return u1(`**/${H}`,$).test(J);return!1},W=X.flat().find(G);return W?{denied:!0,matchedPattern:W}:{denied:!1}}async function A6(Q,X,$,J=[]){let Y=Q.startsWith("/")?Q:JR($??process.cwd(),Q),G;try{G=await PY(Y)}catch{return{ok:!1,error:Y0(`Error reading file: "${Q}" not found`)}}let W=await $R(G);if(!W.isFile())return{ok:!1,error:Y0(`"${Q}" is not a regular file`)};let H;try{H=await PY($?$:process.cwd())}catch{return{ok:!1,error:Y0("Project directory not found or inaccessible")}}let z=await Promise.all(J.map((D)=>PY(D).catch(()=>D)));if(![H,...z].some((D)=>G===D||G.startsWith(D+YR)))return{ok:!1,error:Y0(`Access denied: "${Q}" is outside the project directory`)};let B=await m1(X,$),{denied:A,matchedPattern:U}=l1(G,B);if(A)return{ok:!1,error:Y0(`Access denied: "${Q}" matched deny pattern "${U}"`)};return{ok:!0,resolvedPath:G,size:W.size,mtimeMs:W.mtimeMs}}function c1(Q){let X=[],$=new Map;for(let W of Q){let H=A4(W.checksum);$.set(W.checksum,H);let z=y1(W.range);if(z.start.line===0&&!z.insertAfter)return{ok:!1,error:Y0("range starting at line 0 requires insert-after (use +0: prefix)")};if(z.start.line>0){if(H.startLine>z.start.line||H.endLine<z.end.line)return{ok:!1,error:Y0(`Checksum range ${H.startLine}-${H.endLine} does not cover edit range ${z.start.line}-${z.end.line}. Re-read with trueline_read to get a checksum covering the target lines.`)}}X.push({startLine:z.start.line,endLine:z.end.line,content:W.content===""?[]:W.content.split(`
54
+ `),G);return{name:aM(W.text,W.nodeType),signature:W.text,bodyHash:tM(q),nodeType:W.nodeType,bodyText:z}})}function aM(Q,X){let $=Q.match(/(?:function|class|interface|type|enum|const|let|var|def|fn|func|fun|pub\s+fn|async\s+function)\s+(\w+)/);if($)return $[1];let J=Q.match(/^\s*(?:(?:public|private|protected|static|async|export|abstract)\s+)*(\w+)\s*[(<]/);if(J)return J[1];let Y=Q.match(/(\w+)/);return Y?Y[1]:Q.slice(0,40)}function g1(Q,X){let $={added:[],removed:[],renamed:[],signatureChanged:[],logicChanged:[]},J=new Map(Q.map((B)=>[B.name,B])),Y=new Map(X.map((B)=>[B.name,B])),G=[],W=[],H=[];for(let B of Q){let A=Y.get(B.name);if(A)G.push({old:B,new:A});else W.push(B)}for(let B of X)if(!J.has(B.name))H.push(B);let z=new Map;for(let B of W){let A=z.get(B.bodyHash)??[];A.push(B),z.set(B.bodyHash,A)}let K=new Set,q=new Set;for(let B of H){let A=z.get(B.bodyHash);if(A&&A.length>0){let U=A.shift();$.renamed.push({oldName:U.name,newName:B.name,signature:B.signature}),K.add(U.name),q.add(B.name)}}for(let B of W)if(!K.has(B.name))$.removed.push(B);for(let B of H)if(!q.has(B.name))$.added.push(B);for(let{old:B,new:A}of G)if(B.signature!==A.signature&&B.bodyHash!==A.bodyHash)$.signatureChanged.push({name:B.name,oldSig:B.signature,newSig:A.signature}),$.logicChanged.push({name:B.name,signature:A.signature,oldBody:B.bodyText,newBody:A.bodyText});else if(B.signature!==A.signature)$.signatureChanged.push({name:B.name,oldSig:B.signature,newSig:A.signature});else if(B.bodyHash!==A.bodyHash)$.logicChanged.push({name:B.name,signature:A.signature,oldBody:B.bodyText,newBody:A.bodyText});return $}import{realpath as PY,stat as $R}from"node:fs/promises";import{resolve as JR,sep as YR}from"node:path";var MY=/^\d+$/;function EY(Q){let X=Q.indexOf(":");if(X===-1)throw Error(`Invalid line:hash reference "${Q}" — missing colon`);let $=Q.slice(0,X),J=Q.slice(X+1);if(!MY.test($))throw Error(`Invalid line number in "${Q}" — must be a non-negative integer`);let Y=Number($);if(Y===0&&J!=="")throw Error(`Invalid line:hash reference "${Q}" — line 0 must have empty hash`);if(Y>0&&!/^[a-z2-7]{2}$/.test(J))throw Error(`Invalid hash in "${Q}" — must be exactly 2 characters from a-z, 2-7`);return{line:Y,hash:J}}function y1(Q){let X=!1,$=Q;if($.startsWith("+"))X=!0,$=$.slice(1);let J=$.indexOf("-");if(X&&J!==-1)throw Error(`Invalid range "${Q}" — insert-after (+) cannot be used with a multi-line range`);if(J===-1){let W=EY($);return{start:W,end:{...W},insertAfter:X}}let Y=EY($.slice(0,J)),G=EY($.slice(J+1));if(Y.line>G.line)throw Error(`Invalid range "${Q}" — start line ${Y.line} must be ≤ end line ${G.line}`);return{start:Y,end:G,insertAfter:X}}function A4(Q){let X=Q.indexOf("-");if(X===-1)throw Error(`Invalid checksum "${Q}" — expected format "startLine-endLine:hex"`);let $=Q.indexOf(":",X);if($===-1)throw Error(`Invalid checksum "${Q}" — expected format "startLine-endLine:hex"`);let J=Q.slice(0,X),Y=Q.slice(X+1,$),G=Q.slice($+1);if(!MY.test(J))throw Error(`Invalid checksum "${Q}" — start line must be a decimal integer`);if(!MY.test(Y))throw Error(`Invalid checksum "${Q}" — end line must be a decimal integer`);if(!/^[0-9a-f]{8}$/.test(G))throw Error(`Invalid checksum "${Q}" — hash must be 8 hex chars, got "${G}"`);let W=Number(J),H=Number(Y);if(W===0&&H!==0)throw Error(`Invalid checksum "${Q}" — startLine 0 requires endLine 0`);if(W>H)throw Error(`Invalid checksum "${Q}" — start ${W} must be ≤ end ${H}`);if(W===0&&H===0&&G!=="00000000")throw Error(`Invalid checksum "${Q}" — empty-file sentinel must have hash 00000000`);return{startLine:W,endLine:H,hash:G}}function h1(Q){if(!Q||Q.length===0)return[{start:1,end:1/0}];let X=Q.map(($)=>{let J=$.start??1,Y=$.end??1/0;if(J<1)throw Error(`Invalid range: start ${J} must be >= 1`);if(J>Y)throw Error(`Invalid range: start ${J} must be <= end ${Y}`);return{start:J,end:Y}});X.sort(($,J)=>$.start-J.start);for(let $=1;$<X.length;$++){let J=X[$-1],Y=X[$];if(J.end===1/0||Y.start<=J.end+1)J.end=Math.max(J.end,Y.end),X.splice($,1),$--}return X}import{readFile as sM,stat as eM}from"node:fs/promises";import{homedir as QR}from"node:os";import{resolve as RY}from"node:path";var SY=new Map,f1=new Map;function XR(Q){let X=Q.match(/^(\w+)\((.+)\)$/);return X?{tool:X[1],glob:X[2]}:null}function u1(Q,X=!1){let $=`${Q}:${X}`,J=f1.get($);if(J)return J;Q=Q.replace(/(\*\*\/)+/g,"**/");let Y=Q.replace(/\*\*\/|\*\*|\*|\?|[^*?]+/g,(W,H)=>{let z=H===0||Q[H-1]==="/";switch(W){case"**/":return z?"(.*/)?":"[^/]*/";case"**":return z?".*":"[^/]*";case"*":return"[^/]*";case"?":return"[^/]";default:return W.replace(/[.+^${}()|[\]\\/-]/g,"\\$&")}}),G=new RegExp(`^${Y}$`,X?"i":"");return f1.set($,G),G}async function m1(Q,X,$){let J=async(W)=>{let H=`${W}:${Q}`,z;try{z=(await eM(W)).mtimeMs}catch{return null}let K=SY.get(H);if(K&&K.mtime===z)return K.globs;let q;try{q=JSON.parse(await sM(W,"utf-8"))}catch{return SY.set(H,{mtime:z,globs:null}),null}let B=typeof q==="object"&&q!==null?q:void 0,U=(typeof B?.permissions==="object"&&B.permissions!==null?B.permissions:void 0)?.deny,D=[];if(Array.isArray(U))for(let F of U){if(typeof F!=="string")continue;let L=XR(F);if(L?.tool===Q)D.push(L.glob)}return SY.set(H,{mtime:z,globs:D}),D},Y=[];if(X)Y.push(RY(X,".claude","settings.local.json")),Y.push(RY(X,".claude","settings.json"));return Y.push($??RY(QR(),".claude","settings.json")),(await Promise.all(Y.map(J))).filter((W)=>W!==null)}function l1(Q,X,$=process.platform==="win32"){let J=Q.replace(/\\/g,"/"),Y=J.split("/").pop()??J,G=(H)=>{let z=u1(H,$);if(z.test(J))return!0;if(!H.includes("/"))return z.test(Y);if(!H.startsWith("/")&&!H.startsWith("*"))return u1(`**/${H}`,$).test(J);return!1},W=X.flat().find(G);return W?{denied:!0,matchedPattern:W}:{denied:!1}}async function A6(Q,X,$,J=[]){let Y=Q.startsWith("/")?Q:JR($??process.cwd(),Q),G;try{G=await PY(Y)}catch{return{ok:!1,error:Y0(`Error reading file: "${Q}" not found`)}}let W=await $R(G);if(!W.isFile())return{ok:!1,error:Y0(`"${Q}" is not a regular file`)};let H;try{H=await PY($?$:process.cwd())}catch{return{ok:!1,error:Y0("Project directory not found or inaccessible")}}let z=await Promise.all(J.map((D)=>PY(D).catch(()=>D)));if(![H,...z].some((D)=>G===D||G.startsWith(D+YR)))return{ok:!1,error:Y0(`Access denied: "${Q}" is outside the project directory`)};let B=await m1(X,$),{denied:A,matchedPattern:U}=l1(G,B);if(A)return{ok:!1,error:Y0(`Access denied: "${Q}" matched deny pattern "${U}"`)};return{ok:!0,resolvedPath:G,size:W.size,mtimeMs:W.mtimeMs}}function c1(Q){let X=[],$=new Map;for(let W of Q){let H=A4(W.checksum);$.set(W.checksum,H);let z=y1(W.range);if(z.start.line===0&&!z.insertAfter)return{ok:!1,error:Y0("range starting at line 0 requires insert-after (use +0: prefix)")};if(z.start.line>0){if(H.startLine>z.start.line||H.endLine<z.end.line)return{ok:!1,error:Y0(`Checksum range ${H.startLine}-${H.endLine} does not cover edit range ${z.start.line}-${z.end.line}. Re-read with trueline_read to get a checksum covering the target lines.`)}}X.push({startLine:z.start.line,endLine:z.end.line,content:W.content===""?[]:W.content.split(`
55
55
  `),insertAfter:z.insertAfter,startHash:z.start.hash,endHash:z.end.hash})}let J=[...$.values()];J.sort((W,H)=>W.startLine-H.startLine);let Y=X.filter((W)=>!W.insertAfter);Y.sort((W,H)=>W.startLine-H.startLine);for(let W=1;W<Y.length;W++)if(Y[W].startLine<=Y[W-1].endLine)return{ok:!1,error:Y0(`Overlapping ranges: line ${Y[W].startLine} targeted by multiple edits`)};let G=X.filter((W)=>W.insertAfter);for(let W of G)for(let H of Y)if(W.startLine>=H.startLine&&W.startLine<H.endLine)return{ok:!1,error:Y0(`Insert-after at line ${W.startLine} conflicts with replace range ${H.startLine}–${H.endLine}. Insert after the last line of the replace instead.`)};return{ok:!0,ops:X,checksumRefs:J}}var GR={"utf-8":"utf-8",utf8:"utf-8",ascii:"ascii",latin1:"latin1"};function D4(Q){if(Q===void 0)return"utf-8";let X=GR[Q.toLowerCase()];if(X===void 0)throw Error(`Unsupported encoding "${Q}". Supported: utf-8, ascii, latin1`);return X}async function i1(Q){let{compare_against:X="HEAD",projectDir:$,allowedDirs:J}=Q,Y=Q.file_paths;if(Y.length===1&&Y[0]==="*"){if(Y=VR($??process.cwd(),X),Y.length===0)return I0("No changed files found.")}let G=[];for(let W of Y){let H=await A6(W,"Read",$,J);if(!H.ok){G.push(`## ${W}
56
56
 
57
57
  Access denied.`);continue}let{resolvedPath:z}=H,K=zR(z).replace(/^\./,""),q=W.startsWith("/")?p1($??process.cwd(),z):W,B;try{B=await HR(z,"utf-8")}catch{G.push(`## ${q}
@@ -103,7 +103,7 @@ ${B}`)}let H=Y.join(`
103
103
  `))}async function s1(Q){let{file_path:X,checksums:$,projectDir:J,allowedDirs:Y}=Q;if(!$||$.length===0)return Y0("No checksums provided");let G=await A6(X,"Read",J,Y);if(!G.ok)return G.error;let{resolvedPath:W}=G,H=[];for(let B of $){let A;try{A=A4(B)}catch(U){return Y0(U.message)}H.push({startLine:A.startLine,endLine:A.endLine,expected:A.hash,hash:U6})}H.sort((B,A)=>B.startLine-A.startLine);let z=0;try{let B=0;for await(let{lineBytes:A,lineNumber:U}of A9(W,{detectBinary:!0})){z=U;while(B<H.length&&H[B].startLine===0)B++;if(B>=H.length)break;if(U<H[B].startLine)continue;while(B<H.length&&H[B].startLine>0&&U>H[B].endLine)B++;if(B>=H.length)break;if(H[B].startLine>0&&U<H[B].startLine)continue;let D=d6(A,0,A.length);for(let F=B;F<H.length&&H[F].startLine<=U;F++)if(H[F].startLine>0&&U<=H[F].endLine)H[F].hash=C6(H[F].hash,D)}}catch(B){if(B instanceof Error&&B.message.includes("binary"))return Y0(`"${X}" appears to be a binary file`);throw B}let K=[],q=!0;for(let B of H){if(B.startLine===0&&B.endLine===0){if(z===0&&B.expected==="00000000")K.push("valid: 0-0:00000000");else if(q=!1,z===0)K.push(`stale: 0-0:${B.expected}`);else K.push(`stale: 0-0:${B.expected} (file now has ${z} lines)`);continue}if(B.startLine>z||B.endLine>z){q=!1;let U=B.startLine>z?`range past EOF (file has ${z} lines)`:`actual: ${T6(B.startLine,Math.min(B.endLine,z),B.hash)}`;K.push(`stale: ${B.startLine}-${B.endLine}:${B.expected} (${U})`);continue}if(B.hash.toString(16).padStart(8,"0")===B.expected)K.push(`valid: ${T6(B.startLine,B.endLine,B.hash)}`);else q=!1,K.push(`stale: ${B.startLine}-${B.endLine}:${B.expected} (actual: ${T6(B.startLine,B.endLine,B.hash)})`)}if(q)return I0("all checksums valid");return I0(K.join(`
104
104
  `))}import{readFile as TR,writeFile as kR}from"node:fs/promises";import{join as vR}from"node:path";import{tmpdir as _R}from"node:os";var xR="trueline-mcp",gR=86400000,Qq=vR(_R(),"trueline-mcp-update-check.json"),yR=3000;async function hR(){try{let Q=await TR(Qq,"utf-8");return JSON.parse(Q)}catch{return null}}async function fR(Q){await kR(Qq,JSON.stringify(Q)).catch(()=>{})}async function uR(){try{let Q=new AbortController,X=setTimeout(()=>Q.abort(),yR),$=await fetch(`https://registry.npmjs.org/${xR}/latest`,{signal:Q.signal});if(clearTimeout(X),!$.ok)return null;return(await $.json()).version??null}catch{return null}}function Xq(Q,X){let $=X??mR;(async()=>{let J=await hR();if(J&&Date.now()-J.timestamp<gR){if(e1(J.latestVersion,Q)>0)$({current:Q,latest:J.latestVersion});return}let Y=await uR();if(!Y)return;if(await fR({timestamp:Date.now(),latestVersion:Y}),e1(Y,Q)>0)$({current:Q,latest:Y})})()}function mR({current:Q,latest:X}){process.stderr.write(`[trueline-mcp] update available: ${Q} → ${X} (npm i -g trueline-mcp)
105
105
  `)}function e1(Q,X){let $=Q.split(".").map(Number),J=X.split(".").map(Number);for(let Y=0;Y<Math.max($.length,J.length);Y++){let G=($[Y]??0)-(J[Y]??0);if(G!==0)return G}return 0}var lR={path:"file_path",filePath:"file_path",file:"file_path",paths:"file_paths",filePaths:"file_paths",files:"file_paths",ref:"compare_against"},cR=/^(\d+)(?:[:-]|\.\.)(\d+)$/,pR=/^(\d+)$/;function iR(Q){if(typeof Q!=="string")return Q;let X=cR.exec(Q);if(X)return{start:Number(X[1]),end:Number(X[2])};let $=pR.exec(Q);if($)return{start:Number($[1])};return Q}function x9(Q){if(typeof Q!=="object"||Q===null)return Q;let X=Q,$={};for(let[J,Y]of Object.entries(X)){let G=lR[J]??J;if(G!==J&&G in X)continue;if(typeof Y==="string"&&(Y.startsWith("[")||Y.startsWith("{")))try{$[G]=JSON.parse(Y);continue}catch{}if(Y==="true"){$[G]=!0;continue}if(Y==="false"){$[G]=!1;continue}if(G==="ranges"&&Array.isArray(Y)){$[G]=Y.map(iR);continue}$[G]=Y}return $}function NQ(Q){return async(X)=>{try{return await Q(X)}catch($){let J=$ instanceof Error?$.message:String($);return process.stderr.write(`[trueline-mcp] tool error: ${J}
106
- `),Y0(`Internal error: ${J}`)}}}var Jq=F1.version,D9=new OY({name:"trueline-mcp",version:Jq}),$q=process.env.CLAUDE_PROJECT_DIR??process.cwd(),EQ=await TY($q).catch(()=>$q);async function tR(){let Q=[];if(process.env.CLAUDE_CODE_ENTRYPOINT){let $=rR(dR(),".claude");await nR($,{recursive:!0}).catch(()=>{});let J=await TY($).catch(()=>null);if(J)Q.push(J)}let X=process.env.TRUELINE_ALLOWED_DIRS;if(X)for(let $ of X.split(oR).filter(Boolean)){let J=await TY($).catch(()=>null);if(J)Q.push(J)}return Q}var MQ=await tR();D9.registerTool("trueline_read",{description:"Read a file; returns 'N:hash\tcontent' per line plus a checksum per range.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),ranges:d.array(d.object({start:d.number().int().positive().describe("First line to read (1-based).").optional(),end:d.number().int().positive().describe("Last line to read (1-based, inclusive).").optional()})).describe("Line ranges to read. Omit to read the whole file. Example: [{start: 10, end: 25}] or [{start: 1, end: 50}, {start: 200, end: 220}] for disjoint ranges. Each range gets its own checksum.").optional(),encoding:d.string().describe("File encoding. Defaults to utf-8. Supported: utf-8, ascii, latin1.").optional(),hashes:d.boolean().describe("Include per-line hashes in output. Defaults to true. Set to false for exploratory reads where you don't plan to edit — saves tokens. Checksums are always included.").optional()}))},NQ(async(Q)=>{return o1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_edit",{description:"Apply hash-verified edits to a file. Each edit carries its own checksum.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),edits:d.array(d.object({checksum:d.string().describe("Checksum from trueline_read for the covering range"),range:d.string().describe("startLine:hash..endLine:hash or startLine:hash; prefix + for insert-after"),content:d.string().describe("Replacement lines, newline-separated. Empty string to delete.")})).min(1),encoding:d.string().describe("File encoding. Defaults to utf-8. Supported: utf-8, ascii, latin1.").optional(),dry_run:d.boolean().describe("Preview edits as unified diff without writing. Defaults to false.").optional()}))},NQ(async(Q)=>{return d1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_diff",{description:"Semantic, AST-based summary of structural changes compared to a git ref. Detects added/removed/renamed symbols, signature changes, and logic modifications. Pass ALL files in a single call via file_paths (never call once per file). Use instead of `git diff` to review changes with minimal token usage.",inputSchema:d.preprocess(x9,d.object({file_paths:d.array(d.string()).min(1).describe('Paths to diff. Pass multiple files in one call. Use ["*"] for all changed files.'),compare_against:d.string().describe('Git ref to compare against. Defaults to "HEAD". Use ":0" for staged content.').optional()}))},NQ(async(Q)=>{return i1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_outline",{description:"Get a compact structural outline of source files (functions, classes, types, etc.) without reading the full content. "+"Much smaller than trueline_read — use first to find line ranges, then read specific sections.",inputSchema:d.preprocess(x9,d.object({file_paths:d.array(d.string()).describe("One or more absolute or project-relative file paths to outline."),depth:d.number().int().min(0).describe("Maximum nesting depth. 0 = top-level only, 1 = include class/interface members. Omit for all levels.").optional()}))},NQ(async(Q)=>{return t1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_search",{description:"Search a file for a literal string or regex pattern. Returns matching lines with context, per-line hashes, and checksums — "+"ready for immediate editing. Use instead of outline+read when you know what to look for.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),pattern:d.string().describe("Search string. Literal by default; set regex=true for regular expressions."),context_lines:d.number().int().min(0).describe("Lines of context above/below each match. Default: 2.").optional(),max_matches:d.number().int().positive().describe("Maximum number of matches to return. Default: 10.").optional(),case_insensitive:d.boolean().describe("Case-insensitive matching. Default: false.").optional(),regex:d.boolean().describe("Treat pattern as a regular expression. Default: false (literal match).").optional()}))},NQ(async(Q)=>{return a1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_verify",{description:"Validate held checksums against a file. Returns which are valid or stale. "+"Cheaper than re-reading — use before editing when the file may have changed.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),checksums:d.array(d.string()).describe('Checksum strings from a prior trueline_read, e.g. ["1-50:abcdef01"].')}))},NQ(async(Q)=>{return s1({...Q,projectDir:EQ,allowedDirs:MQ})}));var aR=new wY;try{await D9.connect(aR)}catch(Q){console.error("Failed to start trueline-mcp server:",Q),process.exit(1)}process.on("uncaughtException",(Q)=>{process.stderr.write(`[trueline-mcp] uncaught exception: ${Q.message}
106
+ `),Y0(`Internal error: ${J}`)}}}var Jq=F1.version,D9=new OY({name:"trueline-mcp",version:Jq}),$q=process.env.CLAUDE_PROJECT_DIR??process.cwd(),EQ=await TY($q).catch(()=>$q);async function tR(){let Q=[];if(process.env.CLAUDE_CODE_ENTRYPOINT){let $=rR(dR(),".claude");await nR($,{recursive:!0}).catch(()=>{});let J=await TY($).catch(()=>null);if(J)Q.push(J)}let X=process.env.TRUELINE_ALLOWED_DIRS;if(X)for(let $ of X.split(oR).filter(Boolean)){let J=await TY($).catch(()=>null);if(J)Q.push(J)}return Q}var MQ=await tR();D9.registerTool("trueline_read",{description:"Read a file; returns 'N:hash\tcontent' per line plus a checksum per range.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),ranges:d.array(d.object({start:d.number().int().positive().describe("First line to read (1-based).").optional(),end:d.number().int().positive().describe("Last line to read (1-based, inclusive).").optional()})).describe("Line ranges to read. Omit to read the whole file. Example: [{start: 10, end: 25}] or [{start: 1, end: 50}, {start: 200, end: 220}] for disjoint ranges. Each range gets its own checksum.").optional(),encoding:d.string().describe("File encoding. Defaults to utf-8. Supported: utf-8, ascii, latin1.").optional(),hashes:d.boolean().describe("Include per-line hashes in output. Defaults to true. Set to false for exploratory reads where you don't plan to edit — saves tokens. Checksums are always included.").optional()}))},NQ(async(Q)=>{return o1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_edit",{description:"Apply hash-verified edits to a file. Each edit carries its own checksum.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),edits:d.array(d.object({checksum:d.string().describe("Checksum from trueline_read for the covering range"),range:d.string().describe("startLine:hash-endLine:hash or startLine:hash; prefix + for insert-after"),content:d.string().describe("Replacement lines, newline-separated. Empty string to delete.")})).min(1),encoding:d.string().describe("File encoding. Defaults to utf-8. Supported: utf-8, ascii, latin1.").optional(),dry_run:d.boolean().describe("Preview edits as unified diff without writing. Defaults to false.").optional()}))},NQ(async(Q)=>{return d1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_diff",{description:"Semantic, AST-based summary of structural changes compared to a git ref. Detects added/removed/renamed symbols, signature changes, and logic modifications. Pass ALL files in a single call via file_paths (never call once per file). Use instead of `git diff` to review changes with minimal token usage.",inputSchema:d.preprocess(x9,d.object({file_paths:d.array(d.string()).min(1).describe('Paths to diff. Pass multiple files in one call. Use ["*"] for all changed files.'),compare_against:d.string().describe('Git ref to compare against. Defaults to "HEAD". Use ":0" for staged content.').optional()}))},NQ(async(Q)=>{return i1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_outline",{description:"Get a compact structural outline of source files (functions, classes, types, etc.) without reading the full content. "+"Much smaller than trueline_read — use first to find line ranges, then read specific sections.",inputSchema:d.preprocess(x9,d.object({file_paths:d.array(d.string()).describe("One or more absolute or project-relative file paths to outline."),depth:d.number().int().min(0).describe("Maximum nesting depth. 0 = top-level only, 1 = include class/interface members. Omit for all levels.").optional()}))},NQ(async(Q)=>{return t1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_search",{description:"Search a file for a literal string or regex pattern. Returns matching lines with context, per-line hashes, and checksums — "+"ready for immediate editing. Use instead of outline+read when you know what to look for.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),pattern:d.string().describe("Search string. Literal by default; set regex=true for regular expressions."),context_lines:d.number().int().min(0).describe("Lines of context above/below each match. Default: 2.").optional(),max_matches:d.number().int().positive().describe("Maximum number of matches to return. Default: 10.").optional(),case_insensitive:d.boolean().describe("Case-insensitive matching. Default: false.").optional(),regex:d.boolean().describe("Treat pattern as a regular expression. Default: false (literal match).").optional()}))},NQ(async(Q)=>{return a1({...Q,projectDir:EQ,allowedDirs:MQ})}));D9.registerTool("trueline_verify",{description:"Validate held checksums against a file. Returns which are valid or stale. "+"Cheaper than re-reading — use before editing when the file may have changed.",inputSchema:d.preprocess(x9,d.object({file_path:d.string().describe("Absolute or project-relative file path."),checksums:d.array(d.string()).describe('Checksum strings from a prior trueline_read, e.g. ["1-50:abcdef01"].')}))},NQ(async(Q)=>{return s1({...Q,projectDir:EQ,allowedDirs:MQ})}));var aR=new wY;try{await D9.connect(aR)}catch(Q){console.error("Failed to start trueline-mcp server:",Q),process.exit(1)}process.on("uncaughtException",(Q)=>{process.stderr.write(`[trueline-mcp] uncaught exception: ${Q.message}
107
107
  `)});process.on("unhandledRejection",(Q)=>{let X=Q instanceof Error?Q.message:String(Q);process.stderr.write(`[trueline-mcp] unhandled rejection: ${X}
108
108
  `)});Xq(Jq,({current:Q,latest:X})=>{let $=`update available: ${Q} → ${X} (npm i -g trueline-mcp)`;process.stderr.write(`[trueline-mcp] ${$}
109
109
  `),D9.sendLoggingMessage({level:"warning",logger:"trueline-mcp",data:$}).catch(()=>{})});
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "trueline-mcp",
3
- "version": "2.5.6",
3
+ "version": "2.5.8",
4
4
  "type": "module",
5
5
  "description": "Truth-verified file editing for AI coding agents via MCP",
6
6
  "license": "Apache-2.0",
@@ -35,35 +35,28 @@ function hasDeno() {
35
35
  // Dependency Installation
36
36
  // ==============================================================================
37
37
 
38
- // When installed as a Claude Code plugin, tree-sitter WASM files are needed
39
- // by trueline_outline but aren't bundled. Install them globally so they don't
40
- // bloat the plugin dir (Claude Code copies the entire plugin directory to a
41
- // versioned cache, and node_modules causes ENAMETOOLONG).
42
- const WASM_PKGS = ["web-tree-sitter@0.24.7", "tree-sitter-wasms@0.1.13"];
43
-
44
- function globalModulesDir() {
45
- try {
46
- return execFileSync("npm", ["root", "-g"], { encoding: "utf-8", stdio: ["ignore", "pipe", "ignore"] }).trim();
47
- } catch {
48
- return null;
49
- }
50
- }
51
-
38
+ // When installed as a Claude Code plugin, node_modules won't exist.
39
+ // Install dependencies on first launch so tree-sitter WASMs (needed by
40
+ // trueline_outline) and other native deps are available.
52
41
  function ensureDeps() {
53
- const globalDir = globalModulesDir();
54
- if (!globalDir) return;
42
+ if (existsSync(path.join(pluginRoot, "node_modules"))) return;
55
43
 
56
- for (const spec of WASM_PKGS) {
57
- const pkg = spec.split("@")[0];
58
- if (existsSync(path.join(globalDir, pkg))) continue;
59
- try {
60
- execFileSync("npm", ["install", "-g", spec, "--silent"], {
61
- stdio: "pipe",
62
- timeout: 60_000,
63
- });
64
- } catch {
65
- // Non-fatal: outline won't work, but read/edit/search/diff/verify will.
66
- }
44
+ process.stderr.write("trueline-mcp: installing dependencies (first run)...\n");
45
+ try {
46
+ // Prefer bun for speed, fall back to npm (ships with node).
47
+ const installer = hasBun() ? "bun" : "npm";
48
+ const args = installer === "bun" ? ["install"] : ["install", "--production"];
49
+ execFileSync(installer, args, {
50
+ cwd: pluginRoot,
51
+ stdio: ["ignore", "ignore", "inherit"],
52
+ timeout: 120_000,
53
+ });
54
+ process.stderr.write("trueline-mcp: dependencies installed.\n");
55
+ } catch (err) {
56
+ // Non-fatal: outline won't work, but read/edit/search/diff/verify will.
57
+ process.stderr.write(
58
+ `trueline-mcp: dependency install failed (${err.message}). trueline_outline will be unavailable.\n`,
59
+ );
67
60
  }
68
61
  }
69
62
 
@@ -85,16 +78,8 @@ if (hasBun()) {
85
78
  args = [path.join(pluginRoot, "dist", "server.js")];
86
79
  }
87
80
 
88
- // Node doesn't search global node_modules by default — expose via NODE_PATH
89
- const env = { ...process.env };
90
- const globalDir = globalModulesDir();
91
- if (globalDir) {
92
- env.NODE_PATH = env.NODE_PATH ? `${globalDir}${path.delimiter}${env.NODE_PATH}` : globalDir;
93
- }
94
-
95
81
  const child = spawn(cmd, [...args, ...process.argv.slice(2)], {
96
82
  stdio: "inherit",
97
- env,
98
83
  });
99
84
 
100
85
  child.on("error", (err) => {