junso-browser 0.2.3 → 0.2.6

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/mcp.js CHANGED
@@ -39,5 +39,5 @@ Set the \`cycles\` parameter to \`"ref"\` to resolve cyclical schemas with defs.
39
39
  `:`[${Y[B]}\r
40
40
  ]`;continue}if($+=Y[B],Y[B]==="\\")J=!0;else if(G&&Y[B]==="]")G=!1;else if(!G&&Y[B]==="[")G=!0}try{new RegExp($)}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 $}function i9(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===M.ZodEnum)return{type:"object",required:Q.keyType._def.values,properties:Q.keyType._def.values.reduce((Y,$)=>({...Y,[$]:g(Q.valueType._def,{...X,currentPath:[...X.currentPath,"properties",$]})??z0(X)}),{}),additionalProperties:X.rejectedAdditionalProperties};let W={type:"object",additionalProperties:g(Q.valueType._def,{...X,currentPath:[...X.currentPath,"additionalProperties"]})??X.allowedAdditionalProperties};if(X.target==="openApi3")return W;if(Q.keyType?._def.typeName===M.ZodString&&Q.keyType._def.checks?.length){let{type:Y,...$}=p9(Q.keyType._def,X);return{...W,propertyNames:$}}else if(Q.keyType?._def.typeName===M.ZodEnum)return{...W,propertyNames:{enum:Q.keyType._def.values}};else if(Q.keyType?._def.typeName===M.ZodBranded&&Q.keyType._def.type._def.typeName===M.ZodString&&Q.keyType._def.type._def.checks?.length){let{type:Y,...$}=c9(Q.keyType._def,X);return{...W,propertyNames:$}}return W}function BJ(Q,X){if(X.mapStrategy==="record")return i9(Q,X);let W=g(Q.keyType._def,{...X,currentPath:[...X.currentPath,"items","items","0"]})||z0(X),Y=g(Q.valueType._def,{...X,currentPath:[...X.currentPath,"items","items","1"]})||z0(X);return{type:"array",maxItems:125,items:{type:"array",items:[W,Y],minItems:2,maxItems:2}}}function zJ(Q){let X=Q.values,Y=Object.keys(Q.values).filter((J)=>{return typeof X[X[J]]!=="number"}).map((J)=>X[J]),$=Array.from(new Set(Y.map((J)=>typeof J)));return{type:$.length===1?$[0]==="string"?"string":"number":["string","number"],enum:Y}}function AJ(Q){return Q.target==="openAi"?void 0:{not:z0({...Q,currentPath:[...Q.currentPath,"not"]})}}function KJ(Q){return Q.target==="openApi3"?{enum:["null"],nullable:!0}:{type:"null"}}var E4={ZodString:"string",ZodNumber:"number",ZodBigInt:"integer",ZodBoolean:"boolean",ZodNull:"null"};function LJ(Q,X){if(X.target==="openApi3")return DJ(Q,X);let W=Q.options instanceof Map?Array.from(Q.options.values()):Q.options;if(W.every((Y)=>(Y._def.typeName in E4)&&(!Y._def.checks||!Y._def.checks.length))){let Y=W.reduce(($,J)=>{let G=E4[J._def.typeName];return G&&!$.includes(G)?[...$,G]:$},[]);return{type:Y.length>1?Y:Y[0]}}else if(W.every((Y)=>Y._def.typeName==="ZodLiteral"&&!Y.description)){let Y=W.reduce(($,J)=>{let G=typeof J._def.value;switch(G){case"string":case"number":case"boolean":return[...$,G];case"bigint":return[...$,"integer"];case"object":if(J._def.value===null)return[...$,"null"];case"symbol":case"undefined":case"function":default:return $}},[]);if(Y.length===W.length){let $=Y.filter((J,G,H)=>H.indexOf(J)===G);return{type:$.length>1?$:$[0],enum:W.reduce((J,G)=>{return J.includes(G._def.value)?J:[...J,G._def.value]},[])}}}else if(W.every((Y)=>Y._def.typeName==="ZodEnum"))return{type:"string",enum:W.reduce((Y,$)=>[...Y,...$._def.values.filter((J)=>!Y.includes(J))],[])};return DJ(Q,X)}var DJ=(Q,X)=>{let W=(Q.options instanceof Map?Array.from(Q.options.values()):Q.options).map((Y,$)=>g(Y._def,{...X,currentPath:[...X.currentPath,"anyOf",`${$}`]})).filter((Y)=>!!Y&&(!X.strictUnions||typeof Y==="object"&&Object.keys(Y).length>0));return W.length?{anyOf:W}:void 0};function qJ(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:E4[Q.innerType._def.typeName],nullable:!0};return{type:[E4[Q.innerType._def.typeName],"null"]}}if(X.target==="openApi3"){let Y=g(Q.innerType._def,{...X,currentPath:[...X.currentPath]});if(Y&&"$ref"in Y)return{allOf:[Y],nullable:!0};return Y&&{...Y,nullable:!0}}let W=g(Q.innerType._def,{...X,currentPath:[...X.currentPath,"anyOf","0"]});return W&&{anyOf:[W,{type:"null"}]}}function VJ(Q,X){let W={type:"number"};if(!Q.checks)return W;for(let Y of Q.checks)switch(Y.kind){case"int":W.type="integer",GW(W,"type",Y.message,X);break;case"min":if(X.target==="jsonSchema7")if(Y.inclusive)n(W,"minimum",Y.value,Y.message,X);else n(W,"exclusiveMinimum",Y.value,Y.message,X);else{if(!Y.inclusive)W.exclusiveMinimum=!0;n(W,"minimum",Y.value,Y.message,X)}break;case"max":if(X.target==="jsonSchema7")if(Y.inclusive)n(W,"maximum",Y.value,Y.message,X);else n(W,"exclusiveMaximum",Y.value,Y.message,X);else{if(!Y.inclusive)W.exclusiveMaximum=!0;n(W,"maximum",Y.value,Y.message,X)}break;case"multipleOf":n(W,"multipleOf",Y.value,Y.message,X);break}return W}function FJ(Q,X){let W=X.target==="openAi",Y={type:"object",properties:{}},$=[],J=Q.shape();for(let H in J){let B=J[H];if(B===void 0||B._def===void 0)continue;let z=ZL(B);if(z&&W){if(B._def.typeName==="ZodOptional")B=B._def.innerType;if(!B.isNullable())B=B.nullable();z=!1}let A=g(B._def,{...X,currentPath:[...X.currentPath,"properties",H],propertyPath:[...X.currentPath,"properties",H]});if(A===void 0)continue;if(Y.properties[H]=A,!z)$.push(H)}if($.length)Y.required=$;let G=CL(Q,X);if(G!==void 0)Y.additionalProperties=G;return Y}function CL(Q,X){if(Q.catchall._def.typeName!=="ZodNever")return g(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 ZL(Q){try{return Q.isOptional()}catch{return!0}}var UJ=(Q,X)=>{if(X.currentPath.toString()===X.propertyPath?.toString())return g(Q.innerType._def,X);let W=g(Q.innerType._def,{...X,currentPath:[...X.currentPath,"anyOf","1"]});return W?{anyOf:[{not:z0(X)},W]}:z0(X)};var wJ=(Q,X)=>{if(X.pipeStrategy==="input")return g(Q.in._def,X);else if(X.pipeStrategy==="output")return g(Q.out._def,X);let W=g(Q.in._def,{...X,currentPath:[...X.currentPath,"allOf","0"]}),Y=g(Q.out._def,{...X,currentPath:[...X.currentPath,"allOf",W?"1":"0"]});return{allOf:[W,Y].filter(($)=>$!==void 0)}};function OJ(Q,X){return g(Q.type._def,X)}function NJ(Q,X){let Y={type:"array",uniqueItems:!0,items:g(Q.valueType._def,{...X,currentPath:[...X.currentPath,"items"]})};if(Q.minSize)n(Y,"minItems",Q.minSize.value,Q.minSize.message,X);if(Q.maxSize)n(Y,"maxItems",Q.maxSize.value,Q.maxSize.message,X);return Y}function MJ(Q,X){if(Q.rest)return{type:"array",minItems:Q.items.length,items:Q.items.map((W,Y)=>g(W._def,{...X,currentPath:[...X.currentPath,"items",`${Y}`]})).reduce((W,Y)=>Y===void 0?W:[...W,Y],[]),additionalItems:g(Q.rest._def,{...X,currentPath:[...X.currentPath,"additionalItems"]})};else return{type:"array",minItems:Q.items.length,maxItems:Q.items.length,items:Q.items.map((W,Y)=>g(W._def,{...X,currentPath:[...X.currentPath,"items",`${Y}`]})).reduce((W,Y)=>Y===void 0?W:[...W,Y],[])}}function jJ(Q){return{not:z0(Q)}}function bJ(Q){return z0(Q)}var EJ=(Q,X)=>{return g(Q.innerType._def,X)};var SJ=(Q,X,W)=>{switch(X){case M.ZodString:return p9(Q,W);case M.ZodNumber:return VJ(Q,W);case M.ZodObject:return FJ(Q,W);case M.ZodBigInt:return e7(Q,W);case M.ZodBoolean:return QJ();case M.ZodDate:return HW(Q,W);case M.ZodUndefined:return jJ(W);case M.ZodNull:return KJ(W);case M.ZodArray:return s7(Q,W);case M.ZodUnion:case M.ZodDiscriminatedUnion:return LJ(Q,W);case M.ZodIntersection:return JJ(Q,W);case M.ZodTuple:return MJ(Q,W);case M.ZodRecord:return i9(Q,W);case M.ZodLiteral:return GJ(Q,W);case M.ZodEnum:return $J(Q);case M.ZodNativeEnum:return zJ(Q);case M.ZodNullable:return qJ(Q,W);case M.ZodOptional:return UJ(Q,W);case M.ZodMap:return BJ(Q,W);case M.ZodSet:return NJ(Q,W);case M.ZodLazy:return()=>Q.getter()._def;case M.ZodPromise:return OJ(Q,W);case M.ZodNaN:case M.ZodNever:return AJ(W);case M.ZodEffects:return YJ(Q,W);case M.ZodAny:return z0(W);case M.ZodUnknown:return bJ(W);case M.ZodDefault:return WJ(Q,W);case M.ZodBranded:return c9(Q,W);case M.ZodReadonly:return EJ(Q,W);case M.ZodCatch:return XJ(Q,W);case M.ZodPipeline:return wJ(Q,W);case M.ZodFunction:case M.ZodVoid:case M.ZodSymbol:return;default:return((Y)=>{return})(X)}};function g(Q,X,W=!1){let Y=X.seen.get(Q);if(X.override){let H=X.override?.(Q,X,Y,W);if(H!==r7)return H}if(Y&&!W){let H=TL(Y,X);if(H!==void 0)return H}let $={def:Q,path:X.currentPath,jsonSchema:void 0};X.seen.set(Q,$);let J=SJ(Q,Q.typeName,X),G=typeof J==="function"?g(J(),X):J;if(G)kL(Q,X,G);if(X.postProcess){let H=X.postProcess(G,Q,X);return $.jsonSchema=G,H}return $.jsonSchema=G,G}var TL=(Q,X)=>{switch(X.$refStrategy){case"root":return{$ref:Q.path.join("/")};case"relative":return{$ref:m9(X.currentPath,Q.path)};case"none":case"seen":{if(Q.path.length<X.currentPath.length&&Q.path.every((W,Y)=>X.currentPath[Y]===W))return console.warn(`Recursive reference detected at ${X.currentPath.join("/")}! Defaulting to any`),z0(X);return X.$refStrategy==="seen"?z0(X):void 0}}},kL=(Q,X,W)=>{if(Q.description){if(W.description=Q.description,X.markdownDescription)W.markdownDescription=Q.description}return W};var AW=(Q,X)=>{let W=a7(X),Y=typeof X==="object"&&X.definitions?Object.entries(X.definitions).reduce((B,[z,A])=>({...B,[z]:g(A._def,{...W,currentPath:[...W.basePath,W.definitionPath,z]},!0)??z0(W)}),{}):void 0,$=typeof X==="string"?X:X?.nameStrategy==="title"?void 0:X?.name,J=g(Q._def,$===void 0?W:{...W,currentPath:[...W.basePath,W.definitionPath,$]},!1)??z0(W),G=typeof X==="object"&&X.name!==void 0&&X.nameStrategy==="title"?X.name:void 0;if(G!==void 0)J.title=G;if(W.flags.hasReferencedOpenAiAnyType){if(!Y)Y={};if(!Y[W.openAiAnyTypeName])Y[W.openAiAnyTypeName]={type:["string","number","integer","boolean","array","null"],items:{$ref:W.$refStrategy==="relative"?"1":[...W.basePath,W.definitionPath,W.openAiAnyTypeName].join("/")}}}let H=$===void 0?Y?{...J,[W.definitionPath]:Y}:J:{$ref:[...W.$refStrategy==="relative"?[]:W.basePath,W.definitionPath,$].join("/"),[W.definitionPath]:{...Y,[$]:J}};if(W.target==="jsonSchema7")H.$schema="http://json-schema.org/draft-07/schema#";else if(W.target==="jsonSchema2019-09"||W.target==="openAi")H.$schema="https://json-schema.org/draft/2019-09/schema#";if(W.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 vL(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 KW(Q,X){if(m0(Q))return _X(Q,{target:vL(X?.target),io:X?.pipeStrategy??"input"});return AW(Q,{strictUnions:X?.strictUnions??!0,pipeStrategy:X?.pipeStrategy??"input"})}function DW(Q){let W=T1(Q)?.method;if(!W)throw Error("Schema is missing a method literal");let Y=w9(W);if(typeof Y!=="string")throw Error("Schema method literal must be a string");return Y}function LW(Q,X){let W=Z1(Q,X);if(!W.success)throw W.error;return W.data}var _L=60000;class qW{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(S9,(X)=>{this._oncancel(X)}),this.setNotificationHandler(I9,(X)=>{this._onprogress(X)}),this.setRequestHandler(R9,(X)=>({})),this._taskStore=Q?.taskStore,this._taskMessageQueue=Q?.taskMessageQueue,this._taskStore)this.setRequestHandler(P9,async(X,W)=>{let Y=await this._taskStore.getTask(X.params.taskId,W.sessionId);if(!Y)throw new C(k.InvalidParams,"Failed to retrieve task: Task not found");return{...Y}}),this.setRequestHandler(Z9,async(X,W)=>{let Y=async()=>{let $=X.params.taskId;if(this._taskMessageQueue){let G;while(G=await this._taskMessageQueue.dequeue($,W.sessionId)){if(G.type==="response"||G.type==="error"){let H=G.message,B=H.id,z=this._requestResolvers.get(B);if(z)if(this._requestResolvers.delete(B),G.type==="response")z(H);else{let A=H,D=new C(A.error.code,A.error.message,A.error.data);z(D)}else{let A=G.type==="response"?"Response":"Error";this._onerror(Error(`${A} handler missing for request ${B}`))}continue}await this._transport?.send(G.message,{relatedRequestId:W.requestId})}}let J=await this._taskStore.getTask($,W.sessionId);if(!J)throw new C(k.InvalidParams,`Task not found: ${$}`);if(!v1(J.status))return await this._waitForTaskUpdate($,W.signal),await Y();if(v1(J.status)){let G=await this._taskStore.getTaskResult($,W.sessionId);return this._clearTaskQueue($),{...G,_meta:{...G._meta,[k1]:{taskId:$}}}}return await Y()};return await Y()}),this.setRequestHandler(T9,async(X,W)=>{try{let{tasks:Y,nextCursor:$}=await this._taskStore.listTasks(X.params?.cursor,W.sessionId);return{tasks:Y,nextCursor:$,_meta:{}}}catch(Y){throw new C(k.InvalidParams,`Failed to list tasks: ${Y instanceof Error?Y.message:String(Y)}`)}}),this.setRequestHandler(v9,async(X,W)=>{try{let Y=await this._taskStore.getTask(X.params.taskId,W.sessionId);if(!Y)throw new C(k.InvalidParams,`Task not found: ${X.params.taskId}`);if(v1(Y.status))throw new C(k.InvalidParams,`Cannot cancel task in terminal status: ${Y.status}`);await this._taskStore.updateTaskStatus(X.params.taskId,"cancelled","Client cancelled task execution.",W.sessionId),this._clearTaskQueue(X.params.taskId);let $=await this._taskStore.getTask(X.params.taskId,W.sessionId);if(!$)throw new C(k.InvalidParams,`Task not found after cancellation: ${X.params.taskId}`);return{_meta:{},...$}}catch(Y){if(Y instanceof C)throw Y;throw new C(k.InvalidRequest,`Failed to cancel task: ${Y instanceof Error?Y.message:String(Y)}`)}})}async _oncancel(Q){if(!Q.params.requestId)return;this._requestHandlerAbortControllers.get(Q.params.requestId)?.abort(Q.params.reason)}_setupTimeout(Q,X,W,Y,$=!1){this._timeoutInfo.set(Q,{timeoutId:setTimeout(Y,X),startTime:Date.now(),timeout:X,maxTotalTimeout:W,resetTimeoutOnProgress:$,onTimeout:Y})}_resetTimeout(Q){let X=this._timeoutInfo.get(Q);if(!X)return!1;let W=Date.now()-X.startTime;if(X.maxTotalTimeout&&W>=X.maxTotalTimeout)throw this._timeoutInfo.delete(Q),C.fromError(k.RequestTimeout,"Maximum total timeout exceeded",{maxTotalTimeout:X.maxTotalTimeout,totalElapsed:W});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 W=this.transport?.onerror;this._transport.onerror=($)=>{W?.($),this._onerror($)};let Y=this._transport?.onmessage;this._transport.onmessage=($,J)=>{if(Y?.($,J),V4($)||g7($))this._onresponse($);else if(nX($))this._onrequest($,J);else if(x7($))this._onnotification($);else this._onerror(Error(`Unknown message type: ${JSON.stringify($)}`))},await this._transport.start()}_onclose(){let Q=this._responseHandlers;this._responseHandlers=new Map,this._progressHandlers.clear(),this._taskProgressTokens.clear(),this._pendingDebouncedNotifications.clear();for(let W of this._timeoutInfo.values())clearTimeout(W.timeoutId);this._timeoutInfo.clear();for(let W of this._requestHandlerAbortControllers.values())W.abort();this._requestHandlerAbortControllers.clear();let X=C.fromError(k.ConnectionClosed,"Connection closed");this._transport=void 0,this.onclose?.();for(let W of Q.values())W(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((W)=>this._onerror(Error(`Uncaught error in notification handler: ${W}`)))}_onrequest(Q,X){let W=this._requestHandlers.get(Q.method)??this.fallbackRequestHandler,Y=this._transport,$=Q.params?._meta?.[k1]?.taskId;if(W===void 0){let z={jsonrpc:"2.0",id:Q.id,error:{code:k.MethodNotFound,message:"Method not found"}};if($&&this._taskMessageQueue)this._enqueueTaskMessage($,{type:"error",message:z,timestamp:Date.now()},Y?.sessionId).catch((A)=>this._onerror(Error(`Failed to enqueue error response: ${A}`)));else Y?.send(z).catch((A)=>this._onerror(Error(`Failed to send an error response: ${A}`)));return}let J=new AbortController;this._requestHandlerAbortControllers.set(Q.id,J);let G=k7(Q.params)?Q.params.task:void 0,H=this._taskStore?this.requestTaskStore(Q,Y?.sessionId):void 0,B={signal:J.signal,sessionId:Y?.sessionId,_meta:Q.params?._meta,sendNotification:async(z)=>{if(J.signal.aborted)return;let A={relatedRequestId:Q.id};if($)A.relatedTask={taskId:$};await this.notification(z,A)},sendRequest:async(z,A,D)=>{if(J.signal.aborted)throw new C(k.ConnectionClosed,"Request was cancelled");let L={...D,relatedRequestId:Q.id};if($&&!L.relatedTask)L.relatedTask={taskId:$};let K=L.relatedTask?.taskId??$;if(K&&H)await H.updateTaskStatus(K,"input_required");return await this.request(z,A,L)},authInfo:X?.authInfo,requestId:Q.id,requestInfo:X?.requestInfo,taskId:$,taskStore:H,taskRequestedTtl:G?.ttl,closeSSEStream:X?.closeSSEStream,closeStandaloneSSEStream:X?.closeStandaloneSSEStream};Promise.resolve().then(()=>{if(G)this.assertTaskHandlerCapability(Q.method)}).then(()=>W(Q,B)).then(async(z)=>{if(J.signal.aborted)return;let A={result:z,jsonrpc:"2.0",id:Q.id};if($&&this._taskMessageQueue)await this._enqueueTaskMessage($,{type:"response",message:A,timestamp:Date.now()},Y?.sessionId);else await Y?.send(A)},async(z)=>{if(J.signal.aborted)return;let A={jsonrpc:"2.0",id:Q.id,error:{code:Number.isSafeInteger(z.code)?z.code:k.InternalError,message:z.message??"Internal error",...z.data!==void 0&&{data:z.data}}};if($&&this._taskMessageQueue)await this._enqueueTaskMessage($,{type:"error",message:A,timestamp:Date.now()},Y?.sessionId);else await Y?.send(A)}).catch((z)=>this._onerror(Error(`Failed to send response: ${z}`))).finally(()=>{if(this._requestHandlerAbortControllers.get(Q.id)===J)this._requestHandlerAbortControllers.delete(Q.id)})}_onprogress(Q){let{progressToken:X,...W}=Q.params,Y=Number(X),$=this._progressHandlers.get(Y);if(!$){this._onerror(Error(`Received a progress notification for an unknown token: ${JSON.stringify(Q)}`));return}let J=this._responseHandlers.get(Y),G=this._timeoutInfo.get(Y);if(G&&J&&G.resetTimeoutOnProgress)try{this._resetTimeout(Y)}catch(H){this._responseHandlers.delete(Y),this._progressHandlers.delete(Y),this._cleanupTimeout(Y),J(H);return}$(W)}_onresponse(Q){let X=Number(Q.id),W=this._requestResolvers.get(X);if(W){if(this._requestResolvers.delete(X),V4(Q))W(Q);else{let J=new C(Q.error.code,Q.error.message,Q.error.data);W(J)}return}let Y=this._responseHandlers.get(X);if(Y===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 $=!1;if(V4(Q)&&Q.result&&typeof Q.result==="object"){let J=Q.result;if(J.task&&typeof J.task==="object"){let G=J.task;if(typeof G.taskId==="string")$=!0,this._taskProgressTokens.set(G.taskId,X)}}if(!$)this._progressHandlers.delete(X);if(V4(Q))Y(Q);else{let J=C.fromError(Q.error.code,Q.error.message,Q.error.data);Y(J)}}get transport(){return this._transport}async close(){await this._transport?.close()}async*requestStream(Q,X,W){let{task:Y}=W??{};if(!Y){try{yield{type:"result",result:await this.request(Q,X,W)}}catch(J){yield{type:"error",error:J instanceof C?J:new C(k.InternalError,String(J))}}return}let $;try{let J=await this.request(Q,C6,W);if(J.task)$=J.task.taskId,yield{type:"taskCreated",task:J.task};else throw new C(k.InternalError,"Task creation did not return a task");while(!0){let G=await this.getTask({taskId:$},W);if(yield{type:"taskStatus",task:G},v1(G.status)){if(G.status==="completed")yield{type:"result",result:await this.getTaskResult({taskId:$},X,W)};else if(G.status==="failed")yield{type:"error",error:new C(k.InternalError,`Task ${$} failed`)};else if(G.status==="cancelled")yield{type:"error",error:new C(k.InternalError,`Task ${$} was cancelled`)};return}if(G.status==="input_required"){yield{type:"result",result:await this.getTaskResult({taskId:$},X,W)};return}let H=G.pollInterval??this._options?.defaultTaskPollInterval??1000;await new Promise((B)=>setTimeout(B,H)),W?.signal?.throwIfAborted()}}catch(J){yield{type:"error",error:J instanceof C?J:new C(k.InternalError,String(J))}}}request(Q,X,W){let{relatedRequestId:Y,resumptionToken:$,onresumptiontoken:J,task:G,relatedTask:H}=W??{};return new Promise((B,z)=>{let A=(w)=>{z(w)};if(!this._transport){A(Error("Not connected"));return}if(this._options?.enforceStrictCapabilities===!0)try{if(this.assertCapabilityForMethod(Q.method),G)this.assertTaskCapability(Q.method)}catch(w){A(w);return}W?.signal?.throwIfAborted();let D=this._requestMessageId++,L={...Q,jsonrpc:"2.0",id:D};if(W?.onprogress)this._progressHandlers.set(D,W.onprogress),L.params={...Q.params,_meta:{...Q.params?._meta||{},progressToken:D}};if(G)L.params={...L.params,task:G};if(H)L.params={...L.params,_meta:{...L.params?._meta||{},[k1]:H}};let K=(w)=>{this._responseHandlers.delete(D),this._progressHandlers.delete(D),this._cleanupTimeout(D),this._transport?.send({jsonrpc:"2.0",method:"notifications/cancelled",params:{requestId:D,reason:String(w)}},{relatedRequestId:Y,resumptionToken:$,onresumptiontoken:J}).catch((E)=>this._onerror(Error(`Failed to send cancellation: ${E}`)));let b=w instanceof C?w:new C(k.RequestTimeout,String(w));z(b)};this._responseHandlers.set(D,(w)=>{if(W?.signal?.aborted)return;if(w instanceof Error)return z(w);try{let b=Z1(X,w.result);if(!b.success)z(b.error);else B(b.data)}catch(b){z(b)}}),W?.signal?.addEventListener("abort",()=>{K(W?.signal?.reason)});let q=W?.timeout??_L,V=()=>K(C.fromError(k.RequestTimeout,"Request timed out",{timeout:q}));this._setupTimeout(D,q,W?.maxTotalTimeout,V,W?.resetTimeoutOnProgress??!1);let O=H?.taskId;if(O){let w=(b)=>{let E=this._responseHandlers.get(D);if(E)E(b);else this._onerror(Error(`Response handler missing for side-channeled request ${D}`))};this._requestResolvers.set(D,w),this._enqueueTaskMessage(O,{type:"request",message:L,timestamp:Date.now()}).catch((b)=>{this._cleanupTimeout(D),z(b)})}else this._transport.send(L,{relatedRequestId:Y,resumptionToken:$,onresumptiontoken:J}).catch((w)=>{this._cleanupTimeout(D),z(w)})})}async getTask(Q,X){return this.request({method:"tasks/get",params:Q},C9,X)}async getTaskResult(Q,X,W){return this.request({method:"tasks/result",params:Q},X,W)}async listTasks(Q,X){return this.request({method:"tasks/list",params:Q},k9,X)}async cancelTask(Q,X){return this.request({method:"tasks/cancel",params:Q},f7,X)}async notification(Q,X){if(!this._transport)throw Error("Not connected");this.assertNotificationCapability(Q.method);let W=X?.relatedTask?.taskId;if(W){let G={...Q,jsonrpc:"2.0",params:{...Q.params,_meta:{...Q.params?._meta||{},[k1]:X.relatedTask}}};await this._enqueueTaskMessage(W,{type:"notification",message:G,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 G={...Q,jsonrpc:"2.0"};if(X?.relatedTask)G={...G,params:{...G.params,_meta:{...G.params?._meta||{},[k1]:X.relatedTask}}};this._transport?.send(G,X).catch((H)=>this._onerror(H))});return}let J={...Q,jsonrpc:"2.0"};if(X?.relatedTask)J={...J,params:{...J.params,_meta:{...J.params?._meta||{},[k1]:X.relatedTask}}};await this._transport.send(J,X)}setRequestHandler(Q,X){let W=DW(Q);this.assertRequestHandlerCapability(W),this._requestHandlers.set(W,(Y,$)=>{let J=LW(Q,Y);return Promise.resolve(X(J,$))})}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 W=DW(Q);this._notificationHandlers.set(W,(Y)=>{let $=LW(Q,Y);return Promise.resolve(X($))})}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,W){if(!this._taskStore||!this._taskMessageQueue)throw Error("Cannot enqueue task message: taskStore and taskMessageQueue are not configured");let Y=this._options?.maxTaskQueueSize;await this._taskMessageQueue.enqueue(Q,X,W,Y)}async _clearTaskQueue(Q,X){if(this._taskMessageQueue){let W=await this._taskMessageQueue.dequeueAll(Q,X);for(let Y of W)if(Y.type==="request"&&nX(Y.message)){let $=Y.message.id,J=this._requestResolvers.get($);if(J)J(new C(k.InternalError,"Task cancelled or completed")),this._requestResolvers.delete($);else this._onerror(Error(`Resolver missing for request ${$} during task ${Q} cleanup`))}}}async _waitForTaskUpdate(Q,X){let W=this._options?.defaultTaskPollInterval??1000;try{let Y=await this._taskStore?.getTask(Q);if(Y?.pollInterval)W=Y.pollInterval}catch{}return new Promise((Y,$)=>{if(X.aborted){$(new C(k.InvalidRequest,"Request cancelled"));return}let J=setTimeout(Y,W);X.addEventListener("abort",()=>{clearTimeout(J),$(new C(k.InvalidRequest,"Request cancelled"))},{once:!0})})}requestTaskStore(Q,X){let W=this._taskStore;if(!W)throw Error("No task store configured");return{createTask:async(Y)=>{if(!Q)throw Error("No request provided");return await W.createTask(Y,Q.id,{method:Q.method,params:Q.params},X)},getTask:async(Y)=>{let $=await W.getTask(Y,X);if(!$)throw new C(k.InvalidParams,"Failed to retrieve task: Task not found");return $},storeTaskResult:async(Y,$,J)=>{await W.storeTaskResult(Y,$,J,X);let G=await W.getTask(Y,X);if(G){let H=N4.parse({method:"notifications/tasks/status",params:G});if(await this.notification(H),v1(G.status))this._cleanupTaskProgressHandler(Y)}},getTaskResult:(Y)=>{return W.getTaskResult(Y,X)},updateTaskStatus:async(Y,$,J)=>{let G=await W.getTask(Y,X);if(!G)throw new C(k.InvalidParams,`Task "${Y}" not found - it may have been cleaned up`);if(v1(G.status))throw new C(k.InvalidParams,`Cannot update task "${Y}" from terminal status "${G.status}" to "${$}". Terminal states (completed, failed, cancelled) cannot transition to other states.`);await W.updateTaskStatus(Y,$,J,X);let H=await W.getTask(Y,X);if(H){let B=N4.parse({method:"notifications/tasks/status",params:H});if(await this.notification(B),v1(H.status))this._cleanupTaskProgressHandler(Y)}},listTasks:(Y)=>{return W.listTasks(Y,X)}}}}function RJ(Q){return Q!==null&&typeof Q==="object"&&!Array.isArray(Q)}function IJ(Q,X){let W={...Q};for(let Y in X){let $=Y,J=X[$];if(J===void 0)continue;let G=W[$];if(RJ(G)&&RJ(J))W[$]={...G,...J};else W[$]=J}return W}var zB=TY(zY(),1),AB=TY(BB(),1);function Wj(){let Q=new zB.default({strict:!1,validateFormats:!0,validateSchema:!1,allErrors:!0});return AB.default(Q),Q}class OY{constructor(Q){this._ajv=Q??Wj()}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(W)=>{if(X(W))return{valid:!0,data:W,errorMessage:void 0};else return{valid:!1,data:void 0,errorMessage:this._ajv.errorsText(X.errors)}}}}class NY{constructor(Q){this._server=Q}requestStream(Q,X,W){return this._server.requestStream(Q,X,W)}createMessageStream(Q,X){let W=this._server.getClientCapabilities();if((Q.tools||Q.toolChoice)&&!W?.sampling?.tools)throw Error("Client does not support sampling tools capability.");if(Q.messages.length>0){let Y=Q.messages[Q.messages.length-1],$=Array.isArray(Y.content)?Y.content:[Y.content],J=$.some((z)=>z.type==="tool_result"),G=Q.messages.length>1?Q.messages[Q.messages.length-2]:void 0,H=G?Array.isArray(G.content)?G.content:[G.content]:[],B=H.some((z)=>z.type==="tool_use");if(J){if($.some((z)=>z.type!=="tool_result"))throw Error("The last message must contain only tool_result content if any is present");if(!B)throw Error("tool_result blocks are not matching any tool_use from the previous message")}if(B){let z=new Set(H.filter((D)=>D.type==="tool_use").map((D)=>D.id)),A=new Set($.filter((D)=>D.type==="tool_result").map((D)=>D.toolUseId));if(z.size!==A.size||![...z].every((D)=>A.has(D)))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},b4,X)}elicitInputStream(Q,X){let W=this._server.getClientCapabilities(),Y=Q.mode??"form";switch(Y){case"url":{if(!W?.elicitation?.url)throw Error("Client does not support url elicitation.");break}case"form":{if(!W?.elicitation?.form)throw Error("Client does not support form elicitation.");break}}let $=Y==="form"&&Q.mode===void 0?{...Q,mode:"form"}:Q;return this.requestStream({method:"elicitation/create",params:$},k6,X)}async getTask(Q,X){return this._server.getTask({taskId:Q},X)}async getTaskResult(Q,X,W){return this._server.getTaskResult({taskId:Q},X,W)}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 KB(Q,X,W){if(!Q)throw Error(`${W} does not support task creation (required for ${X})`);switch(X){case"tools/call":if(!Q.tools?.call)throw Error(`${W} does not support task creation for tools/call (required for ${X})`);break;default:break}}function DB(Q,X,W){if(!Q)throw Error(`${W} does not support task creation (required for ${X})`);switch(X){case"sampling/createMessage":if(!Q.sampling?.createMessage)throw Error(`${W} does not support task creation for sampling/createMessage (required for ${X})`);break;case"elicitation/create":if(!Q.elicitation?.create)throw Error(`${W} does not support task creation for elicitation/create (required for ${X})`);break;default:break}}class MY extends qW{constructor(Q,X){super(X);if(this._serverInfo=Q,this._loggingLevels=new Map,this.LOG_LEVEL_SEVERITY=new Map(j4.options.map((W,Y)=>[W,Y])),this.isMessageIgnored=(W,Y)=>{let $=this._loggingLevels.get(Y);return $?this.LOG_LEVEL_SEVERITY.get(W)<this.LOG_LEVEL_SEVERITY.get($):!1},this._capabilities=X?.capabilities??{},this._instructions=X?.instructions,this._jsonSchemaValidator=X?.jsonSchemaValidator??new OY,this.setRequestHandler(rX,(W)=>this._oninitialize(W)),this.setNotificationHandler(tX,()=>this.oninitialized?.()),this._capabilities.logging)this.setRequestHandler(YW,async(W,Y)=>{let $=Y.sessionId||Y.requestInfo?.headers["mcp-session-id"]||void 0,{level:J}=W.params,G=j4.safeParse(J);if(G.success)this._loggingLevels.set($,G.data);return{}})}get experimental(){if(!this._experimental)this._experimental={tasks:new NY(this)};return this._experimental}registerCapabilities(Q){if(this.transport)throw Error("Cannot register capabilities after connecting to transport");this._capabilities=IJ(this._capabilities,Q)}setRequestHandler(Q,X){let Y=T1(Q)?.method;if(!Y)throw Error("Schema is missing a method literal");let $;if(m0(Y)){let G=Y;$=G._zod?.def?.value??G.value}else{let G=Y;$=G._def?.value??G.value}if(typeof $!=="string")throw Error("Schema method literal must be a string");if($==="tools/call"){let G=async(H,B)=>{let z=Z1(T6,H);if(!z.success){let K=z.error instanceof Error?z.error.message:String(z.error);throw new C(k.InvalidParams,`Invalid tools/call request: ${K}`)}let{params:A}=z.data,D=await Promise.resolve(X(H,B));if(A.task){let K=Z1(C6,D);if(!K.success){let q=K.error instanceof Error?K.error.message:String(K.error);throw new C(k.InvalidParams,`Invalid task creation result: ${q}`)}return K.data}let L=Z1(u9,D);if(!L.success){let K=L.error instanceof Error?L.error.message:String(L.error);throw new C(k.InvalidParams,`Invalid tools/call result: ${K}`)}return L.data};return super.setRequestHandler(Q,G)}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){DB(this._clientCapabilities?.tasks?.requests,Q,"Client")}assertTaskHandlerCapability(Q){if(!this._capabilities)return;KB(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:C7.includes(X)?X:pX,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"},E9)}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 W=Q.messages[Q.messages.length-1],Y=Array.isArray(W.content)?W.content:[W.content],$=Y.some((B)=>B.type==="tool_result"),J=Q.messages.length>1?Q.messages[Q.messages.length-2]:void 0,G=J?Array.isArray(J.content)?J.content:[J.content]:[],H=G.some((B)=>B.type==="tool_use");if($){if(Y.some((B)=>B.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 B=new Set(G.filter((A)=>A.type==="tool_use").map((A)=>A.id)),z=new Set(Y.filter((A)=>A.type==="tool_result").map((A)=>A.toolUseId));if(B.size!==z.size||![...B].every((A)=>z.has(A)))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},$W,X);return this.request({method:"sampling/createMessage",params:Q},b4,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 Y=Q;return this.request({method:"elicitation/create",params:Y},k6,X)}case"form":{if(!this._clientCapabilities?.elicitation?.form)throw Error("Client does not support form elicitation.");let Y=Q.mode==="form"?Q:{...Q,mode:"form"},$=await this.request({method:"elicitation/create",params:Y},k6,X);if($.action==="accept"&&$.content&&Y.requestedSchema)try{let G=this._jsonSchemaValidator.getValidator(Y.requestedSchema)($.content);if(!G.valid)throw new C(k.InvalidParams,`Elicitation response content does not match requested schema: ${G.errorMessage}`)}catch(J){if(J instanceof C)throw J;throw new C(k.InternalError,`Error validating elicitation response: ${J instanceof Error?J.message:String(J)}`)}return $}}}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},JW,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 qB=Symbol.for("mcp.completable");function jY(Q){return!!Q&&typeof Q==="object"&&qB in Q}function VB(Q){return Q[qB]?.complete}var LB;(function(Q){Q.Completable="McpCompletable"})(LB||(LB={}));var Yj=/^[A-Za-z0-9._-]{1,128}$/;function $j(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(!Yj.test(Q)){let W=Q.split("").filter((Y)=>!/[A-Za-z0-9._-]/.test(Y)).filter((Y,$,J)=>J.indexOf(Y)===$);return X.push(`Tool name contains invalid characters: ${W.map((Y)=>`"${Y}"`).join(", ")}`,"Allowed characters are: A-Z, a-z, 0-9, underscore (_), dash (-), and dot (.)"),{isValid:!1,warnings:X}}return{isValid:!0,warnings:X}}function Jj(Q,X){if(X.length>0){console.warn(`Tool name validation warning for "${Q}":`);for(let W of X)console.warn(` - ${W}`);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 bY(Q){let X=$j(Q);return Jj(Q,X.warnings),X.isValid}class EY{constructor(Q){this._mcpServer=Q}registerToolTask(Q,X,W){let Y={taskSupport:"required",...X.execution};if(Y.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,Y,X._meta,W)}}class RY{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 MY(Q,X)}get experimental(){if(!this._experimental)this._experimental={tasks:new EY(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(l1(f9)),this.server.assertCanSetRequestHandler(l1(T6)),this.server.registerCapabilities({tools:{listChanged:!0}}),this.server.setRequestHandler(f9,()=>({tools:Object.entries(this._registeredTools).filter(([,Q])=>Q.enabled).map(([Q,X])=>{let W={name:Q,title:X.title,description:X.description,inputSchema:(()=>{let Y=I6(X.inputSchema);return Y?KW(Y,{strictUnions:!0,pipeStrategy:"input"}):Gj})(),annotations:X.annotations,execution:X.execution,_meta:X._meta};if(X.outputSchema){let Y=I6(X.outputSchema);if(Y)W.outputSchema=KW(Y,{strictUnions:!0,pipeStrategy:"output"})}return W})})),this.server.setRequestHandler(T6,async(Q,X)=>{try{let W=this._registeredTools[Q.params.name];if(!W)throw new C(k.InvalidParams,`Tool ${Q.params.name} not found`);if(!W.enabled)throw new C(k.InvalidParams,`Tool ${Q.params.name} disabled`);let Y=!!Q.params.task,$=W.execution?.taskSupport,J="createTask"in W.handler;if(($==="required"||$==="optional")&&!J)throw new C(k.InternalError,`Tool ${Q.params.name} has taskSupport '${$}' but was not registered with registerToolTask`);if($==="required"&&!Y)throw new C(k.MethodNotFound,`Tool ${Q.params.name} requires task augmentation (taskSupport: 'required')`);if($==="optional"&&!Y&&J)return await this.handleAutomaticTaskPolling(W,Q,X);let G=await this.validateToolInput(W,Q.params.arguments,Q.params.name),H=await this.executeToolHandler(W,G,X);if(Y)return H;return await this.validateToolOutput(W,H,Q.params.name),H}catch(W){if(W instanceof C){if(W.code===k.UrlElicitationRequired)throw W}return this.createToolError(W instanceof Error?W.message:String(W))}}),this._toolHandlersInitialized=!0}createToolError(Q){return{content:[{type:"text",text:Q}],isError:!0}}async validateToolInput(Q,X,W){if(!Q.inputSchema)return;let $=I6(Q.inputSchema)??Q.inputSchema,J=await F9($,X);if(!J.success){let G="error"in J?J.error:"Unknown error",H=U9(G);throw new C(k.InvalidParams,`Input validation error: Invalid arguments for tool ${W}: ${H}`)}return J.data}async validateToolOutput(Q,X,W){if(!Q.outputSchema)return;if(!("content"in X))return;if(X.isError)return;if(!X.structuredContent)throw new C(k.InvalidParams,`Output validation error: Tool ${W} has an output schema but no structured content was provided`);let Y=I6(Q.outputSchema),$=await F9(Y,X.structuredContent);if(!$.success){let J="error"in $?$.error:"Unknown error",G=U9(J);throw new C(k.InvalidParams,`Output validation error: Invalid structured content for tool ${W}: ${G}`)}}async executeToolHandler(Q,X,W){let Y=Q.handler;if("createTask"in Y){if(!W.taskStore)throw Error("No task store provided.");let J={...W,taskStore:W.taskStore};if(Q.inputSchema)return await Promise.resolve(Y.createTask(X,J));else return await Promise.resolve(Y.createTask(J))}if(Q.inputSchema)return await Promise.resolve(Y(X,W));else return await Promise.resolve(Y(W))}async handleAutomaticTaskPolling(Q,X,W){if(!W.taskStore)throw Error("No task store provided for task-capable tool.");let Y=await this.validateToolInput(Q,X.params.arguments,X.params.name),$=Q.handler,J={...W,taskStore:W.taskStore},G=Y?await Promise.resolve($.createTask(Y,J)):await Promise.resolve($.createTask(J)),H=G.task.taskId,B=G.task,z=B.pollInterval??5000;while(B.status!=="completed"&&B.status!=="failed"&&B.status!=="cancelled"){await new Promise((D)=>setTimeout(D,z));let A=await W.taskStore.getTask(H);if(!A)throw new C(k.InternalError,`Task ${H} not found during polling`);B=A}return await W.taskStore.getTaskResult(H)}setCompletionRequestHandler(){if(this._completionHandlerInitialized)return;this.server.assertCanSetRequestHandler(l1(l9)),this.server.registerCapabilities({completions:{}}),this.server.setRequestHandler(l9,async(Q)=>{switch(Q.params.ref.type){case"ref/prompt":return i7(Q),this.handlePromptCompletion(Q,Q.params.ref);case"ref/resource":return n7(Q),this.handleResourceCompletion(Q,Q.params.ref);default:throw new C(k.InvalidParams,`Invalid completion reference: ${Q.params.ref}`)}}),this._completionHandlerInitialized=!0}async handlePromptCompletion(Q,X){let W=this._registeredPrompts[X.name];if(!W)throw new C(k.InvalidParams,`Prompt ${X.name} not found`);if(!W.enabled)throw new C(k.InvalidParams,`Prompt ${X.name} disabled`);if(!W.argsSchema)return o4;let $=T1(W.argsSchema)?.[Q.params.argument.name];if(!jY($))return o4;let J=VB($);if(!J)return o4;let G=await J(Q.params.argument.value,Q.params.context);return UB(G)}async handleResourceCompletion(Q,X){let W=Object.values(this._registeredResourceTemplates).find((J)=>J.resourceTemplate.uriTemplate.toString()===X.uri);if(!W){if(this._registeredResources[X.uri])return o4;throw new C(k.InvalidParams,`Resource template ${Q.params.ref.uri} not found`)}let Y=W.resourceTemplate.completeCallback(Q.params.argument.name);if(!Y)return o4;let $=await Y(Q.params.argument.value,Q.params.context);return UB($)}setResourceRequestHandlers(){if(this._resourceHandlersInitialized)return;this.server.assertCanSetRequestHandler(l1(_9)),this.server.assertCanSetRequestHandler(l1(x9)),this.server.assertCanSetRequestHandler(l1(g9)),this.server.registerCapabilities({resources:{listChanged:!0}}),this.server.setRequestHandler(_9,async(Q,X)=>{let W=Object.entries(this._registeredResources).filter(([$,J])=>J.enabled).map(([$,J])=>({uri:$,name:J.name,...J.metadata})),Y=[];for(let $ of Object.values(this._registeredResourceTemplates)){if(!$.resourceTemplate.listCallback)continue;let J=await $.resourceTemplate.listCallback(X);for(let G of J.resources)Y.push({...$.metadata,...G})}return{resources:[...W,...Y]}}),this.server.setRequestHandler(x9,async()=>{return{resourceTemplates:Object.entries(this._registeredResourceTemplates).map(([X,W])=>({name:X,uriTemplate:W.resourceTemplate.uriTemplate.toString(),...W.metadata}))}}),this.server.setRequestHandler(g9,async(Q,X)=>{let W=new URL(Q.params.uri),Y=this._registeredResources[W.toString()];if(Y){if(!Y.enabled)throw new C(k.InvalidParams,`Resource ${W} disabled`);return Y.readCallback(W,X)}for(let $ of Object.values(this._registeredResourceTemplates)){let J=$.resourceTemplate.uriTemplate.match(W.toString());if(J)return $.readCallback(W,J,X)}throw new C(k.InvalidParams,`Resource ${W} not found`)}),this._resourceHandlersInitialized=!0}setPromptRequestHandlers(){if(this._promptHandlersInitialized)return;this.server.assertCanSetRequestHandler(l1(y9)),this.server.assertCanSetRequestHandler(l1(h9)),this.server.registerCapabilities({prompts:{listChanged:!0}}),this.server.setRequestHandler(y9,()=>({prompts:Object.entries(this._registeredPrompts).filter(([,Q])=>Q.enabled).map(([Q,X])=>{return{name:Q,title:X.title,description:X.description,arguments:X.argsSchema?Hj(X.argsSchema):void 0}})})),this.server.setRequestHandler(h9,async(Q,X)=>{let W=this._registeredPrompts[Q.params.name];if(!W)throw new C(k.InvalidParams,`Prompt ${Q.params.name} not found`);if(!W.enabled)throw new C(k.InvalidParams,`Prompt ${Q.params.name} disabled`);if(W.argsSchema){let Y=I6(W.argsSchema),$=await F9(Y,Q.params.arguments);if(!$.success){let H="error"in $?$.error:"Unknown error",B=U9(H);throw new C(k.InvalidParams,`Invalid arguments for prompt ${Q.params.name}: ${B}`)}let J=$.data,G=W.callback;return await Promise.resolve(G(J,X))}else{let Y=W.callback;return await Promise.resolve(Y(X))}}),this._promptHandlersInitialized=!0}resource(Q,X,...W){let Y;if(typeof W[0]==="object")Y=W.shift();let $=W[0];if(typeof X==="string"){if(this._registeredResources[X])throw Error(`Resource ${X} is already registered`);let J=this._createRegisteredResource(Q,void 0,X,Y,$);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),J}else{if(this._registeredResourceTemplates[Q])throw Error(`Resource template ${Q} is already registered`);let J=this._createRegisteredResourceTemplate(Q,void 0,X,Y,$);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),J}}registerResource(Q,X,W,Y){if(typeof X==="string"){if(this._registeredResources[X])throw Error(`Resource ${X} is already registered`);let $=this._createRegisteredResource(Q,W.title,X,W,Y);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),$}else{if(this._registeredResourceTemplates[Q])throw Error(`Resource template ${Q} is already registered`);let $=this._createRegisteredResourceTemplate(Q,W.title,X,W,Y);return this.setResourceRequestHandlers(),this.sendResourceListChanged(),$}}_createRegisteredResource(Q,X,W,Y,$){let J={name:Q,title:X,metadata:Y,readCallback:$,enabled:!0,disable:()=>J.update({enabled:!1}),enable:()=>J.update({enabled:!0}),remove:()=>J.update({uri:null}),update:(G)=>{if(typeof G.uri<"u"&&G.uri!==W){if(delete this._registeredResources[W],G.uri)this._registeredResources[G.uri]=J}if(typeof G.name<"u")J.name=G.name;if(typeof G.title<"u")J.title=G.title;if(typeof G.metadata<"u")J.metadata=G.metadata;if(typeof G.callback<"u")J.readCallback=G.callback;if(typeof G.enabled<"u")J.enabled=G.enabled;this.sendResourceListChanged()}};return this._registeredResources[W]=J,J}_createRegisteredResourceTemplate(Q,X,W,Y,$){let J={resourceTemplate:W,title:X,metadata:Y,readCallback:$,enabled:!0,disable:()=>J.update({enabled:!1}),enable:()=>J.update({enabled:!0}),remove:()=>J.update({name:null}),update:(B)=>{if(typeof B.name<"u"&&B.name!==Q){if(delete this._registeredResourceTemplates[Q],B.name)this._registeredResourceTemplates[B.name]=J}if(typeof B.title<"u")J.title=B.title;if(typeof B.template<"u")J.resourceTemplate=B.template;if(typeof B.metadata<"u")J.metadata=B.metadata;if(typeof B.callback<"u")J.readCallback=B.callback;if(typeof B.enabled<"u")J.enabled=B.enabled;this.sendResourceListChanged()}};this._registeredResourceTemplates[Q]=J;let G=W.uriTemplate.variableNames;if(Array.isArray(G)&&G.some((B)=>!!W.completeCallback(B)))this.setCompletionRequestHandler();return J}_createRegisteredPrompt(Q,X,W,Y,$){let J={title:X,description:W,argsSchema:Y===void 0?void 0:e1(Y),callback:$,enabled:!0,disable:()=>J.update({enabled:!1}),enable:()=>J.update({enabled:!0}),remove:()=>J.update({name:null}),update:(G)=>{if(typeof G.name<"u"&&G.name!==Q){if(delete this._registeredPrompts[Q],G.name)this._registeredPrompts[G.name]=J}if(typeof G.title<"u")J.title=G.title;if(typeof G.description<"u")J.description=G.description;if(typeof G.argsSchema<"u")J.argsSchema=e1(G.argsSchema);if(typeof G.callback<"u")J.callback=G.callback;if(typeof G.enabled<"u")J.enabled=G.enabled;this.sendPromptListChanged()}};if(this._registeredPrompts[Q]=J,Y){if(Object.values(Y).some((H)=>{let B=H instanceof y0?H._def?.innerType:H;return jY(B)}))this.setCompletionRequestHandler()}return J}_createRegisteredTool(Q,X,W,Y,$,J,G,H,B){bY(Q);let z={title:X,description:W,inputSchema:FB(Y),outputSchema:FB($),annotations:J,execution:G,_meta:H,handler:B,enabled:!0,disable:()=>z.update({enabled:!1}),enable:()=>z.update({enabled:!0}),remove:()=>z.update({name:null}),update:(A)=>{if(typeof A.name<"u"&&A.name!==Q){if(typeof A.name==="string")bY(A.name);if(delete this._registeredTools[Q],A.name)this._registeredTools[A.name]=z}if(typeof A.title<"u")z.title=A.title;if(typeof A.description<"u")z.description=A.description;if(typeof A.paramsSchema<"u")z.inputSchema=e1(A.paramsSchema);if(typeof A.outputSchema<"u")z.outputSchema=e1(A.outputSchema);if(typeof A.callback<"u")z.handler=A.callback;if(typeof A.annotations<"u")z.annotations=A.annotations;if(typeof A._meta<"u")z._meta=A._meta;if(typeof A.enabled<"u")z.enabled=A.enabled;this.sendToolListChanged()}};return this._registeredTools[Q]=z,this.setToolRequestHandlers(),this.sendToolListChanged(),z}tool(Q,...X){if(this._registeredTools[Q])throw Error(`Tool ${Q} is already registered`);let W,Y,$,J;if(typeof X[0]==="string")W=X.shift();if(X.length>1){let H=X[0];if(SY(H)){if(Y=X.shift(),X.length>1&&typeof X[0]==="object"&&X[0]!==null&&!SY(X[0]))J=X.shift()}else if(typeof H==="object"&&H!==null){if(Object.values(H).some((B)=>typeof B==="object"&&B!==null))throw Error(`Tool ${Q} expected a Zod schema or ToolAnnotations, but received an unrecognized object`);J=X.shift()}}let G=X[0];return this._createRegisteredTool(Q,void 0,W,Y,$,J,{taskSupport:"forbidden"},void 0,G)}registerTool(Q,X,W){if(this._registeredTools[Q])throw Error(`Tool ${Q} is already registered`);let{title:Y,description:$,inputSchema:J,outputSchema:G,annotations:H,_meta:B}=X;return this._createRegisteredTool(Q,Y,$,J,G,H,{taskSupport:"forbidden"},B,W)}prompt(Q,...X){if(this._registeredPrompts[Q])throw Error(`Prompt ${Q} is already registered`);let W;if(typeof X[0]==="string")W=X.shift();let Y;if(X.length>1)Y=X.shift();let $=X[0],J=this._createRegisteredPrompt(Q,void 0,W,Y,$);return this.setPromptRequestHandlers(),this.sendPromptListChanged(),J}registerPrompt(Q,X,W){if(this._registeredPrompts[Q])throw Error(`Prompt ${Q} is already registered`);let{title:Y,description:$,argsSchema:J}=X,G=this._createRegisteredPrompt(Q,Y,$,J,W);return this.setPromptRequestHandlers(),this.sendPromptListChanged(),G}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 Gj={type:"object",properties:{}};function wB(Q){return Q!==null&&typeof Q==="object"&&"parse"in Q&&typeof Q.parse==="function"&&"safeParse"in Q&&typeof Q.safeParse==="function"}function OB(Q){return"_def"in Q||"_zod"in Q||wB(Q)}function SY(Q){if(typeof Q!=="object"||Q===null)return!1;if(OB(Q))return!1;if(Object.keys(Q).length===0)return!0;return Object.values(Q).some(wB)}function FB(Q){if(!Q)return;if(SY(Q))return e1(Q);if(!OB(Q))throw Error("inputSchema must be a Zod schema or raw shape, received an unrecognized object");return Q}function Hj(Q){let X=T1(Q);if(!X)return[];return Object.entries(X).map(([W,Y])=>{let $=J7(Y),J=G7(Y);return{name:W,description:$,required:!J}})}function l1(Q){let W=T1(Q)?.method;if(!W)throw Error("Schema is missing a method literal");let Y=w9(W);if(typeof Y==="string")return Y;throw Error("Schema method literal must be a string")}function UB(Q){return{completion:{values:Q.slice(0,100),total:Q.length,hasMore:Q.length>100}}}var o4={completion:{values:[],hasMore:!1}};import MB from"process";class IY{append(Q){this._buffer=this._buffer?Buffer.concat([this._buffer,Q]):Q}readMessage(){if(!this._buffer)return null;let Q=this._buffer.indexOf(`
41
41
  `);if(Q===-1)return null;let X=this._buffer.toString("utf8",0,Q).replace(/\r$/,"");return this._buffer=this._buffer.subarray(Q+1),Bj(X)}clear(){this._buffer=void 0}}function Bj(Q){return y7.parse(JSON.parse(Q))}function NB(Q){return JSON.stringify(Q)+`
42
- `}class PY{constructor(Q=MB.stdin,X=MB.stdout){this._stdin=Q,this._stdout=X,this._readBuffer=new IY,this._started=!1,this._ondata=(W)=>{this._readBuffer.append(W),this.processReadBuffer()},this._onerror=(W)=>{this.onerror?.(W)}}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 W=NB(Q);if(this._stdout.write(W))X();else this._stdout.once("drain",X)})}}import{existsSync as Aj}from"fs";import bB from"path";var jB={name:"junso-browser",version:"0.2.3",description:"Standalone CloakBrowser host \u2014 runs fingerprinted stealth-Chromium per profile, exposes each over a token-authed CDP gateway + interactive viewport, and ships an optional MCP server so any client gets browser tools with no local browser.",type:"module",bin:{"junso-browser":"bin/junso-browser.js","junso-browser-mcp":"bin/junso-browser-mcp.js"},files:["bin","dist","README.md"],engines:{bun:">=1.2.0"},keywords:["browser","cloakbrowser","stealth","fingerprint","cdp","mcp","proxy","automation"],license:"UNLICENSED",scripts:{dev:"bun --watch src/server.ts",start:"bun src/server.ts",mcp:"bun src/mcp.ts",build:"bun build src/server.ts src/mcp.ts --target=bun --minify --outdir=dist","build:binary":"bun build src/server.ts --compile --outfile dist/junso-browser",typecheck:"bunx tsc --noEmit",test:"bun test",prepublishOnly:"bun run build"},dependencies:{"agent-browser":"^0.27.1",cloakbrowser:"^0.3.31"},devDependencies:{"@modelcontextprotocol/sdk":"^1.29.0","@types/bun":"latest",hono:"^4.6.0",typescript:"^5.6.0",zod:"3"}};var Kj=(process.env.JUNSO_BROWSER_URL??"http://127.0.0.1:8790").replace(/\/$/,""),EB=process.env.JUNSO_BROWSER_TOKEN??"",E0=process.env.JUNSO_BROWSER_PROFILE??"default";async function N1(Q,X,W){let Y=await fetch(`${Kj}${X}`,{method:Q,headers:{...W?{"content-type":"application/json"}:{},...EB?{authorization:`Bearer ${EB}`}:{}},body:W?JSON.stringify(W):void 0}),$=await Y.text(),J=$;try{J=$?JSON.parse($):null}catch{}if(!Y.ok){let G=(J&&typeof J==="object"&&"error"in J?J.error:void 0)??`junso-browser ${Y.status} ${Y.statusText}`;throw Error(G)}return J}async function RB(Q){try{await N1("POST","/api/profiles",{name:Q})}catch(X){if(!/already exists/i.test(String(X)))throw X}}var H6=null;async function Dj(){if(H6&&H6.id===E0)return H6.url;await RB(E0);let{cdpUrl:Q}=await N1("GET",`/api/profiles/${E0}/cdp`);return H6={id:E0,url:Q},Q}var Lj=process.platform==="win32"?"agent-browser.exe":"agent-browser";function qj(){let Q=import.meta.dir;for(let X=0;X<6;X++){let W=bB.join(Q,"node_modules",".bin",Lj);if(Aj(W))return W;let Y=bB.dirname(Q);if(Y===Q)break;Q=Y}return"agent-browser"}var Vj=qj();async function SB(Q,X=80,W=1500){if(!Q)return"";let Y=Q.getReader(),$=new TextDecoder,J="",G=Date.now()+W;try{while(Date.now()<G){let H=new Promise((z)=>setTimeout(()=>z(null),Math.min(X,G-Date.now()))),B=await Promise.race([Y.read(),H]);if(!B||B.done)break;J+=$.decode(B.value,{stream:!0})}}finally{Y.releaseLock()}return J}async function D1(Q,X={}){let W=await Dj(),Y=[Vj,"--cdp",W,...Q,...X.json?["--json"]:[]],$=Bun.spawn(Y,{env:{...process.env,AGENT_BROWSER_SESSION:`jbmcp-${E0}`},stdout:"pipe",stderr:"pipe"}),J=setTimeout(()=>$.kill(),X.timeoutMs??120000),G=await $.exited;clearTimeout(J);let[H,B]=await Promise.all([SB($.stdout),SB($.stderr,60,300)]);return{ok:G===0,stdout:H.trim(),stderr:B.trim()}}var M0=new RY({name:"junso-browser",version:jB.version},{instructions:["junso-browser drives fingerprinted stealth-Chromium profiles over CDP. Each profile is a","separate identity: its own cookies/logins, device fingerprint, AND exit IP (proxy). Switch","identities with browser_switch_profile; cookies persist per profile on disk, so once a profile","is logged in it stays logged in.","","LOGGING IN / ANTI-BOT SITES (e.g. Reddit) \u2014 follow this recipe, because login success is gated","by EXIT-IP REPUTATION, not by the browser (the stealth + fingerprint are already handled):","1. The profile needs a RESIDENTIAL or MOBILE proxy. Run browser_exit_ip \u2014 `kind` should be"," 'mobile' or 'residential'. 'hosting'/'proxy' (datacenter) IPs get flagged; expect failures."," Also check `tzMismatch` is false (fingerprint tz matches the IP \u2014 auto-set on create/set_proxy).","2. Attempt the login ONCE: browser_fill the username + password, then click submit. Note many"," sites' submit buttons have NO type=submit \u2014 match by visible text/role via browser_snapshot.","3. If it fails (reCAPTCHA / 'something went wrong' / a 4xx), the cause is almost always the IP,"," not your input. Call browser_rotate_proxy to draw a FRESH exit IP, then retry from step 1.","4. ONE login attempt per IP. Do NOT retry on the same IP \u2014 repeated failures degrade that IP's"," score. Stop as soon as a login succeeds.","Datacenter/no proxy \u2192 logins will likely fail no matter what; that is an IP problem, not a bug."].join(`
42
+ `}class PY{constructor(Q=MB.stdin,X=MB.stdout){this._stdin=Q,this._stdout=X,this._readBuffer=new IY,this._started=!1,this._ondata=(W)=>{this._readBuffer.append(W),this.processReadBuffer()},this._onerror=(W)=>{this.onerror?.(W)}}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 W=NB(Q);if(this._stdout.write(W))X();else this._stdout.once("drain",X)})}}import{existsSync as Aj}from"fs";import bB from"path";var jB={name:"junso-browser",version:"0.2.6",description:"Standalone CloakBrowser host \u2014 runs fingerprinted stealth-Chromium per profile, exposes each over a token-authed CDP gateway + interactive viewport, and ships an optional MCP server so any client gets browser tools with no local browser.",type:"module",bin:{"junso-browser":"bin/junso-browser.js","junso-browser-mcp":"bin/junso-browser-mcp.js"},files:["bin","dist","README.md"],engines:{bun:">=1.2.0"},keywords:["browser","cloakbrowser","stealth","fingerprint","cdp","mcp","proxy","automation"],license:"UNLICENSED",scripts:{dev:"bun --watch src/server.ts",start:"bun src/server.ts",mcp:"bun src/mcp.ts",build:"bun build src/server.ts src/mcp.ts --target=bun --minify --outdir=dist","build:binary":"bun build src/server.ts --compile --outfile dist/junso-browser",typecheck:"bunx tsc --noEmit",test:"bun test",prepublishOnly:"bun run build"},dependencies:{"agent-browser":"^0.27.1",cloakbrowser:"^0.3.31"},devDependencies:{"@modelcontextprotocol/sdk":"^1.29.0","@types/bun":"latest",hono:"^4.6.0",typescript:"^5.6.0",zod:"3"}};var Kj=(process.env.JUNSO_BROWSER_URL??"http://127.0.0.1:8790").replace(/\/$/,""),EB=process.env.JUNSO_BROWSER_TOKEN??"",E0=process.env.JUNSO_BROWSER_PROFILE??"default";async function N1(Q,X,W){let Y=await fetch(`${Kj}${X}`,{method:Q,headers:{...W?{"content-type":"application/json"}:{},...EB?{authorization:`Bearer ${EB}`}:{}},body:W?JSON.stringify(W):void 0}),$=await Y.text(),J=$;try{J=$?JSON.parse($):null}catch{}if(!Y.ok){let G=(J&&typeof J==="object"&&"error"in J?J.error:void 0)??`junso-browser ${Y.status} ${Y.statusText}`;throw Error(G)}return J}async function RB(Q){try{await N1("POST","/api/profiles",{name:Q})}catch(X){if(!/already exists/i.test(String(X)))throw X}}var H6=null;async function Dj(){if(H6&&H6.id===E0)return H6.url;await RB(E0);let{cdpUrl:Q}=await N1("GET",`/api/profiles/${E0}/cdp`);return H6={id:E0,url:Q},Q}var Lj=process.platform==="win32"?"agent-browser.exe":"agent-browser";function qj(){let Q=import.meta.dir;for(let X=0;X<6;X++){let W=bB.join(Q,"node_modules",".bin",Lj);if(Aj(W))return W;let Y=bB.dirname(Q);if(Y===Q)break;Q=Y}return"agent-browser"}var Vj=qj();async function SB(Q,X=80,W=1500){if(!Q)return"";let Y=Q.getReader(),$=new TextDecoder,J="",G=Date.now()+W;try{while(Date.now()<G){let H=new Promise((z)=>setTimeout(()=>z(null),Math.min(X,G-Date.now()))),B=await Promise.race([Y.read(),H]);if(!B||B.done)break;J+=$.decode(B.value,{stream:!0})}}finally{Y.releaseLock()}return J}async function D1(Q,X={}){let W=await Dj(),Y=[Vj,"--cdp",W,...Q,...X.json?["--json"]:[]],$=Bun.spawn(Y,{env:{...process.env,AGENT_BROWSER_SESSION:`jbmcp-${E0}`},stdout:"pipe",stderr:"pipe"}),J=setTimeout(()=>$.kill(),X.timeoutMs??120000),G=await $.exited;clearTimeout(J);let[H,B]=await Promise.all([SB($.stdout),SB($.stderr,60,300)]);return{ok:G===0,stdout:H.trim(),stderr:B.trim()}}var M0=new RY({name:"junso-browser",version:jB.version},{instructions:["junso-browser drives fingerprinted stealth-Chromium profiles over CDP. Each profile is a","separate identity: its own cookies/logins, device fingerprint, AND exit IP (proxy). Switch","identities with browser_switch_profile; cookies persist per profile on disk, so once a profile","is logged in it stays logged in.","","LOGGING IN / ANTI-BOT SITES (e.g. Reddit) \u2014 follow this recipe, because login success is gated","by EXIT-IP REPUTATION, not by the browser (the stealth + fingerprint are already handled):","0. Create the profile (browser_create_profile) with a RESIDENTIAL or MOBILE proxy and OMIT"," timezone/locale \u2014 they're auto-matched to the proxy's exit-IP region. Then verify with"," browser_exit_ip before attempting a login.","1. The profile needs a RESIDENTIAL or MOBILE proxy. Run browser_exit_ip \u2014 `kind` should be"," 'mobile' or 'residential'. 'hosting'/'proxy' (datacenter) IPs get flagged; expect failures."," Also check `tzMismatch` is false (fingerprint tz matches the IP \u2014 auto-set on create/set_proxy).","2. Attempt the login ONCE: browser_fill the username + password, then click submit. Note many"," sites' submit buttons have NO type=submit \u2014 match by visible text/role via browser_snapshot.","3. If it fails (reCAPTCHA / 'something went wrong' / a 4xx), the cause is almost always the IP,"," not your input. Call browser_rotate_proxy to draw a FRESH exit IP, then retry from step 1.","4. ONE login attempt per IP. Do NOT retry on the same IP \u2014 repeated failures degrade that IP's"," score. Stop as soon as a login succeeds.","Datacenter/no proxy \u2192 logins will likely fail no matter what; that is an IP problem, not a bug."].join(`
43
43
  `)}),S0=(Q)=>({content:[{type:"text",text:typeof Q==="string"?Q:JSON.stringify(Q,null,2)}]}),J0=(Q)=>({content:[{type:"text",text:`Error: ${Q instanceof Error?Q.message:String(Q)}`}],isError:!0}),M1="Drives the active junso-browser profile (a fingerprinted stealth browser) over CDP. Switch identity with browser_switch_profile.";M0.tool("browser_status","Current page URL + title + the active profile (browser identity). "+M1,{},async()=>{try{let[Q,X]=await Promise.all([D1(["get","url"]),D1(["get","title"])]);return S0({profile:E0,url:Q.stdout,title:X.stdout})}catch(Q){return J0(Q)}});M0.tool("browser_navigate","Navigate the browser to a URL. "+M1,{url:t.string().describe("Absolute URL, e.g. https://www.reddit.com")},async({url:Q})=>{try{let X=await D1(["open",Q],{timeoutMs:120000});return X.ok?S0({navigated:Q}):J0(X.stderr||X.stdout||"navigate failed")}catch(X){return J0(X)}});M0.tool("browser_snapshot","Accessibility snapshot with @ref handles (decide what to click/fill). @refs go stale on any "+"page change \u2014 re-snapshot before reusing them. "+M1,{interactive:t.boolean().optional().describe("Only interactive elements (default true)")},async({interactive:Q})=>{try{let X=await D1(["snapshot",...Q===!1?[]:["-i"]],{timeoutMs:60000});return X.ok?S0(X.stdout):J0(X.stderr||"snapshot failed")}catch(X){return J0(X)}});var P8=t.string().describe("An @ref like '@e12' or a CSS selector"),n6=(Q,X,W,Y)=>M0.tool(Q,Y+" "+M1,W,async($)=>{try{let J=await D1(X($));return J.ok?S0({ok:!0,stdout:J.stdout}):J0(J.stderr||J.stdout||`${Q} failed`)}catch(J){return J0(J)}});n6("browser_click",(Q)=>["click",Q.target],{target:P8},"Click an @ref or CSS selector.");n6("browser_fill",(Q)=>["fill",Q.target,Q.text],{target:P8,text:t.string()},"Clear + fill an input.");n6("browser_type",(Q)=>["type",Q.target,Q.text],{target:P8,text:t.string()},"Type into an element WITHOUT clearing.");n6("browser_press",(Q)=>["press",Q.key],{key:t.string().describe("e.g. Enter, Tab, Control+a")},"Press a key/combo.");n6("browser_select",(Q)=>["select",Q.target,Q.value],{target:P8,value:t.string()},"Select a dropdown option by value.");n6("browser_scroll",(Q)=>["scroll",Q.direction??"down",...Q.amount?[Q.amount]:[]],{direction:t.enum(["up","down","left","right"]).optional(),amount:t.string().optional()},"Scroll the page.");M0.tool("browser_wait","Wait for a selector, text, or ms before continuing (after an async navigation/click). "+M1,{selector:t.string().optional(),text:t.string().optional(),ms:t.number().optional()},async({selector:Q,text:X,ms:W})=>{try{let Y=Q?["wait",Q]:X?["wait","--text",X]:W?["wait",String(W)]:null;if(!Y)return J0("provide selector, text, or ms");let $=await D1(Y,{timeoutMs:60000});return $.ok?S0({waited:!0}):J0($.stderr||"wait failed")}catch(Y){return J0(Y)}});M0.tool("browser_read","Read visible text (whole page if no selector). "+M1,{selector:t.string().optional()},async({selector:Q})=>{try{let X=await D1(Q?["get","text",Q]:["get","text","body"]);return X.ok?S0(X.stdout):J0(X.stderr||"read failed")}catch(X){return J0(X)}});M0.tool("browser_eval","Run a JavaScript expression in the page and return its result. "+M1,{expression:t.string()},async({expression:Q})=>{try{let X=await D1(["eval",Q],{timeoutMs:30000});return X.ok?S0(X.stdout):J0(X.stderr||"eval failed")}catch(X){return J0(X)}});M0.tool("browser_screenshot","Capture a screenshot; returns the saved path. "+M1,{},async()=>{try{let Q=`shot-${Date.now()}.png`,X=await D1(["screenshot",Q],{timeoutMs:60000});return X.ok?S0({path:Q}):J0(X.stderr||"screenshot failed")}catch(Q){return J0(Q)}});M0.tool("browser_exec","Run ANY agent-browser command on the active profile (escape hatch): e.g. ['tab','new',url], ['hover','@e5'], ['network','requests']. Returns {ok,stdout,stderr}. "+M1,{args:t.array(t.string())},async({args:Q})=>{try{if(Q[0]==="close")return J0("`close` is not allowed \u2014 junso-browser owns the browser lifecycle");return S0(await D1(Q))}catch(X){return J0(X)}});M0.tool("browser_list_profiles","List the junso-browser identities (each has its own cookies/login + device fingerprint). The active one is what these tools drive.",{},async()=>{try{let{profiles:Q}=await N1("GET","/api/profiles");return S0({active:E0,profiles:Q})}catch(Q){return J0(Q)}});M0.tool("browser_switch_profile","Switch the active browser identity (creates it on the host if new). Subsequent browser tools drive that profile's fingerprinted browser.",{name:t.string()},async({name:Q})=>{try{return E0=Q.trim(),H6=null,await RB(E0),S0({active:E0})}catch(X){return J0(X)}});M0.tool("browser_create_profile","Create a browser identity with its own DEVICE FINGERPRINT + PROXY (exit IP). Each profile is a separate, isolated browser (own cookies/login, own canvas/WebGL/timezone, own IP). Use a residential/mobile proxy to avoid datacenter-IP blocks.",{name:t.string().describe("Profile id/name, e.g. 'acct-a'"),proxy:t.string().optional().describe("http://user:pass@host:port or socks5://user:pass@host:port"),platform:t.enum(["windows","macos"]).optional().describe("OS to present (default windows)"),timezone:t.string().optional().describe("IANA tz, e.g. America/New_York. Omit to AUTO-match the proxy's exit-IP region."),locale:t.string().optional().describe("BCP-47, e.g. en-US. Omit to AUTO-match the proxy's country."),seed:t.number().optional().describe("Fingerprint seed (same seed = same device); omit for random")},async({name:Q,proxy:X,platform:W,timezone:Y,locale:$,seed:J})=>{try{let G=await N1("POST","/api/profiles",{name:Q,proxy:X,fingerprint:{seed:J,platform:W,timezone:Y,locale:$}});return S0(G)}catch(G){return J0(G)}});M0.tool("browser_set_proxy","Set (or clear) a profile's PROXY / exit IP. Empty/omitted `proxy` clears it (direct connection). The fingerprint timezone/locale are AUTO-re-synced to the new proxy's exit-IP region. The proxy binds at browser launch, so this restarts that profile's browser to apply.",{name:t.string().optional().describe("Profile (default: the active one)"),proxy:t.string().optional().describe("http://user:pass@host:port or socks5://\u2026 ; empty/omit to clear")},async({name:Q,proxy:X})=>{try{let W=Q?.trim()||E0,Y=X&&X.trim()?X.trim():null,{profile:$}=await N1("PATCH",`/api/profiles/${W}`,{proxy:Y});if(await N1("POST",`/api/profiles/${W}/stop`).catch(()=>{}),W===E0)H6=null;return S0({profile:W,proxy:$.proxy??null})}catch(W){return J0(W)}});M0.tool("browser_get_proxy","Show the proxy / exit IP CONFIGURED for a profile (default: the active one), password-masked. For the LIVE exit IP + reputation, use browser_exit_ip.",{name:t.string().optional().describe("Profile (default: the active one)")},async({name:Q})=>{try{let X=Q?.trim()||E0,{profile:W}=await N1("GET",`/api/profiles/${X}`);return S0({profile:X,proxy:W.proxy??null})}catch(X){return J0(X)}});M0.tool("browser_exit_ip","LIVE exit IP + geo HEALTH of a profile's proxy \u2014 the IP/identity sites actually see: "+"{ip,country,isp,kind,timezone,tzMismatch}. `kind` = mobile|residential|hosting|proxy (mobile/residential pass anti-bot/login checks; hosting/proxy get flagged). `tzMismatch`=true means the fingerprint timezone no longer matches the IP. Check this before/after a login or when anti-bot checks fail.",{name:t.string().optional().describe("Profile (default: the active one)")},async({name:Q})=>{try{let X=Q?.trim()||E0;return S0(await N1("GET",`/api/profiles/${X}/exit-ip`))}catch(X){return J0(X)}});M0.tool("browser_rotate_proxy","Draw a FRESH exit IP for a profile by rotating its proxy's sticky session id, and auto-re-sync "+"the fingerprint (timezone/locale) to the new IP. Use when the current IP is flagged \u2014 e.g. a "+"login or anti-bot check fails (success is gated by exit-IP reputation, not the browser). Restarts that profile's browser on next use. Errors if the proxy has no rotatable session id.",{name:t.string().optional().describe("Profile (default: the active one)")},async({name:Q})=>{try{let X=Q?.trim()||E0,W=await N1("POST",`/api/profiles/${X}/rotate-proxy`);if(X===E0)H6=null;return S0({profile:X,exitIp:W.exitIp??null,country:W.country??null,isp:W.isp??null,kind:W.kind??null})}catch(X){return J0(X)}});await M0.connect(new PY);
package/dist/server.js CHANGED
@@ -1,7 +1,7 @@
1
1
  // @bun
2
2
  var Pn=Object.create;var{getPrototypeOf:Tn,defineProperty:et,getOwnPropertyNames:jn}=Object;var Jn=Object.prototype.hasOwnProperty;function mn(t){return this[t]}var _n,Wn,zi=(t,n,i)=>{var c=t!=null&&typeof t==="object";if(c){var l=n?_n??=new WeakMap:Wn??=new WeakMap,s=l.get(t);if(s)return s}i=t!=null?Pn(Tn(t)):{};let f=n||!t||!t.__esModule?et(i,"default",{value:t,enumerable:!0}):i;for(let e of jn(t))if(!Jn.call(f,e))et(f,e,{get:mn.bind(t,e),enumerable:!0});if(c)l.set(t,f);return f};var Ki=(t,n)=>()=>(n||t((n={exports:{}}).exports,n),n.exports);var Xn=(t)=>t;function Gn(t,n){this[t]=Xn.bind(null,n)}var gi=(t,n)=>{for(var i in n)et(t,i,{get:n[i],enumerable:!0,configurable:!0,set:Gn.bind(n,i)})};var ot=(t,n,i)=>{return(c,l)=>{let s=-1;return f(0);async function f(e){if(e<=s)throw Error("next() called multiple times");s=e;let b,o=!1,r;if(t[e])r=t[e][0][0],c.req.routeIndex=e;else r=e===t.length&&l||void 0;if(r)try{b=await r(c,()=>f(e+1))}catch(x){if(x instanceof Error&&n)c.error=x,b=await n(x,c),o=!0;else throw x}else if(c.finalized===!1&&i)b=await i(c);if(b&&(c.finalized===!1||o))c.res=b;return c}}};var Xt=Symbol();var Gt=async(t,n=Object.create(null))=>{let{all:i=!1,dot:c=!1}=n,s=(t instanceof F?t.raw.headers:t.headers).get("Content-Type");if(s?.startsWith("multipart/form-data")||s?.startsWith("application/x-www-form-urlencoded"))return Qn(t,{all:i,dot:c});return{}};async function Qn(t,n){let i=await t.formData();if(i)return Yn(i,n);return{}}function Yn(t,n){let i=Object.create(null);if(t.forEach((c,l)=>{if(!(n.all||l.endsWith("[]")))i[l]=c;else Vn(i,l,c)}),n.dot)Object.entries(i).forEach(([c,l])=>{if(c.includes("."))wn(i,c,l),delete i[c]});return i}var Vn=(t,n,i)=>{if(t[n]!==void 0)if(Array.isArray(t[n]))t[n].push(i);else t[n]=[t[n],i];else if(!n.endsWith("[]"))t[n]=i;else t[n]=[i]},wn=(t,n,i)=>{if(/(?:^|\.)__proto__\./.test(n))return;let c=t,l=n.split(".");l.forEach((s,f)=>{if(f===l.length-1)c[s]=i;else{if(!c[s]||typeof c[s]!=="object"||Array.isArray(c[s])||c[s]instanceof File)c[s]=Object.create(null);c=c[s]}})};var xt=(t)=>{let n=t.split("/");if(n[0]==="")n.shift();return n},Qt=(t)=>{let{groups:n,path:i}=An(t),c=xt(i);return Zn(c,n)},An=(t)=>{let n=[];return t=t.replace(/\{[^}]+\}/g,(i,c)=>{let l=`@${c}`;return n.push([l,i]),l}),{groups:n,path:t}},Zn=(t,n)=>{for(let i=n.length-1;i>=0;i--){let[c]=n[i];for(let l=t.length-1;l>=0;l--)if(t[l].includes(c)){t[l]=t[l].replace(c,n[i][1]);break}}return t},D={},Yt=(t,n)=>{if(t==="*")return"*";let i=t.match(/^\:([^\{\}]+)(?:\{(.+)\})?$/);if(i){let c=`${t}#${n}`;if(!D[c])if(i[2])D[c]=n&&n[0]!==":"&&n[0]!=="*"?[c,i[1],new RegExp(`^${i[2]}(?=/${n})`)]:[t,i[1],new RegExp(`^${i[2]}$`)];else D[c]=[t,i[1],!0];return D[c]}return null},C=(t,n)=>{try{return n(t)}catch{return t.replace(/(?:%[0-9A-Fa-f]{2})+/g,(i)=>{try{return n(i)}catch{return i}})}},Mn=(t)=>C(t,decodeURI),ut=(t)=>{let n=t.url,i=n.indexOf("/",n.indexOf(":")+4),c=i;for(;c<n.length;c++){let l=n.charCodeAt(c);if(l===37){let s=n.indexOf("?",c),f=n.indexOf("#",c),e=s===-1?f===-1?void 0:f:f===-1?s:Math.min(s,f),b=n.slice(i,e);return Mn(b.includes("%25")?b.replace(/%25/g,"%2525"):b)}else if(l===63||l===35)break}return n.slice(i,c)};var Vt=(t)=>{let n=ut(t);return n.length>1&&n.at(-1)==="/"?n.slice(0,-1):n},g=(t,n,...i)=>{if(i.length)n=g(n,...i);return`${t?.[0]==="/"?"":"/"}${t}${n==="/"?"":`${t?.at(-1)==="/"?"":"/"}${n?.[0]==="/"?n.slice(1):n}`}`},H=(t)=>{if(t.charCodeAt(t.length-1)!==63||!t.includes(":"))return null;let n=t.split("/"),i=[],c="";return n.forEach((l)=>{if(l!==""&&!/\:/.test(l))c+="/"+l;else if(/\:/.test(l))if(/\?/.test(l)){if(i.length===0&&c==="")i.push("/");else i.push(c);let s=l.replace("?","");c+="/"+s,i.push(c)}else c+="/"+l}),i.filter((l,s,f)=>f.indexOf(l)===s)},rt=(t)=>{if(!/[%+]/.test(t))return t;if(t.indexOf("+")!==-1)t=t.replace(/\+/g," ");return t.indexOf("%")!==-1?C(t,dt):t},wt=(t,n,i)=>{let c;if(!i&&n&&!/[%+]/.test(n)){let f=t.indexOf("?",8);if(f===-1)return;if(!t.startsWith(n,f+1))f=t.indexOf(`&${n}`,f+1);while(f!==-1){let e=t.charCodeAt(f+n.length+1);if(e===61){let b=f+n.length+2,o=t.indexOf("&",b);return rt(t.slice(b,o===-1?void 0:o))}else if(e==38||isNaN(e))return"";f=t.indexOf(`&${n}`,f+1)}if(c=/[%+]/.test(t),!c)return}let l={};c??=/[%+]/.test(t);let s=t.indexOf("?",8);while(s!==-1){let f=t.indexOf("&",s+1),e=t.indexOf("=",s);if(e>f&&f!==-1)e=-1;let b=t.slice(s+1,e===-1?f===-1?void 0:f:e);if(c)b=rt(b);if(s=f,b==="")continue;let o;if(e===-1)o="";else if(o=t.slice(e+1,f===-1?void 0:f),c)o=rt(o);if(i){if(!(l[b]&&Array.isArray(l[b])))l[b]=[];l[b].push(o)}else l[b]??=o}return n?l[n]:l},At=wt,Zt=(t,n)=>{return wt(t,n,!0)},dt=decodeURIComponent;var Mt=(t)=>C(t,dt),F=class{raw;#t;#n;routeIndex=0;path;bodyCache={};constructor(t,n="/",i=[[]]){this.raw=t,this.path=n,this.#n=i,this.#t={}}param(t){return t?this.#i(t):this.#s()}#i(t){let n=this.#n[0][this.routeIndex][1][t],i=this.#l(n);return i&&/\%/.test(i)?Mt(i):i}#s(){let t={},n=Object.keys(this.#n[0][this.routeIndex][1]);for(let i of n){let c=this.#l(this.#n[0][this.routeIndex][1][i]);if(c!==void 0)t[i]=/\%/.test(c)?Mt(c):c}return t}#l(t){return this.#n[1]?this.#n[1][t]:t}query(t){return At(this.url,t)}queries(t){return Zt(this.url,t)}header(t){if(t)return this.raw.headers.get(t)??void 0;let n={};return this.raw.headers.forEach((i,c)=>{n[c]=i}),n}async parseBody(t){return Gt(this,t)}#c=(t)=>{let{bodyCache:n,raw:i}=this,c=n[t];if(c)return c;let l=Object.keys(n)[0];if(l)return n[l].then((s)=>{if(l==="json")s=JSON.stringify(s);return new Response(s)[t]()});return n[t]=i[t]()};json(){return this.#c("text").then((t)=>JSON.parse(t))}text(){return this.#c("text")}arrayBuffer(){return this.#c("arrayBuffer")}bytes(){return this.#c("arrayBuffer").then((t)=>new Uint8Array(t))}blob(){return this.#c("blob")}formData(){return this.#c("formData")}addValidatedData(t,n){this.#t[t]=n}valid(t){return this.#t[t]}get url(){return this.raw.url}get method(){return this.raw.method}get[Xt](){return this.#n}get matchedRoutes(){return this.#n[0].map(([[,t]])=>t)}get routePath(){return this.#n[0].map(([[,t]])=>t)[this.routeIndex].path}};var It={Stringify:1,BeforeStream:2,Stream:3},In=(t,n)=>{let i=new String(t);return i.isEscaped=!0,i.callbacks=n,i};var Et=async(t,n,i,c,l)=>{if(typeof t==="object"&&!(t instanceof String)){if(!(t instanceof Promise))t=t.toString();if(t instanceof Promise)t=await t}let s=t.callbacks;if(!s?.length)return Promise.resolve(t);if(l)l[0]+=t;else l=[t];let f=Promise.all(s.map((e)=>e({phase:n,buffer:l,context:c}))).then((e)=>Promise.all(e.filter(Boolean).map((b)=>Et(b,n,!1,c,l))).then(()=>l[0]));if(i)return In(await f,s);else return f};var Un="text/plain; charset=UTF-8",Ot=(t,n)=>{return{"Content-Type":t,...n}},Y=(t,n)=>new Response(t,n),yt=class{#t;#n;env={};#i;finalized=!1;error;#s;#l;#c;#r;#e;#o;#b;#x;#u;constructor(t,n){if(this.#t=t,n)this.#l=n.executionCtx,this.env=n.env,this.#o=n.notFoundHandler,this.#u=n.path,this.#x=n.matchResult}get req(){return this.#n??=new F(this.#t,this.#u,this.#x),this.#n}get event(){if(this.#l&&"respondWith"in this.#l)return this.#l;else throw Error("This context has no FetchEvent")}get executionCtx(){if(this.#l)return this.#l;else throw Error("This context has no ExecutionContext")}get res(){return this.#c||=Y(null,{headers:this.#b??=new Headers})}set res(t){if(this.#c&&t){t=Y(t.body,t);for(let[n,i]of this.#c.headers.entries()){if(n==="content-type")continue;if(n==="set-cookie"){let c=this.#c.headers.getSetCookie();t.headers.delete("set-cookie");for(let l of c)t.headers.append("set-cookie",l)}else t.headers.set(n,i)}}this.#c=t,this.finalized=!0}render=(...t)=>{return this.#e??=(n)=>this.html(n),this.#e(...t)};setLayout=(t)=>this.#r=t;getLayout=()=>this.#r;setRenderer=(t)=>{this.#e=t};header=(t,n,i)=>{if(this.finalized)this.#c=Y(this.#c.body,this.#c);let c=this.#c?this.#c.headers:this.#b??=new Headers;if(n===void 0)c.delete(t);else if(i?.append)c.append(t,n);else c.set(t,n)};status=(t)=>{this.#s=t};set=(t,n)=>{this.#i??=new Map,this.#i.set(t,n)};get=(t)=>{return this.#i?this.#i.get(t):void 0};get var(){if(!this.#i)return{};return Object.fromEntries(this.#i)}#f(t,n,i){let c=this.#c?new Headers(this.#c.headers):this.#b??new Headers;if(typeof n==="object"&&"headers"in n){let s=n.headers instanceof Headers?n.headers:new Headers(n.headers);for(let[f,e]of s)if(f.toLowerCase()==="set-cookie")c.append(f,e);else c.set(f,e)}if(i)for(let[s,f]of Object.entries(i))if(typeof f==="string")c.set(s,f);else{c.delete(s);for(let e of f)c.append(s,e)}let l=typeof n==="number"?n:n?.status??this.#s;return Y(t,{status:l,headers:c})}newResponse=(...t)=>this.#f(...t);body=(t,n,i)=>this.#f(t,n,i);text=(t,n,i)=>{return!this.#b&&!this.#s&&!n&&!i&&!this.finalized?new Response(t):this.#f(t,n,Ot(Un,i))};json=(t,n,i)=>{return this.#f(JSON.stringify(t),n,Ot("application/json",i))};html=(t,n,i)=>{let c=(l)=>this.#f(l,n,Ot("text/html; charset=UTF-8",i));return typeof t==="object"?Et(t,It.Stringify,!1,{}).then(c):c(t)};redirect=(t,n)=>{let i=String(t);return this.header("Location",!/[^\x00-\xFF]/.test(i)?i:encodeURI(i)),this.newResponse(null,n??302)};notFound=()=>{return this.#o??=()=>Y(),this.#o(this)}};var d="ALL",Ut="all",Rt=["get","post","put","delete","options","patch"],p="Can not add a route since the matcher is already built.",h=class extends Error{};var kt="__COMPOSED_HANDLER";var Rn=(t)=>{return t.text("404 Not Found",404)},Ft=(t,n)=>{if("getResponse"in t){let i=t.getResponse();return n.newResponse(i.body,i)}return console.error(t),n.text("Internal Server Error",500)},Dt=class t{get;post;put;delete;options;patch;all;on;use;router;getPath;_basePath="/";#t="/";routes=[];constructor(n={}){[...Rt,Ut].forEach((s)=>{this[s]=(f,...e)=>{if(typeof f==="string")this.#t=f;else this.#s(s,this.#t,f);return e.forEach((b)=>{this.#s(s,this.#t,b)}),this}}),this.on=(s,f,...e)=>{for(let b of[f].flat()){this.#t=b;for(let o of[s].flat())e.map((r)=>{this.#s(o.toUpperCase(),this.#t,r)})}return this},this.use=(s,...f)=>{if(typeof s==="string")this.#t=s;else this.#t="*",f.unshift(s);return f.forEach((e)=>{this.#s(d,this.#t,e)}),this};let{strict:c,...l}=n;Object.assign(this,l),this.getPath=c??!0?n.getPath??ut:Vt}#n(){let n=new t({router:this.router,getPath:this.getPath});return n.errorHandler=this.errorHandler,n.#i=this.#i,n.routes=this.routes,n}#i=Rn;errorHandler=Ft;route(n,i){let c=this.basePath(n);return i.routes.map((l)=>{let s;if(i.errorHandler===Ft)s=l.handler;else s=async(f,e)=>(await ot([],i.errorHandler)(f,()=>l.handler(f,e))).res,s[kt]=l.handler;c.#s(l.method,l.path,s,l.basePath)}),this}basePath(n){let i=this.#n();return i._basePath=g(this._basePath,n),i}onError=(n)=>{return this.errorHandler=n,this};notFound=(n)=>{return this.#i=n,this};mount(n,i,c){let l,s;if(c)if(typeof c==="function")s=c;else if(s=c.optionHandler,c.replaceRequest===!1)l=(b)=>b;else l=c.replaceRequest;let f=s?(b)=>{let o=s(b);return Array.isArray(o)?o:[o]}:(b)=>{let o=void 0;try{o=b.executionCtx}catch{}return[b.env,o]};l||=(()=>{let b=g(this._basePath,n),o=b==="/"?0:b.length;return(r)=>{let x=new URL(r.url);return x.pathname=this.getPath(r).slice(o)||"/",new Request(x,r)}})();let e=async(b,o)=>{let r=await i(l(b.req.raw),...f(b));if(r)return r;await o()};return this.#s(d,g(n,"*"),e),this}#s(n,i,c,l){n=n.toUpperCase(),i=g(this._basePath,i);let s={basePath:l!==void 0?g(this._basePath,l):this._basePath,path:i,method:n,handler:c};this.router.add(n,i,[c,s]),this.routes.push(s)}#l(n,i){if(n instanceof Error)return this.errorHandler(n,i);throw n}#c(n,i,c,l){if(l==="HEAD")return(async()=>new Response(null,await this.#c(n,i,c,"GET")))();let s=this.getPath(n,{env:c}),f=this.router.match(l,s),e=new yt(n,{path:s,matchResult:f,env:c,executionCtx:i,notFoundHandler:this.#i});if(f[0].length===1){let o;try{o=f[0][0][0][0](e,async()=>{e.res=await this.#i(e)})}catch(r){return this.#l(r,e)}return o instanceof Promise?o.then((r)=>r||(e.finalized?e.res:this.#i(e))).catch((r)=>this.#l(r,e)):o??this.#i(e)}let b=ot(f[0],this.errorHandler,this.#i);return(async()=>{try{let o=await b(e);if(!o.finalized)throw Error("Context is not finalized. Did you forget to return a Response object or `await next()`?");return o.res}catch(o){return this.#l(o,e)}})()}fetch=(n,...i)=>{return this.#c(n,i[1],i[0],n.method)};request=(n,i,c,l)=>{if(n instanceof Request)return this.fetch(i?new Request(n,i):n,c,l);return n=n.toString(),this.fetch(new Request(/^https?:\/\//.test(n)?n:`http://localhost${g("/",n)}`,i),c,l)};fire=()=>{addEventListener("fetch",(n)=>{n.respondWith(this.#c(n.request,n,void 0,n.request.method))})}};var V=[];function q(t,n){let i=this.buildAllMatchers(),c=(l,s)=>{let f=i[l]||i[d],e=f[2][s];if(e)return e;let b=s.match(f[0]);if(!b)return[[],V];let o=b.indexOf("",1);return[f[1][o],b]};return this.match=c,c(t,n)}var tt="[^/]+",w=".*",A="(?:|/.*)",P=Symbol(),kn=new Set(".\\+*[^]$()");function Fn(t,n){if(t.length===1)return n.length===1?t<n?-1:1:-1;if(n.length===1)return 1;if(t===w||t===A)return 1;else if(n===w||n===A)return-1;if(t===tt)return 1;else if(n===tt)return-1;return t.length===n.length?t<n?-1:1:n.length-t.length}var Ct=class t{#t;#n;#i=Object.create(null);insert(n,i,c,l,s){if(n.length===0){if(this.#t!==void 0)throw P;if(s)return;this.#t=i;return}let[f,...e]=n,b=f==="*"?e.length===0?["","",w]:["","",tt]:f==="/*"?["","",A]:f.match(/^\:([^\{\}]+)(?:\{(.+)\})?$/),o;if(b){let r=b[1],x=b[2]||tt;if(r&&b[2]){if(x===".*")throw P;if(x=x.replace(/^\((?!\?:)(?=[^)]+\)$)/,"(?:"),/\((?!\?:)/.test(x))throw P}if(o=this.#i[x],!o){if(Object.keys(this.#i).some((E)=>E!==w&&E!==A))throw P;if(s)return;if(o=this.#i[x]=new t,r!=="")o.#n=l.varIndex++}if(!s&&r!=="")c.push([r,o.#n])}else if(o=this.#i[f],!o){if(Object.keys(this.#i).some((r)=>r.length>1&&r!==w&&r!==A))throw P;if(s)return;o=this.#i[f]=new t}o.insert(e,i,c,l,s)}buildRegExpStr(){let i=Object.keys(this.#i).sort(Fn).map((c)=>{let l=this.#i[c];return(typeof l.#n==="number"?`(${c})@${l.#n}`:kn.has(c)?`\\${c}`:c)+l.buildRegExpStr()});if(typeof this.#t==="number")i.unshift(`#${this.#t}`);if(i.length===0)return"";if(i.length===1)return i[0];return"(?:"+i.join("|")+")"}};var Ht=class{#t={varIndex:0};#n=new Ct;insert(t,n,i){let c=[],l=[];for(let f=0;;){let e=!1;if(t=t.replace(/\{[^}]+\}/g,(b)=>{let o=`@\\${f}`;return l[f]=[o,b],f++,e=!0,o}),!e)break}let s=t.match(/(?::[^\/]+)|(?:\/\*$)|./g)||[];for(let f=l.length-1;f>=0;f--){let[e]=l[f];for(let b=s.length-1;b>=0;b--)if(s[b].indexOf(e)!==-1){s[b]=s[b].replace(e,l[f][1]);break}}return this.#n.insert(s,n,c,this.#t,i),c}buildRegExp(){let t=this.#n.buildRegExpStr();if(t==="")return[/^$/,[],[]];let n=0,i=[],c=[];return t=t.replace(/#(\d+)|@(\d+)|\.\*\$/g,(l,s,f)=>{if(s!==void 0)return i[++n]=Number(s),"$()";if(f!==void 0)return c[Number(f)]=++n,"";return""}),[new RegExp(`^${t}`),i,c]}};var Dn=[/^$/,[],Object.create(null)],pt=Object.create(null);function ht(t){return pt[t]??=new RegExp(t==="*"?"":`^${t.replace(/\/\*$|([.\\+*[^\]$()])/g,(n,i)=>i?`\\${i}`:"(?:|/.*)")}$`)}function Cn(){pt=Object.create(null)}function Hn(t){let n=new Ht,i=[];if(t.length===0)return Dn;let c=t.map((o)=>[!/\*|\/:/.test(o[0]),...o]).sort(([o,r],[x,E])=>o?1:x?-1:r.length-E.length),l=Object.create(null);for(let o=0,r=-1,x=c.length;o<x;o++){let[E,O,$]=c[o];if(E)l[O]=[$.map(([a])=>[a,Object.create(null)]),V];else r++;let v;try{v=n.insert(O,r,E)}catch(a){throw a===P?new h(O):a}if(E)continue;i[r]=$.map(([a,y])=>{let B=Object.create(null);y-=1;for(;y>=0;y--){let[X,ft]=v[y];B[X]=ft}return[a,B]})}let[s,f,e]=n.buildRegExp();for(let o=0,r=i.length;o<r;o++)for(let x=0,E=i[o].length;x<E;x++){let O=i[o][x]?.[1];if(!O)continue;let $=Object.keys(O);for(let v=0,a=$.length;v<a;v++)O[$[v]]=e[O[$[v]]]}let b=[];for(let o in f)b[o]=i[f[o]];return[s,b,l]}function J(t,n){if(!t)return;for(let i of Object.keys(t).sort((c,l)=>l.length-c.length))if(ht(i).test(n))return[...t[i]];return}var nt=class{name="RegExpRouter";#t;#n;constructor(){this.#t={[d]:Object.create(null)},this.#n={[d]:Object.create(null)}}add(t,n,i){let c=this.#t,l=this.#n;if(!c||!l)throw Error(p);if(!c[t])[c,l].forEach((e)=>{e[t]=Object.create(null),Object.keys(e[d]).forEach((b)=>{e[t][b]=[...e[d][b]]})});if(n==="/*")n="*";let s=(n.match(/\/:/g)||[]).length;if(/\*$/.test(n)){let e=ht(n);if(t===d)Object.keys(c).forEach((b)=>{c[b][n]||=J(c[b],n)||J(c[d],n)||[]});else c[t][n]||=J(c[t],n)||J(c[d],n)||[];Object.keys(c).forEach((b)=>{if(t===d||t===b)Object.keys(c[b]).forEach((o)=>{e.test(o)&&c[b][o].push([i,s])})}),Object.keys(l).forEach((b)=>{if(t===d||t===b)Object.keys(l[b]).forEach((o)=>e.test(o)&&l[b][o].push([i,s]))});return}let f=H(n)||[n];for(let e=0,b=f.length;e<b;e++){let o=f[e];Object.keys(l).forEach((r)=>{if(t===d||t===r)l[r][o]||=[...J(c[r],o)||J(c[d],o)||[]],l[r][o].push([i,s-b+e+1])})}}match=q;buildAllMatchers(){let t=Object.create(null);return Object.keys(this.#n).concat(Object.keys(this.#t)).forEach((n)=>{t[n]||=this.#i(n)}),this.#t=this.#n=void 0,Cn(),t}#i(t){let n=[],i=t===d;if([this.#t,this.#n].forEach((c)=>{let l=c[t]?Object.keys(c[t]).map((s)=>[s,c[t][s]]):[];if(l.length!==0)i||=!0,n.push(...l);else if(t!==d)n.push(...Object.keys(c[d]).map((s)=>[s,c[d][s]]))}),!i)return null;else return Hn(n)}};var pn=class{name="PreparedRegExpRouter";#t;#n;constructor(t,n){this.#t=t,this.#n=n}#i(t,n){let i=this.#t[t];i[1].forEach((c)=>c&&c.push(n)),Object.values(i[2]).forEach((c)=>c[0].push(n))}#s(t,n,i,c,l){let s=this.#t[t];if(!l)s[2][n][0].push([i,{}]);else c.forEach((f)=>{if(typeof f==="number")s[1][f].push([i,l]);else s[2][f||n][0].push([i,l])})}add(t,n,i){if(!this.#t[t]){let l=this.#t[d],s={};for(let f in l[2])s[f]=[l[2][f][0].slice(),V];this.#t[t]=[l[0],l[1].map((f)=>Array.isArray(f)?f.slice():0),s]}if(n==="/*"||n==="*"){let l=[i,{}];if(t===d)for(let s in this.#t)this.#i(s,l);else this.#i(t,l);return}let c=this.#n[n];if(!c)throw Error(`Path ${n} is not registered`);for(let[l,s]of c)if(t===d)for(let f in this.#t)this.#s(f,n,i,l,s);else this.#s(t,n,i,l,s)}buildAllMatchers(){return this.#t}match=q};var Nt=class{name="SmartRouter";#t=[];#n=[];constructor(t){this.#t=t.routers}add(t,n,i){if(!this.#n)throw Error(p);this.#n.push([t,n,i])}match(t,n){if(!this.#n)throw Error("Fatal error");let i=this.#t,c=this.#n,l=i.length,s=0,f;for(;s<l;s++){let e=i[s];try{for(let b=0,o=c.length;b<o;b++)e.add(...c[b]);f=e.match(t,n)}catch(b){if(b instanceof h)continue;throw b}this.match=e.match.bind(e),this.#t=[e],this.#n=void 0;break}if(s===l)throw Error("Fatal error");return this.name=`SmartRouter + ${this.activeRouter.name}`,f}get activeRouter(){if(this.#n||this.#t.length!==1)throw Error("No active router has been determined yet.");return this.#t[0]}};var Z=Object.create(null),hn=(t)=>{for(let n in t)return!0;return!1},qt=class t{#t;#n;#i;#s=0;#l=Z;constructor(n,i,c){if(this.#n=c||Object.create(null),this.#t=[],n&&i){let l=Object.create(null);l[n]={handler:i,possibleKeys:[],score:0},this.#t=[l]}this.#i=[]}insert(n,i,c){this.#s=++this.#s;let l=this,s=Qt(i),f=[];for(let e=0,b=s.length;e<b;e++){let o=s[e],r=s[e+1],x=Yt(o,r),E=Array.isArray(x)?x[0]:o;if(E in l.#n){if(l=l.#n[E],x)f.push(x[1]);continue}if(l.#n[E]=new t,x)l.#i.push(x),f.push(x[1]);l=l.#n[E]}return l.#t.push({[n]:{handler:c,possibleKeys:f.filter((e,b,o)=>o.indexOf(e)===b),score:this.#s}}),l}#c(n,i,c,l,s){for(let f=0,e=i.#t.length;f<e;f++){let b=i.#t[f],o=b[c]||b[d],r={};if(o!==void 0){if(o.params=Object.create(null),n.push(o),l!==Z||s&&s!==Z)for(let x=0,E=o.possibleKeys.length;x<E;x++){let O=o.possibleKeys[x],$=r[o.score];o.params[O]=s?.[O]&&!$?s[O]:l[O]??s?.[O],r[o.score]=!0}}}}search(n,i){let c=[];this.#l=Z;let s=[this],f=xt(i),e=[],b=f.length,o=null;for(let r=0;r<b;r++){let x=f[r],E=r===b-1,O=[];for(let v=0,a=s.length;v<a;v++){let y=s[v],B=y.#n[x];if(B)if(B.#l=y.#l,E){if(B.#n["*"])this.#c(c,B.#n["*"],n,y.#l);this.#c(c,B,n,y.#l)}else O.push(B);for(let X=0,ft=y.#i.length;X<ft;X++){let _t=y.#i[X],K=y.#l===Z?{}:{...y.#l};if(_t==="*"){let j=y.#n["*"];if(j)this.#c(c,j,n,y.#l),j.#l=K,O.push(j);continue}let[gn,Wt,G]=_t;if(!x&&!(G instanceof RegExp))continue;let L=y.#n[gn];if(G instanceof RegExp){if(o===null){o=Array(b);let k=i[0]==="/"?1:0;for(let Q=0;Q<b;Q++)o[Q]=k,k+=f[Q].length+1}let j=i.substring(o[r]),bt=G.exec(j);if(bt){if(K[Wt]=bt[0],this.#c(c,L,n,y.#l,K),hn(L.#n)){L.#l=K;let k=bt[0].match(/\//)?.length??0;(e[k]||=[]).push(L)}continue}}if(G===!0||G.test(x))if(K[Wt]=x,E){if(this.#c(c,L,n,K,y.#l),L.#n["*"])this.#c(c,L.#n["*"],n,K,y.#l)}else L.#l=K,O.push(L)}}let $=e.shift();s=$?O.concat($):O}if(c.length>1)c.sort((r,x)=>{return r.score-x.score});return[c.map(({handler:r,params:x})=>[r,x])]}};var St=class{name="TrieRouter";#t;constructor(){this.#t=new qt}add(t,n,i){let c=H(n);if(c){for(let l=0,s=c.length;l<s;l++)this.#t.insert(t,c[l],i);return}this.#t.insert(t,n,i)}match(t,n){return this.#t.search(t,n)}};var $t=class extends Dt{constructor(t={}){super(t);this.router=t.router??new Nt({routers:[new nt,new St]})}};import qn from"os";import vt from"path";var u={port:Number(process.env.JUNSO_BROWSER_PORT??8790),host:process.env.JUNSO_BROWSER_HOST??"127.0.0.1",token:process.env.JUNSO_BROWSER_TOKEN??"",dataDir:process.env.JUNSO_BROWSER_DATA_DIR??vt.join(qn.homedir(),".junso-browser"),maxInstances:Number(process.env.JUNSO_BROWSER_MAX_INSTANCES??3),idleReapMs:Number(process.env.JUNSO_BROWSER_IDLE_MS??1800000),cloakPath:process.env.JUNSO_BROWSER_CLOAK_PATH,publicUrl:process.env.JUNSO_BROWSER_PUBLIC_URL};function at(){return vt.join(u.dataDir,"profiles.json")}function it(t){return vt.join(u.dataDir,"profiles",t,"user-data")}function ct(){return u.host==="127.0.0.1"||u.host==="localhost"||u.host==="::1"}import{mkdir as ci,readFile as li,writeFile as si,rm as fi}from"fs/promises";import bi from"path";import{randomUUID as ei}from"crypto";import ti from"os";import Lt from"path";import{existsSync as tn,readdirSync as ni}from"fs";function m(){if(u.cloakPath&&tn(u.cloakPath))return u.cloakPath;let t=Lt.join(ti.homedir(),".cloakbrowser"),n;try{n=ni(t).filter((c)=>c.startsWith("chromium-"))}catch{return null}n.sort().reverse();let i=process.platform==="win32"?"chrome.exe":"chrome";for(let c of n){let l=Lt.join(t,c,i);if(tn(l))return l}return null}var Bt=!1;function nn(){if(Bt)return;if(process.env.JUNSO_BROWSER_NO_CLOAK_DOWNLOAD==="true")return;if(m())return;Bt=!0,console.log("[junso-browser] CloakBrowser binary not found \u2014 downloading in background\u2026");let t=Lt.resolve(import.meta.dir,"..");Bun.spawn([process.execPath,"x","cloakbrowser","install"],{cwd:t,stdout:"inherit",stderr:"inherit"}).exited.then((i)=>{Bt=!1,console.log(i===0?"[junso-browser] CloakBrowser ready.":`[junso-browser] cloak download exited ${i}.`)})}function ii(t){try{let n=new URL(t);return`${n.protocol}//${n.host}`}catch{return t}}function cn(t){if(!t)return null;try{let n=new URL(t);if(!n.username)return null;return{username:decodeURIComponent(n.username),password:decodeURIComponent(n.password)}}catch{return null}}function zt(){return Math.floor(Math.random()*90000)+1e4}function ln(t,n,i){let c=t.fingerprint,l=["--no-sandbox",`--fingerprint=${c.seed}`,`--fingerprint-platform=${c.platform??"windows"}`];if(c.timezone)l.push(`--fingerprint-timezone=${c.timezone}`);if(c.locale)l.push(`--fingerprint-locale=${c.locale}`,`--accept-lang=${c.locale}`);if(t.proxy)l.push(`--proxy-server=${ii(t.proxy)}`);return l.push(`--user-data-dir=${i}`,`--remote-debugging-port=${n}`,"--remote-debugging-address=127.0.0.1","--headless=new"),l}function sn(t){let n={...process.env};if(t.fingerprint.timezone)n.TZ=t.fingerprint.timezone;return n}var _=null,fn=Promise.resolve();async function M(){if(_)return _;try{let t=await li(at(),"utf8"),n=JSON.parse(t);_=n&&typeof n==="object"?n:{}}catch{_={}}return _}async function Kt(t){await ci(u.dataDir,{recursive:!0}),await si(at(),JSON.stringify(t,null,2)+`
3
3
  `,"utf8"),_=t}function gt(t){let n=fn.catch(()=>{}).then(t);return fn=n.catch(()=>{}),n}function oi(t){return(t??"").trim().toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"")}async function bn(){return Object.values(await M()).sort((t,n)=>t.name.localeCompare(n.name))}async function z(t){return(await M())[t]??null}async function en(t){let n=oi(t.name)||ei().slice(0,8);return gt(async()=>{let i={...await M()};if(i[n])throw Error(`Profile "${n}" already exists`);let c=new Date().toISOString(),l={id:n,name:t.name.trim(),fingerprint:{seed:t.fingerprint?.seed??zt(),platform:t.fingerprint?.platform??"windows",timezone:t.fingerprint?.timezone,locale:t.fingerprint?.locale},proxy:t.proxy?.trim()||void 0,createdAt:c,updatedAt:c};return i[n]=l,await Kt(i),l})}async function Pt(t,n){return gt(async()=>{let i={...await M()},c=i[t];if(!c)throw Error(`No such profile: ${t}`);let l={...c.fingerprint,...n.fingerprint??{}};if(n.rotateSeed)l.seed=zt();let s={...c,name:n.name?.trim()||c.name,fingerprint:l,proxy:n.proxy===null?void 0:n.proxy?.trim()||c.proxy,updatedAt:new Date().toISOString()};return i[t]=s,await Kt(i),s})}async function on(t){return gt(async()=>{let n={...await M()};if(!n[t])return!1;return delete n[t],await Kt(n),await fi(bi.dirname(it(t)),{recursive:!0,force:!0}),!0})}function rn(t){if(!/session-[a-z0-9]+/i.test(t))return null;let n="s"+Math.random().toString(36).slice(2,10);return t.replace(/session-[a-z0-9]+/i,`session-${n}`)}function ri(t){if(!t)return null;return t.replace(/^([a-z0-9+.-]+:\/\/[^:@/]+):[^@/]+@/i,"$1:\u2022\u2022\u2022\u2022@")}function W(t){return{...t,proxy:ri(t.proxy)??void 0}}import{mkdir as xi,readFile as ui,rm as xn}from"fs/promises";import jt from"path";class Tt{browserWsUrl;username;password;ws=null;nextId=1;constructor(t,n,i){this.browserWsUrl=t;this.username=n;this.password=i}async start(){let t=new WebSocket(this.browserWsUrl);this.ws=t,await new Promise((n,i)=>{t.onopen=()=>n(),t.onerror=()=>i(Error("proxy-auth CDP connect failed"))}),t.onmessage=(n)=>this.onMessage(JSON.parse(String(n.data))),t.onclose=()=>{this.ws=null},this.send(void 0,"Target.setAutoAttach",{autoAttach:!0,waitForDebuggerOnStart:!1,flatten:!0})}send(t,n,i){this.ws?.send(JSON.stringify({id:this.nextId++,sessionId:t,method:n,params:i}))}onMessage(t){if(t.method==="Target.attachedToTarget"){let n=t.params.sessionId;this.send(n,"Fetch.enable",{handleAuthRequests:!0})}else if(t.method==="Fetch.authRequired"){let n=t.params,i=n.authChallenge?.source==="Proxy";this.send(t.sessionId,"Fetch.continueWithAuth",{requestId:n.requestId,authChallengeResponse:i?{response:"ProvideCredentials",username:this.username,password:this.password}:{response:"Default"}})}else if(t.method==="Fetch.requestPaused")this.send(t.sessionId,"Fetch.continueRequest",{requestId:t.params.requestId})}close(){try{this.ws?.close()}catch{}this.ws=null}}var N=new Map;async function di(t,n=20000){let i=jt.join(t,"DevToolsActivePort"),c=Date.now()+n;while(Date.now()<c){try{let l=await ui(i,"utf8"),[s,f]=l.split(`
4
- `),e=Number(s);if(e>0&&f)return{port:e,wsPath:f.trim()}}catch{}await Bun.sleep(150)}throw Error("Browser did not become ready (no DevToolsActivePort)")}async function Ei(){let t=null;for(let n of N.values())if(!t||n.instance.lastUsedAt<t.instance.lastUsedAt)t=n;if(t)await T(t.instance.profileId)}async function I(t){let n=N.get(t.id);if(n)return n.instance.lastUsedAt=new Date().toISOString(),n.instance;let i=m();if(!i)throw Error("CloakBrowser binary not found \u2014 run `cloakbrowser install`");if(u.maxInstances>0&&N.size>=u.maxInstances)await Ei();let c=it(t.id);await xi(c,{recursive:!0}),await xn(jt.join(c,"DevToolsActivePort"),{force:!0}),await xn(jt.join(c,"SingletonLock"),{force:!0});let l=ln(t,0,c),s=Bun.spawn([i,...l],{stdout:"ignore",stderr:"ignore",env:sn(t)}),f;try{f=await di(c)}catch(x){throw s.kill(),x}let e=new Date().toISOString(),b={profileId:t.id,pid:s.pid,port:f.port,browserWsPath:f.wsPath,startedAt:e,lastUsedAt:e},o={instance:b,proc:s};N.set(t.id,o);let r=cn(t.proxy);if(r){let x=new Tt(`ws://127.0.0.1:${f.port}${f.wsPath}`,r.username,r.password);try{await x.start(),o.auth=x}catch{}}return s.exited.then(()=>{if(N.get(t.id)?.proc===s)o.auth?.close(),N.delete(t.id)}),b}function lt(t){return N.has(t)}function un(t){let n=N.get(t);if(n)n.instance.lastUsedAt=new Date().toISOString()}async function T(t){let n=N.get(t);if(!n)return!1;return N.delete(t),n.auth?.close(),n.proc.kill(),await n.proc.exited.catch(()=>{}),!0}async function dn(){await Promise.all([...N.keys()].map((t)=>T(t)))}function En(){return[...N.values()].map((t)=>t.instance)}function On(){if(u.idleReapMs<=0)return null;return setInterval(()=>{let t=Date.now()-u.idleReapMs;for(let n of N.values())if(new Date(n.instance.lastUsedAt).getTime()<t)T(n.instance.profileId)},60000)}async function yi(t,n=15000){try{return(await(await fetch("https://api.ipify.org?format=json",{proxy:t||void 0,signal:AbortSignal.timeout(n)})).json()).ip??null}catch{return null}}async function Ni(t,n=1e4){try{let l=await(await fetch(`http://ip-api.com/json/${t}?fields=status,country,countryCode,timezone,isp,mobile,proxy,hosting,query`,{signal:AbortSignal.timeout(n)})).json();if(l.status!=="success")return null;let s="residential";if(l.proxy)s="proxy";else if(l.hosting)s="hosting";else if(l.mobile)s="mobile";return{ip:l.query??t,country:l.country,countryCode:l.countryCode,timezone:l.timezone,isp:l.isp,kind:s}}catch{return null}}async function st(t){let n=await yi(t);if(!n)return null;return await Ni(n)??{ip:n,kind:"unknown"}}function Jt(t){return{timezone:t.timezone,locale:t.countryCode?`en-${t.countryCode}`:void 0}}var yn={name:"junso-browser",version:"0.2.3",description:"Standalone CloakBrowser host \u2014 runs fingerprinted stealth-Chromium per profile, exposes each over a token-authed CDP gateway + interactive viewport, and ships an optional MCP server so any client gets browser tools with no local browser.",type:"module",bin:{"junso-browser":"bin/junso-browser.js","junso-browser-mcp":"bin/junso-browser-mcp.js"},files:["bin","dist","README.md"],engines:{bun:">=1.2.0"},keywords:["browser","cloakbrowser","stealth","fingerprint","cdp","mcp","proxy","automation"],license:"UNLICENSED",scripts:{dev:"bun --watch src/server.ts",start:"bun src/server.ts",mcp:"bun src/mcp.ts",build:"bun build src/server.ts src/mcp.ts --target=bun --minify --outdir=dist","build:binary":"bun build src/server.ts --compile --outfile dist/junso-browser",typecheck:"bunx tsc --noEmit",test:"bun test",prepublishOnly:"bun run build"},dependencies:{"agent-browser":"^0.27.1",cloakbrowser:"^0.3.31"},devDependencies:{"@modelcontextprotocol/sdk":"^1.29.0","@types/bun":"latest",hono:"^4.6.0",typescript:"^5.6.0",zod:"3"}};function $i(t){return/^https?:|^file:/.test(t)}class Nn{browserWsUrl;cdp=null;sessionId=null;activeTargetId=null;nextId=1;pending=new Map;starting=null;targets=new Map;lastFrame=null;lastUrl=null;viewers=new Set;constructor(t){this.browserWsUrl=t}rpc(t,n={},i){if(!this.cdp)return Promise.reject(Error("cdp closed"));let c=this.nextId++;return this.cdp.send(JSON.stringify({id:c,method:t,params:n,sessionId:i})),new Promise((l)=>this.pending.set(c,l))}async start(){if(this.starting)return this.starting;return this.starting=(async()=>{let t=new WebSocket(this.browserWsUrl);this.cdp=t,await new Promise((l,s)=>{t.onopen=()=>l(),t.onerror=()=>s(Error("CDP connect failed"))}),t.onmessage=(l)=>this.onMessage(JSON.parse(String(l.data))),t.onclose=()=>{this.cdp=null,this.sessionId=null},await this.rpc("Target.setDiscoverTargets",{discover:!0});let{targetInfos:n}=await this.rpc("Target.getTargets"),i=n.filter((l)=>l.type==="page");for(let l of i)this.targets.set(l.targetId,{url:l.url,title:l.title});let c=i.find((l)=>$i(l.url))??i[0];if(!c)throw Error("no page target to stream");await this.attachTo(c.targetId)})(),this.starting}async attachTo(t){if(this.sessionId){let i=this.sessionId;this.sessionId=null;try{await this.rpc("Target.detachFromTarget",{sessionId:i})}catch{}}let{sessionId:n}=await this.rpc("Target.attachToTarget",{targetId:t,flatten:!0});this.sessionId=n,this.activeTargetId=t,await this.rpc("Page.enable",{},n),await this.rpc("Page.startScreencast",{format:"jpeg",quality:60,maxWidth:1280,maxHeight:800,everyNthFrame:1},n);try{let i=await this.rpc("Page.getNavigationHistory",{},n);this.lastUrl=i.entries[i.currentIndex]?.url??null}catch{}if(this.broadcastTabs(),this.lastUrl)this.broadcast({type:"url",url:this.lastUrl})}async switchTarget(t){if(!this.targets.has(t)||t===this.activeTargetId)return;this.lastFrame=null,await this.attachTo(t).catch(()=>{})}tabList(){return[...this.targets.entries()].map(([t,n])=>({targetId:t,url:n.url,title:n.title,active:t===this.activeTargetId}))}broadcastTabs(){this.broadcast({type:"tabs",tabs:this.tabList()})}onMessage(t){if(t.id!==void 0&&this.pending.has(t.id)){this.pending.get(t.id)(t.result),this.pending.delete(t.id);return}if(t.method==="Page.screencastFrame"){let n=t.params;this.rpc("Page.screencastFrameAck",{sessionId:n.sessionId},this.sessionId??void 0);let i=JSON.stringify({type:"frame",data:n.data,metadata:n.metadata});this.lastFrame=i;for(let c of this.viewers)c.send(i)}else if(t.method==="Page.frameNavigated"){let n=t.params.frame;if(n&&!n.parentId&&n.url){if(this.lastUrl=n.url,this.activeTargetId){let i=this.targets.get(this.activeTargetId);if(i)i.url=n.url}this.broadcast({type:"url",url:n.url})}}else if(t.method==="Target.targetCreated"||t.method==="Target.targetInfoChanged"){let n=t.params.targetInfo;if(n&&n.type==="page")this.targets.set(n.targetId,{url:n.url,title:n.title}),this.broadcastTabs()}else if(t.method==="Target.targetDestroyed"){let n=t.params.targetId;if(this.targets.delete(n)){if(this.broadcastTabs(),n===this.activeTargetId){let i=[...this.targets.keys()][0];if(i)this.switchTarget(i)}}}}broadcast(t){let n=JSON.stringify(t);for(let i of this.viewers)i.send(n)}sendStateTo(t){if(this.lastFrame)t.send(this.lastFrame);if(this.lastUrl)t.send(JSON.stringify({type:"url",url:this.lastUrl}));t.send(JSON.stringify({type:"tabs",tabs:this.tabList()}))}input(t){if(!this.cdp||!this.sessionId)return;let n=t.t;if(n==="m")this.rpc("Input.dispatchMouseEvent",{type:t.type,x:t.x,y:t.y,button:t.button??"none",buttons:t.buttons??0,clickCount:t.clickCount??0,deltaX:t.deltaX??0,deltaY:t.deltaY??0,modifiers:t.modifiers??0},this.sessionId);else if(n==="k")this.rpc("Input.dispatchKeyEvent",{type:t.type,key:t.key,code:t.code,text:t.text,windowsVirtualKeyCode:t.keyCode,nativeVirtualKeyCode:t.keyCode,modifiers:t.modifiers??0},this.sessionId);else if(n==="tab"){let i=String(t.targetId??"");if(i)this.switchTarget(i)}else if(n==="nav"){let i=String(t.url??"").trim();if(i)this.rpc("Page.navigate",{url:/^[a-z]+:\/\//i.test(i)?i:`https://${i}`},this.sessionId)}else if(n==="reload")this.rpc("Page.reload",{},this.sessionId);else if(n==="back"||n==="forward")(async()=>{let i=await this.rpc("Page.getNavigationHistory",{},this.sessionId),c=i.currentIndex+(n==="forward"?1:-1),l=i.entries[c];if(l)await this.rpc("Page.navigateToHistoryEntry",{entryId:l.id},this.sessionId)})()}close(){try{this.cdp?.close()}catch{}this.cdp=null}}var U=new Map;async function Sn(t,n){let i=await z(t);if(!i)throw Error(`no such profile: ${t}`);let c=await I(i),l=U.get(t);if(!l)l=new Nn(`ws://127.0.0.1:${c.port}${c.browserWsPath}`),U.set(t,l);await l.start(),l.viewers.add(n),l.sendStateTo(n)}function $n(t,n){let i=U.get(t);if(!i)return;if(i.viewers.delete(n),i.viewers.size===0)i.close(),U.delete(t)}function vn(t,n){let i=U.get(t);if(!i)return;try{i.input(JSON.parse(n))}catch{}}function an(t,n){let i=JSON.stringify(t),c=JSON.stringify(n||"");return`<!doctype html>
4
+ `),e=Number(s);if(e>0&&f)return{port:e,wsPath:f.trim()}}catch{}await Bun.sleep(150)}throw Error("Browser did not become ready (no DevToolsActivePort)")}async function Ei(){let t=null;for(let n of N.values())if(!t||n.instance.lastUsedAt<t.instance.lastUsedAt)t=n;if(t)await T(t.instance.profileId)}async function I(t){let n=N.get(t.id);if(n)return n.instance.lastUsedAt=new Date().toISOString(),n.instance;let i=m();if(!i)throw Error("CloakBrowser binary not found \u2014 run `cloakbrowser install`");if(u.maxInstances>0&&N.size>=u.maxInstances)await Ei();let c=it(t.id);await xi(c,{recursive:!0}),await xn(jt.join(c,"DevToolsActivePort"),{force:!0}),await xn(jt.join(c,"SingletonLock"),{force:!0});let l=ln(t,0,c),s=Bun.spawn([i,...l],{stdout:"ignore",stderr:"ignore",env:sn(t)}),f;try{f=await di(c)}catch(x){throw s.kill(),x}let e=new Date().toISOString(),b={profileId:t.id,pid:s.pid,port:f.port,browserWsPath:f.wsPath,startedAt:e,lastUsedAt:e},o={instance:b,proc:s};N.set(t.id,o);let r=cn(t.proxy);if(r){let x=new Tt(`ws://127.0.0.1:${f.port}${f.wsPath}`,r.username,r.password);try{await x.start(),o.auth=x}catch{}}return s.exited.then(()=>{if(N.get(t.id)?.proc===s)o.auth?.close(),N.delete(t.id)}),b}function lt(t){return N.has(t)}function un(t){let n=N.get(t);if(n)n.instance.lastUsedAt=new Date().toISOString()}async function T(t){let n=N.get(t);if(!n)return!1;return N.delete(t),n.auth?.close(),n.proc.kill(),await n.proc.exited.catch(()=>{}),!0}async function dn(){await Promise.all([...N.keys()].map((t)=>T(t)))}function En(){return[...N.values()].map((t)=>t.instance)}function On(){if(u.idleReapMs<=0)return null;return setInterval(()=>{let t=Date.now()-u.idleReapMs;for(let n of N.values())if(new Date(n.instance.lastUsedAt).getTime()<t)T(n.instance.profileId)},60000)}async function yi(t,n=15000){try{return(await(await fetch("https://api.ipify.org?format=json",{proxy:t||void 0,signal:AbortSignal.timeout(n)})).json()).ip??null}catch{return null}}async function Ni(t,n=1e4){try{let l=await(await fetch(`http://ip-api.com/json/${t}?fields=status,country,countryCode,timezone,isp,mobile,proxy,hosting,query`,{signal:AbortSignal.timeout(n)})).json();if(l.status!=="success")return null;let s="residential";if(l.proxy)s="proxy";else if(l.hosting)s="hosting";else if(l.mobile)s="mobile";return{ip:l.query??t,country:l.country,countryCode:l.countryCode,timezone:l.timezone,isp:l.isp,kind:s}}catch{return null}}async function st(t){let n=await yi(t);if(!n)return null;return await Ni(n)??{ip:n,kind:"unknown"}}function Jt(t){return{timezone:t.timezone,locale:t.countryCode?`en-${t.countryCode}`:void 0}}var yn={name:"junso-browser",version:"0.2.6",description:"Standalone CloakBrowser host \u2014 runs fingerprinted stealth-Chromium per profile, exposes each over a token-authed CDP gateway + interactive viewport, and ships an optional MCP server so any client gets browser tools with no local browser.",type:"module",bin:{"junso-browser":"bin/junso-browser.js","junso-browser-mcp":"bin/junso-browser-mcp.js"},files:["bin","dist","README.md"],engines:{bun:">=1.2.0"},keywords:["browser","cloakbrowser","stealth","fingerprint","cdp","mcp","proxy","automation"],license:"UNLICENSED",scripts:{dev:"bun --watch src/server.ts",start:"bun src/server.ts",mcp:"bun src/mcp.ts",build:"bun build src/server.ts src/mcp.ts --target=bun --minify --outdir=dist","build:binary":"bun build src/server.ts --compile --outfile dist/junso-browser",typecheck:"bunx tsc --noEmit",test:"bun test",prepublishOnly:"bun run build"},dependencies:{"agent-browser":"^0.27.1",cloakbrowser:"^0.3.31"},devDependencies:{"@modelcontextprotocol/sdk":"^1.29.0","@types/bun":"latest",hono:"^4.6.0",typescript:"^5.6.0",zod:"3"}};function $i(t){return/^https?:|^file:/.test(t)}class Nn{browserWsUrl;cdp=null;sessionId=null;activeTargetId=null;nextId=1;pending=new Map;starting=null;targets=new Map;lastFrame=null;lastUrl=null;viewers=new Set;constructor(t){this.browserWsUrl=t}rpc(t,n={},i){if(!this.cdp)return Promise.reject(Error("cdp closed"));let c=this.nextId++;return this.cdp.send(JSON.stringify({id:c,method:t,params:n,sessionId:i})),new Promise((l)=>this.pending.set(c,l))}async start(){if(this.starting)return this.starting;return this.starting=(async()=>{let t=new WebSocket(this.browserWsUrl);this.cdp=t,await new Promise((l,s)=>{t.onopen=()=>l(),t.onerror=()=>s(Error("CDP connect failed"))}),t.onmessage=(l)=>this.onMessage(JSON.parse(String(l.data))),t.onclose=()=>{this.cdp=null,this.sessionId=null},await this.rpc("Target.setDiscoverTargets",{discover:!0});let{targetInfos:n}=await this.rpc("Target.getTargets"),i=n.filter((l)=>l.type==="page");for(let l of i)this.targets.set(l.targetId,{url:l.url,title:l.title});let c=i.find((l)=>$i(l.url))??i[0];if(!c)throw Error("no page target to stream");await this.attachTo(c.targetId)})(),this.starting}async attachTo(t){if(this.sessionId){let i=this.sessionId;this.sessionId=null;try{await this.rpc("Target.detachFromTarget",{sessionId:i})}catch{}}let{sessionId:n}=await this.rpc("Target.attachToTarget",{targetId:t,flatten:!0});this.sessionId=n,this.activeTargetId=t,await this.rpc("Page.enable",{},n),await this.rpc("Page.startScreencast",{format:"jpeg",quality:60,maxWidth:1280,maxHeight:800,everyNthFrame:1},n);try{let i=await this.rpc("Page.getNavigationHistory",{},n);this.lastUrl=i.entries[i.currentIndex]?.url??null}catch{}if(this.broadcastTabs(),this.lastUrl)this.broadcast({type:"url",url:this.lastUrl})}async switchTarget(t){if(!this.targets.has(t)||t===this.activeTargetId)return;this.lastFrame=null,await this.attachTo(t).catch(()=>{})}tabList(){return[...this.targets.entries()].map(([t,n])=>({targetId:t,url:n.url,title:n.title,active:t===this.activeTargetId}))}broadcastTabs(){this.broadcast({type:"tabs",tabs:this.tabList()})}onMessage(t){if(t.id!==void 0&&this.pending.has(t.id)){this.pending.get(t.id)(t.result),this.pending.delete(t.id);return}if(t.method==="Page.screencastFrame"){let n=t.params;this.rpc("Page.screencastFrameAck",{sessionId:n.sessionId},this.sessionId??void 0);let i=JSON.stringify({type:"frame",data:n.data,metadata:n.metadata});this.lastFrame=i;for(let c of this.viewers)c.send(i)}else if(t.method==="Page.frameNavigated"){let n=t.params.frame;if(n&&!n.parentId&&n.url){if(this.lastUrl=n.url,this.activeTargetId){let i=this.targets.get(this.activeTargetId);if(i)i.url=n.url}this.broadcast({type:"url",url:n.url})}}else if(t.method==="Target.targetCreated"||t.method==="Target.targetInfoChanged"){let n=t.params.targetInfo;if(n&&n.type==="page")this.targets.set(n.targetId,{url:n.url,title:n.title}),this.broadcastTabs()}else if(t.method==="Target.targetDestroyed"){let n=t.params.targetId;if(this.targets.delete(n)){if(this.broadcastTabs(),n===this.activeTargetId){let i=[...this.targets.keys()][0];if(i)this.switchTarget(i)}}}}broadcast(t){let n=JSON.stringify(t);for(let i of this.viewers)i.send(n)}sendStateTo(t){if(this.lastFrame)t.send(this.lastFrame);if(this.lastUrl)t.send(JSON.stringify({type:"url",url:this.lastUrl}));t.send(JSON.stringify({type:"tabs",tabs:this.tabList()}))}input(t){if(!this.cdp||!this.sessionId)return;let n=t.t;if(n==="m")this.rpc("Input.dispatchMouseEvent",{type:t.type,x:t.x,y:t.y,button:t.button??"none",buttons:t.buttons??0,clickCount:t.clickCount??0,deltaX:t.deltaX??0,deltaY:t.deltaY??0,modifiers:t.modifiers??0},this.sessionId);else if(n==="k")this.rpc("Input.dispatchKeyEvent",{type:t.type,key:t.key,code:t.code,text:t.text,windowsVirtualKeyCode:t.keyCode,nativeVirtualKeyCode:t.keyCode,modifiers:t.modifiers??0},this.sessionId);else if(n==="tab"){let i=String(t.targetId??"");if(i)this.switchTarget(i)}else if(n==="nav"){let i=String(t.url??"").trim();if(i)this.rpc("Page.navigate",{url:/^[a-z]+:\/\//i.test(i)?i:`https://${i}`},this.sessionId)}else if(n==="reload")this.rpc("Page.reload",{},this.sessionId);else if(n==="back"||n==="forward")(async()=>{let i=await this.rpc("Page.getNavigationHistory",{},this.sessionId),c=i.currentIndex+(n==="forward"?1:-1),l=i.entries[c];if(l)await this.rpc("Page.navigateToHistoryEntry",{entryId:l.id},this.sessionId)})()}close(){try{this.cdp?.close()}catch{}this.cdp=null}}var U=new Map;async function Sn(t,n){let i=await z(t);if(!i)throw Error(`no such profile: ${t}`);let c=await I(i),l=U.get(t);if(!l)l=new Nn(`ws://127.0.0.1:${c.port}${c.browserWsPath}`),U.set(t,l);await l.start(),l.viewers.add(n),l.sendStateTo(n)}function $n(t,n){let i=U.get(t);if(!i)return;if(i.viewers.delete(n),i.viewers.size===0)i.close(),U.delete(t)}function vn(t,n){let i=U.get(t);if(!i)return;try{i.input(JSON.parse(n))}catch{}}function an(t,n){let i=JSON.stringify(t),c=JSON.stringify(n||"");return`<!doctype html>
5
5
  <html><head><meta charset="utf-8"><title>junso-browser \xB7 ${t}</title>
6
6
  <style>
7
7
  html,body{margin:0;height:100%;background:#111;color:#ccc;font:13px system-ui}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "junso-browser",
3
- "version": "0.2.3",
3
+ "version": "0.2.6",
4
4
  "description": "Standalone CloakBrowser host — runs fingerprinted stealth-Chromium per profile, exposes each over a token-authed CDP gateway + interactive viewport, and ships an optional MCP server so any client gets browser tools with no local browser.",
5
5
  "type": "module",
6
6
  "bin": {