obsidian-agent-fleet 0.14.0 → 0.15.0

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/plugin/main.js CHANGED
@@ -1,10 +1,10 @@
1
- "use strict";var al=Object.create;var $s=Object.defineProperty;var il=Object.getOwnPropertyDescriptor;var rl=Object.getOwnPropertyNames;var ol=Object.getPrototypeOf,ll=Object.prototype.hasOwnProperty;var Ve=(i,t)=>()=>(t||i((t={exports:{}}).exports,t),t.exports),cl=(i,t)=>{for(var e in t)$s(i,e,{get:t[e],enumerable:!0})},ai=(i,t,e,s)=>{if(t&&typeof t=="object"||typeof t=="function")for(let n of rl(t))!ll.call(i,n)&&n!==e&&$s(i,n,{get:()=>t[n],enumerable:!(s=il(t,n))||s.enumerable});return i};var Ye=(i,t,e)=>(e=i!=null?al(ol(i)):{},ai(t||!i||!i.__esModule?$s(e,"default",{value:i,enumerable:!0}):e,i)),dl=i=>ai($s({},"__esModule",{value:!0}),i);var gt=Ve((wp,Er)=>{"use strict";var _r=["nodebuffer","arraybuffer","fragments"],Ar=typeof Blob<"u";Ar&&_r.push("blob");Er.exports={BINARY_TYPES:_r,CLOSE_TIMEOUT:3e4,EMPTY_BUFFER:Buffer.alloc(0),GUID:"258EAFA5-E914-47DA-95CA-C5AB0DC85B11",hasBlob:Ar,kForOnEventAttribute:Symbol("kIsForOnEventAttribute"),kListener:Symbol("kListener"),kStatusCode:Symbol("status-code"),kWebSocket:Symbol("websocket"),NOOP:()=>{}}});var _s=Ve((bp,xn)=>{"use strict";var{EMPTY_BUFFER:kc}=gt(),Ca=Buffer[Symbol.species];function xc(i,t){if(i.length===0)return kc;if(i.length===1)return i[0];let e=Buffer.allocUnsafe(t),s=0;for(let n=0;n<i.length;n++){let a=i[n];e.set(a,s),s+=a.length}return s<t?new Ca(e.buffer,e.byteOffset,s):e}function Pr(i,t,e,s,n){for(let a=0;a<n;a++)e[s+a]=i[a]^t[a&3]}function Rr(i,t){for(let e=0;e<i.length;e++)i[e]^=t[e&3]}function Sc(i){return i.length===i.buffer.byteLength?i.buffer:i.buffer.slice(i.byteOffset,i.byteOffset+i.length)}function Ta(i){if(Ta.readOnly=!0,Buffer.isBuffer(i))return i;let t;return i instanceof ArrayBuffer?t=new Ca(i):ArrayBuffer.isView(i)?t=new Ca(i.buffer,i.byteOffset,i.byteLength):(t=Buffer.from(i),Ta.readOnly=!1),t}xn.exports={concat:xc,mask:Pr,toArrayBuffer:Sc,toBuffer:Ta,unmask:Rr};if(!process.env.WS_NO_BUFFER_UTIL)try{let i=require("bufferutil");xn.exports.mask=function(t,e,s,n,a){a<48?Pr(t,e,s,n,a):i.mask(t,e,s,n,a)},xn.exports.unmask=function(t,e){t.length<32?Rr(t,e):i.unmask(t,e)}}catch{}});var Mr=Ve((kp,Ir)=>{"use strict";var Dr=Symbol("kDone"),_a=Symbol("kRun"),Aa=class{constructor(t){this[Dr]=()=>{this.pending--,this[_a]()},this.concurrency=t||1/0,this.jobs=[],this.pending=0}add(t){this.jobs.push(t),this[_a]()}[_a](){if(this.pending!==this.concurrency&&this.jobs.length){let t=this.jobs.shift();this.pending++,t(this[Dr])}}};Ir.exports=Aa});var Xt=Ve((xp,Nr)=>{"use strict";var As=require("zlib"),Lr=_s(),Cc=Mr(),{kStatusCode:Fr}=gt(),Tc=Buffer[Symbol.species],_c=Buffer.from([0,0,255,255]),Cn=Symbol("permessage-deflate"),yt=Symbol("total-length"),Kt=Symbol("callback"),St=Symbol("buffers"),Jt=Symbol("error"),Sn,Ea=class{constructor(t){if(this._options=t||{},this._threshold=this._options.threshold!==void 0?this._options.threshold:1024,this._maxPayload=this._options.maxPayload|0,this._isServer=!!this._options.isServer,this._deflate=null,this._inflate=null,this.params=null,!Sn){let e=this._options.concurrencyLimit!==void 0?this._options.concurrencyLimit:10;Sn=new Cc(e)}}static get extensionName(){return"permessage-deflate"}offer(){let t={};return this._options.serverNoContextTakeover&&(t.server_no_context_takeover=!0),this._options.clientNoContextTakeover&&(t.client_no_context_takeover=!0),this._options.serverMaxWindowBits&&(t.server_max_window_bits=this._options.serverMaxWindowBits),this._options.clientMaxWindowBits?t.client_max_window_bits=this._options.clientMaxWindowBits:this._options.clientMaxWindowBits==null&&(t.client_max_window_bits=!0),t}accept(t){return t=this.normalizeParams(t),this.params=this._isServer?this.acceptAsServer(t):this.acceptAsClient(t),this.params}cleanup(){if(this._inflate&&(this._inflate.close(),this._inflate=null),this._deflate){let t=this._deflate[Kt];this._deflate.close(),this._deflate=null,t&&t(new Error("The deflate stream was closed while data was being processed"))}}acceptAsServer(t){let e=this._options,s=t.find(n=>!(e.serverNoContextTakeover===!1&&n.server_no_context_takeover||n.server_max_window_bits&&(e.serverMaxWindowBits===!1||typeof e.serverMaxWindowBits=="number"&&e.serverMaxWindowBits>n.server_max_window_bits)||typeof e.clientMaxWindowBits=="number"&&!n.client_max_window_bits));if(!s)throw new Error("None of the extension offers can be accepted");return e.serverNoContextTakeover&&(s.server_no_context_takeover=!0),e.clientNoContextTakeover&&(s.client_no_context_takeover=!0),typeof e.serverMaxWindowBits=="number"&&(s.server_max_window_bits=e.serverMaxWindowBits),typeof e.clientMaxWindowBits=="number"?s.client_max_window_bits=e.clientMaxWindowBits:(s.client_max_window_bits===!0||e.clientMaxWindowBits===!1)&&delete s.client_max_window_bits,s}acceptAsClient(t){let e=t[0];if(this._options.clientNoContextTakeover===!1&&e.client_no_context_takeover)throw new Error('Unexpected parameter "client_no_context_takeover"');if(!e.client_max_window_bits)typeof this._options.clientMaxWindowBits=="number"&&(e.client_max_window_bits=this._options.clientMaxWindowBits);else if(this._options.clientMaxWindowBits===!1||typeof this._options.clientMaxWindowBits=="number"&&e.client_max_window_bits>this._options.clientMaxWindowBits)throw new Error('Unexpected or invalid parameter "client_max_window_bits"');return e}normalizeParams(t){return t.forEach(e=>{Object.keys(e).forEach(s=>{let n=e[s];if(n.length>1)throw new Error(`Parameter "${s}" must have only a single value`);if(n=n[0],s==="client_max_window_bits"){if(n!==!0){let a=+n;if(!Number.isInteger(a)||a<8||a>15)throw new TypeError(`Invalid value for parameter "${s}": ${n}`);n=a}else if(!this._isServer)throw new TypeError(`Invalid value for parameter "${s}": ${n}`)}else if(s==="server_max_window_bits"){let a=+n;if(!Number.isInteger(a)||a<8||a>15)throw new TypeError(`Invalid value for parameter "${s}": ${n}`);n=a}else if(s==="client_no_context_takeover"||s==="server_no_context_takeover"){if(n!==!0)throw new TypeError(`Invalid value for parameter "${s}": ${n}`)}else throw new Error(`Unknown parameter "${s}"`);e[s]=n})}),t}decompress(t,e,s){Sn.add(n=>{this._decompress(t,e,(a,r)=>{n(),s(a,r)})})}compress(t,e,s){Sn.add(n=>{this._compress(t,e,(a,r)=>{n(),s(a,r)})})}_decompress(t,e,s){let n=this._isServer?"client":"server";if(!this._inflate){let a=`${n}_max_window_bits`,r=typeof this.params[a]!="number"?As.Z_DEFAULT_WINDOWBITS:this.params[a];this._inflate=As.createInflateRaw({...this._options.zlibInflateOptions,windowBits:r}),this._inflate[Cn]=this,this._inflate[yt]=0,this._inflate[St]=[],this._inflate.on("error",Ec),this._inflate.on("data",Or)}this._inflate[Kt]=s,this._inflate.write(t),e&&this._inflate.write(_c),this._inflate.flush(()=>{let a=this._inflate[Jt];if(a){this._inflate.close(),this._inflate=null,s(a);return}let r=Lr.concat(this._inflate[St],this._inflate[yt]);this._inflate._readableState.endEmitted?(this._inflate.close(),this._inflate=null):(this._inflate[yt]=0,this._inflate[St]=[],e&&this.params[`${n}_no_context_takeover`]&&this._inflate.reset()),s(null,r)})}_compress(t,e,s){let n=this._isServer?"server":"client";if(!this._deflate){let a=`${n}_max_window_bits`,r=typeof this.params[a]!="number"?As.Z_DEFAULT_WINDOWBITS:this.params[a];this._deflate=As.createDeflateRaw({...this._options.zlibDeflateOptions,windowBits:r}),this._deflate[yt]=0,this._deflate[St]=[],this._deflate.on("data",Ac)}this._deflate[Kt]=s,this._deflate.write(t),this._deflate.flush(As.Z_SYNC_FLUSH,()=>{if(!this._deflate)return;let a=Lr.concat(this._deflate[St],this._deflate[yt]);e&&(a=new Tc(a.buffer,a.byteOffset,a.length-4)),this._deflate[Kt]=null,this._deflate[yt]=0,this._deflate[St]=[],e&&this.params[`${n}_no_context_takeover`]&&this._deflate.reset(),s(null,a)})}};Nr.exports=Ea;function Ac(i){this[St].push(i),this[yt]+=i.length}function Or(i){if(this[yt]+=i.length,this[Cn]._maxPayload<1||this[yt]<=this[Cn]._maxPayload){this[St].push(i);return}this[Jt]=new RangeError("Max payload size exceeded"),this[Jt].code="WS_ERR_UNSUPPORTED_MESSAGE_LENGTH",this[Jt][Fr]=1009,this.removeListener("data",Or),this.reset()}function Ec(i){if(this[Cn]._inflate=null,this[Jt]){this[Kt](this[Jt]);return}i[Fr]=1007,this[Kt](i)}});var Qt=Ve((Sp,Tn)=>{"use strict";var{isUtf8:Br}=require("buffer"),{hasBlob:Pc}=gt(),Rc=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,0,1,1,1,1,1,0,0,1,1,0,1,1,0,1,1,1,1,1,1,1,1,1,1,0,0,0,0,0,0,0,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,0,0,0,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,0,1,0,1,0];function Dc(i){return i>=1e3&&i<=1014&&i!==1004&&i!==1005&&i!==1006||i>=3e3&&i<=4999}function Pa(i){let t=i.length,e=0;for(;e<t;)if((i[e]&128)===0)e++;else if((i[e]&224)===192){if(e+1===t||(i[e+1]&192)!==128||(i[e]&254)===192)return!1;e+=2}else if((i[e]&240)===224){if(e+2>=t||(i[e+1]&192)!==128||(i[e+2]&192)!==128||i[e]===224&&(i[e+1]&224)===128||i[e]===237&&(i[e+1]&224)===160)return!1;e+=3}else if((i[e]&248)===240){if(e+3>=t||(i[e+1]&192)!==128||(i[e+2]&192)!==128||(i[e+3]&192)!==128||i[e]===240&&(i[e+1]&240)===128||i[e]===244&&i[e+1]>143||i[e]>244)return!1;e+=4}else return!1;return!0}function Ic(i){return Pc&&typeof i=="object"&&typeof i.arrayBuffer=="function"&&typeof i.type=="string"&&typeof i.stream=="function"&&(i[Symbol.toStringTag]==="Blob"||i[Symbol.toStringTag]==="File")}Tn.exports={isBlob:Ic,isValidStatusCode:Dc,isValidUTF8:Pa,tokenChars:Rc};if(Br)Tn.exports.isValidUTF8=function(i){return i.length<24?Pa(i):Br(i)};else if(!process.env.WS_NO_UTF_8_VALIDATE)try{let i=require("utf-8-validate");Tn.exports.isValidUTF8=function(t){return t.length<32?Pa(t):i(t)}}catch{}});var La=Ve((Cp,zr)=>{"use strict";var{Writable:Mc}=require("stream"),Ur=Xt(),{BINARY_TYPES:Lc,EMPTY_BUFFER:$r,kStatusCode:Fc,kWebSocket:Oc}=gt(),{concat:Ra,toArrayBuffer:Nc,unmask:Bc}=_s(),{isValidStatusCode:Uc,isValidUTF8:jr}=Qt(),_n=Buffer[Symbol.species],et=0,Wr=1,Hr=2,qr=3,Da=4,Ia=5,An=6,Ma=class extends Mc{constructor(t={}){super(),this._allowSynchronousEvents=t.allowSynchronousEvents!==void 0?t.allowSynchronousEvents:!0,this._binaryType=t.binaryType||Lc[0],this._extensions=t.extensions||{},this._isServer=!!t.isServer,this._maxBufferedChunks=t.maxBufferedChunks|0,this._maxFragments=t.maxFragments|0,this._maxPayload=t.maxPayload|0,this._skipUTF8Validation=!!t.skipUTF8Validation,this[Oc]=void 0,this._bufferedBytes=0,this._buffers=[],this._compressed=!1,this._payloadLength=0,this._mask=void 0,this._fragmented=0,this._masked=!1,this._fin=!1,this._opcode=0,this._totalPayloadLength=0,this._messageLength=0,this._fragments=[],this._errored=!1,this._loop=!1,this._state=et}_write(t,e,s){if(this._opcode===8&&this._state==et)return s();if(this._maxBufferedChunks>0&&this._buffers.length>=this._maxBufferedChunks){s(this.createError(RangeError,"Too many buffered chunks",!1,1008,"WS_ERR_TOO_MANY_BUFFERED_PARTS"));return}this._bufferedBytes+=t.length,this._buffers.push(t),this.startLoop(s)}consume(t){if(this._bufferedBytes-=t,t===this._buffers[0].length)return this._buffers.shift();if(t<this._buffers[0].length){let s=this._buffers[0];return this._buffers[0]=new _n(s.buffer,s.byteOffset+t,s.length-t),new _n(s.buffer,s.byteOffset,t)}let e=Buffer.allocUnsafe(t);do{let s=this._buffers[0],n=e.length-t;t>=s.length?e.set(this._buffers.shift(),n):(e.set(new Uint8Array(s.buffer,s.byteOffset,t),n),this._buffers[0]=new _n(s.buffer,s.byteOffset+t,s.length-t)),t-=s.length}while(t>0);return e}startLoop(t){this._loop=!0;do switch(this._state){case et:this.getInfo(t);break;case Wr:this.getPayloadLength16(t);break;case Hr:this.getPayloadLength64(t);break;case qr:this.getMask();break;case Da:this.getData(t);break;case Ia:case An:this._loop=!1;return}while(this._loop);this._errored||t()}getInfo(t){if(this._bufferedBytes<2){this._loop=!1;return}let e=this.consume(2);if((e[0]&48)!==0){let n=this.createError(RangeError,"RSV2 and RSV3 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_2_3");t(n);return}let s=(e[0]&64)===64;if(s&&!this._extensions[Ur.extensionName]){let n=this.createError(RangeError,"RSV1 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_1");t(n);return}if(this._fin=(e[0]&128)===128,this._opcode=e[0]&15,this._payloadLength=e[1]&127,this._opcode===0){if(s){let n=this.createError(RangeError,"RSV1 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_1");t(n);return}if(!this._fragmented){let n=this.createError(RangeError,"invalid opcode 0",!0,1002,"WS_ERR_INVALID_OPCODE");t(n);return}this._opcode=this._fragmented}else if(this._opcode===1||this._opcode===2){if(this._fragmented){let n=this.createError(RangeError,`invalid opcode ${this._opcode}`,!0,1002,"WS_ERR_INVALID_OPCODE");t(n);return}this._compressed=s}else if(this._opcode>7&&this._opcode<11){if(!this._fin){let n=this.createError(RangeError,"FIN must be set",!0,1002,"WS_ERR_EXPECTED_FIN");t(n);return}if(s){let n=this.createError(RangeError,"RSV1 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_1");t(n);return}if(this._payloadLength>125||this._opcode===8&&this._payloadLength===1){let n=this.createError(RangeError,`invalid payload length ${this._payloadLength}`,!0,1002,"WS_ERR_INVALID_CONTROL_PAYLOAD_LENGTH");t(n);return}}else{let n=this.createError(RangeError,`invalid opcode ${this._opcode}`,!0,1002,"WS_ERR_INVALID_OPCODE");t(n);return}if(!this._fin&&!this._fragmented&&(this._fragmented=this._opcode),this._masked=(e[1]&128)===128,this._isServer){if(!this._masked){let n=this.createError(RangeError,"MASK must be set",!0,1002,"WS_ERR_EXPECTED_MASK");t(n);return}}else if(this._masked){let n=this.createError(RangeError,"MASK must be clear",!0,1002,"WS_ERR_UNEXPECTED_MASK");t(n);return}this._payloadLength===126?this._state=Wr:this._payloadLength===127?this._state=Hr:this.haveLength(t)}getPayloadLength16(t){if(this._bufferedBytes<2){this._loop=!1;return}this._payloadLength=this.consume(2).readUInt16BE(0),this.haveLength(t)}getPayloadLength64(t){if(this._bufferedBytes<8){this._loop=!1;return}let e=this.consume(8),s=e.readUInt32BE(0);if(s>Math.pow(2,21)-1){let n=this.createError(RangeError,"Unsupported WebSocket frame: payload length > 2^53 - 1",!1,1009,"WS_ERR_UNSUPPORTED_DATA_PAYLOAD_LENGTH");t(n);return}this._payloadLength=s*Math.pow(2,32)+e.readUInt32BE(4),this.haveLength(t)}haveLength(t){if(this._payloadLength&&this._opcode<8&&(this._totalPayloadLength+=this._payloadLength,this._totalPayloadLength>this._maxPayload&&this._maxPayload>0)){let e=this.createError(RangeError,"Max payload size exceeded",!1,1009,"WS_ERR_UNSUPPORTED_MESSAGE_LENGTH");t(e);return}this._masked?this._state=qr:this._state=Da}getMask(){if(this._bufferedBytes<4){this._loop=!1;return}this._mask=this.consume(4),this._state=Da}getData(t){let e=$r;if(this._payloadLength){if(this._bufferedBytes<this._payloadLength){this._loop=!1;return}e=this.consume(this._payloadLength),this._masked&&(this._mask[0]|this._mask[1]|this._mask[2]|this._mask[3])!==0&&Bc(e,this._mask)}if(this._opcode>7){this.controlMessage(e,t);return}if(this._compressed){this._state=Ia,this.decompress(e,t);return}if(e.length){if(this._maxFragments>0&&this._fragments.length>=this._maxFragments){let s=this.createError(RangeError,"Too many message fragments",!1,1008,"WS_ERR_TOO_MANY_BUFFERED_PARTS");t(s);return}this._messageLength=this._totalPayloadLength,this._fragments.push(e)}this.dataMessage(t)}decompress(t,e){this._extensions[Ur.extensionName].decompress(t,this._fin,(n,a)=>{if(n)return e(n);if(a.length){if(this._messageLength+=a.length,this._messageLength>this._maxPayload&&this._maxPayload>0){let r=this.createError(RangeError,"Max payload size exceeded",!1,1009,"WS_ERR_UNSUPPORTED_MESSAGE_LENGTH");e(r);return}if(this._maxFragments>0&&this._fragments.length>=this._maxFragments){let r=this.createError(RangeError,"Too many message fragments",!1,1008,"WS_ERR_TOO_MANY_BUFFERED_PARTS");e(r);return}this._fragments.push(a)}this.dataMessage(e),this._state===et&&this.startLoop(e)})}dataMessage(t){if(!this._fin){this._state=et;return}let e=this._messageLength,s=this._fragments;if(this._totalPayloadLength=0,this._messageLength=0,this._fragmented=0,this._fragments=[],this._opcode===2){let n;this._binaryType==="nodebuffer"?n=Ra(s,e):this._binaryType==="arraybuffer"?n=Nc(Ra(s,e)):this._binaryType==="blob"?n=new Blob(s):n=s,this._allowSynchronousEvents?(this.emit("message",n,!0),this._state=et):(this._state=An,setImmediate(()=>{this.emit("message",n,!0),this._state=et,this.startLoop(t)}))}else{let n=Ra(s,e);if(!this._skipUTF8Validation&&!jr(n)){let a=this.createError(Error,"invalid UTF-8 sequence",!0,1007,"WS_ERR_INVALID_UTF8");t(a);return}this._state===Ia||this._allowSynchronousEvents?(this.emit("message",n,!1),this._state=et):(this._state=An,setImmediate(()=>{this.emit("message",n,!1),this._state=et,this.startLoop(t)}))}}controlMessage(t,e){if(this._opcode===8){if(t.length===0)this._loop=!1,this.emit("conclude",1005,$r),this.end();else{let s=t.readUInt16BE(0);if(!Uc(s)){let a=this.createError(RangeError,`invalid status code ${s}`,!0,1002,"WS_ERR_INVALID_CLOSE_CODE");e(a);return}let n=new _n(t.buffer,t.byteOffset+2,t.length-2);if(!this._skipUTF8Validation&&!jr(n)){let a=this.createError(Error,"invalid UTF-8 sequence",!0,1007,"WS_ERR_INVALID_UTF8");e(a);return}this._loop=!1,this.emit("conclude",s,n),this.end()}this._state=et;return}this._allowSynchronousEvents?(this.emit(this._opcode===9?"ping":"pong",t),this._state=et):(this._state=An,setImmediate(()=>{this.emit(this._opcode===9?"ping":"pong",t),this._state=et,this.startLoop(e)}))}createError(t,e,s,n,a){this._loop=!1,this._errored=!0;let r=new t(s?`Invalid WebSocket frame: ${e}`:e);return Error.captureStackTrace(r,this.createError),r.code=a,r[Fc]=n,r}};zr.exports=Ma});var Na=Ve((_p,Yr)=>{"use strict";var{Duplex:Tp}=require("stream"),{randomFillSync:$c}=require("crypto"),{types:{isUint8Array:jc}}=require("util"),Gr=Xt(),{EMPTY_BUFFER:Wc,kWebSocket:Hc,NOOP:qc}=gt(),{isBlob:Zt,isValidStatusCode:zc}=Qt(),{mask:Vr,toBuffer:It}=_s(),tt=Symbol("kByteLength"),Gc=Buffer.alloc(4),En=8*1024,Mt,es=En,lt=0,Vc=1,Yc=2,Fa=class i{constructor(t,e,s){this._extensions=e||{},s&&(this._generateMask=s,this._maskBuffer=Buffer.alloc(4)),this._socket=t,this._firstFragment=!0,this._compress=!1,this._bufferedBytes=0,this._queue=[],this._state=lt,this.onerror=qc,this[Hc]=void 0}static frame(t,e){let s,n=!1,a=2,r=!1;e.mask&&(s=e.maskBuffer||Gc,e.generateMask?e.generateMask(s):(es===En&&(Mt===void 0&&(Mt=Buffer.alloc(En)),$c(Mt,0,En),es=0),s[0]=Mt[es++],s[1]=Mt[es++],s[2]=Mt[es++],s[3]=Mt[es++]),r=(s[0]|s[1]|s[2]|s[3])===0,a=6);let o;typeof t=="string"?(!e.mask||r)&&e[tt]!==void 0?o=e[tt]:(t=Buffer.from(t),o=t.length):(o=t.length,n=e.mask&&e.readOnly&&!r);let c=o;o>=65536?(a+=8,c=127):o>125&&(a+=2,c=126);let l=Buffer.allocUnsafe(n?o+a:a);return l[0]=e.fin?e.opcode|128:e.opcode,e.rsv1&&(l[0]|=64),l[1]=c,c===126?l.writeUInt16BE(o,2):c===127&&(l[2]=l[3]=0,l.writeUIntBE(o,4,6)),e.mask?(l[1]|=128,l[a-4]=s[0],l[a-3]=s[1],l[a-2]=s[2],l[a-1]=s[3],r?[l,t]:n?(Vr(t,s,l,a,o),[l]):(Vr(t,s,t,0,o),[l,t])):[l,t]}close(t,e,s,n){let a;if(t===void 0)a=Wc;else{if(typeof t!="number"||!zc(t))throw new TypeError("First argument must be a valid error code number");if(e===void 0||!e.length)a=Buffer.allocUnsafe(2),a.writeUInt16BE(t,0);else{let o=Buffer.byteLength(e);if(o>123)throw new RangeError("The message must not be greater than 123 bytes");if(a=Buffer.allocUnsafe(2+o),a.writeUInt16BE(t,0),typeof e=="string")a.write(e,2);else if(jc(e))a.set(e,2);else throw new TypeError("Second argument must be a string or a Uint8Array")}}let r={[tt]:a.length,fin:!0,generateMask:this._generateMask,mask:s,maskBuffer:this._maskBuffer,opcode:8,readOnly:!1,rsv1:!1};this._state!==lt?this.enqueue([this.dispatch,a,!1,r,n]):this.sendFrame(i.frame(a,r),n)}ping(t,e,s){let n,a;if(typeof t=="string"?(n=Buffer.byteLength(t),a=!1):Zt(t)?(n=t.size,a=!1):(t=It(t),n=t.length,a=It.readOnly),n>125)throw new RangeError("The data size must not be greater than 125 bytes");let r={[tt]:n,fin:!0,generateMask:this._generateMask,mask:e,maskBuffer:this._maskBuffer,opcode:9,readOnly:a,rsv1:!1};Zt(t)?this._state!==lt?this.enqueue([this.getBlobData,t,!1,r,s]):this.getBlobData(t,!1,r,s):this._state!==lt?this.enqueue([this.dispatch,t,!1,r,s]):this.sendFrame(i.frame(t,r),s)}pong(t,e,s){let n,a;if(typeof t=="string"?(n=Buffer.byteLength(t),a=!1):Zt(t)?(n=t.size,a=!1):(t=It(t),n=t.length,a=It.readOnly),n>125)throw new RangeError("The data size must not be greater than 125 bytes");let r={[tt]:n,fin:!0,generateMask:this._generateMask,mask:e,maskBuffer:this._maskBuffer,opcode:10,readOnly:a,rsv1:!1};Zt(t)?this._state!==lt?this.enqueue([this.getBlobData,t,!1,r,s]):this.getBlobData(t,!1,r,s):this._state!==lt?this.enqueue([this.dispatch,t,!1,r,s]):this.sendFrame(i.frame(t,r),s)}send(t,e,s){let n=this._extensions[Gr.extensionName],a=e.binary?2:1,r=e.compress,o,c;typeof t=="string"?(o=Buffer.byteLength(t),c=!1):Zt(t)?(o=t.size,c=!1):(t=It(t),o=t.length,c=It.readOnly),this._firstFragment?(this._firstFragment=!1,r&&n&&n.params[n._isServer?"server_no_context_takeover":"client_no_context_takeover"]&&(r=o>=n._threshold),this._compress=r):(r=!1,a=0),e.fin&&(this._firstFragment=!0);let l={[tt]:o,fin:e.fin,generateMask:this._generateMask,mask:e.mask,maskBuffer:this._maskBuffer,opcode:a,readOnly:c,rsv1:r};Zt(t)?this._state!==lt?this.enqueue([this.getBlobData,t,this._compress,l,s]):this.getBlobData(t,this._compress,l,s):this._state!==lt?this.enqueue([this.dispatch,t,this._compress,l,s]):this.dispatch(t,this._compress,l,s)}getBlobData(t,e,s,n){this._bufferedBytes+=s[tt],this._state=Yc,t.arrayBuffer().then(a=>{if(this._socket.destroyed){let o=new Error("The socket was closed while the blob was being read");process.nextTick(Oa,this,o,n);return}this._bufferedBytes-=s[tt];let r=It(a);e?this.dispatch(r,e,s,n):(this._state=lt,this.sendFrame(i.frame(r,s),n),this.dequeue())}).catch(a=>{process.nextTick(Kc,this,a,n)})}dispatch(t,e,s,n){if(!e){this.sendFrame(i.frame(t,s),n);return}let a=this._extensions[Gr.extensionName];this._bufferedBytes+=s[tt],this._state=Vc,a.compress(t,s.fin,(r,o)=>{if(this._socket.destroyed){let c=new Error("The socket was closed while data was being compressed");Oa(this,c,n);return}this._bufferedBytes-=s[tt],this._state=lt,s.readOnly=!1,this.sendFrame(i.frame(o,s),n),this.dequeue()})}dequeue(){for(;this._state===lt&&this._queue.length;){let t=this._queue.shift();this._bufferedBytes-=t[3][tt],Reflect.apply(t[0],this,t.slice(1))}}enqueue(t){this._bufferedBytes+=t[3][tt],this._queue.push(t)}sendFrame(t,e){t.length===2?(this._socket.cork(),this._socket.write(t[0]),this._socket.write(t[1],e),this._socket.uncork()):this._socket.write(t[0],e)}};Yr.exports=Fa;function Oa(i,t,e){typeof e=="function"&&e(t);for(let s=0;s<i._queue.length;s++){let n=i._queue[s],a=n[n.length-1];typeof a=="function"&&a(t)}}function Kc(i,t,e){Oa(i,t,e),i.onerror(t)}});var no=Ve((Ap,so)=>{"use strict";var{kForOnEventAttribute:Es,kListener:Ba}=gt(),Kr=Symbol("kCode"),Jr=Symbol("kData"),Xr=Symbol("kError"),Qr=Symbol("kMessage"),Zr=Symbol("kReason"),ts=Symbol("kTarget"),eo=Symbol("kType"),to=Symbol("kWasClean"),vt=class{constructor(t){this[ts]=null,this[eo]=t}get target(){return this[ts]}get type(){return this[eo]}};Object.defineProperty(vt.prototype,"target",{enumerable:!0});Object.defineProperty(vt.prototype,"type",{enumerable:!0});var Lt=class extends vt{constructor(t,e={}){super(t),this[Kr]=e.code===void 0?0:e.code,this[Zr]=e.reason===void 0?"":e.reason,this[to]=e.wasClean===void 0?!1:e.wasClean}get code(){return this[Kr]}get reason(){return this[Zr]}get wasClean(){return this[to]}};Object.defineProperty(Lt.prototype,"code",{enumerable:!0});Object.defineProperty(Lt.prototype,"reason",{enumerable:!0});Object.defineProperty(Lt.prototype,"wasClean",{enumerable:!0});var ss=class extends vt{constructor(t,e={}){super(t),this[Xr]=e.error===void 0?null:e.error,this[Qr]=e.message===void 0?"":e.message}get error(){return this[Xr]}get message(){return this[Qr]}};Object.defineProperty(ss.prototype,"error",{enumerable:!0});Object.defineProperty(ss.prototype,"message",{enumerable:!0});var Ps=class extends vt{constructor(t,e={}){super(t),this[Jr]=e.data===void 0?null:e.data}get data(){return this[Jr]}};Object.defineProperty(Ps.prototype,"data",{enumerable:!0});var Jc={addEventListener(i,t,e={}){for(let n of this.listeners(i))if(!e[Es]&&n[Ba]===t&&!n[Es])return;let s;if(i==="message")s=function(a,r){let o=new Ps("message",{data:r?a:a.toString()});o[ts]=this,Pn(t,this,o)};else if(i==="close")s=function(a,r){let o=new Lt("close",{code:a,reason:r.toString(),wasClean:this._closeFrameReceived&&this._closeFrameSent});o[ts]=this,Pn(t,this,o)};else if(i==="error")s=function(a){let r=new ss("error",{error:a,message:a.message});r[ts]=this,Pn(t,this,r)};else if(i==="open")s=function(){let a=new vt("open");a[ts]=this,Pn(t,this,a)};else return;s[Es]=!!e[Es],s[Ba]=t,e.once?this.once(i,s):this.on(i,s)},removeEventListener(i,t){for(let e of this.listeners(i))if(e[Ba]===t&&!e[Es]){this.removeListener(i,e);break}}};so.exports={CloseEvent:Lt,ErrorEvent:ss,Event:vt,EventTarget:Jc,MessageEvent:Ps};function Pn(i,t,e){typeof i=="object"&&i.handleEvent?i.handleEvent.call(i,e):i.call(t,e)}});var Rn=Ve((Ep,ao)=>{"use strict";var{tokenChars:Rs}=Qt();function pt(i,t,e){i[t]===void 0?i[t]=[e]:i[t].push(e)}function Xc(i){let t=Object.create(null),e=Object.create(null),s=!1,n=!1,a=!1,r,o,c=-1,l=-1,h=-1,d=0;for(;d<i.length;d++)if(l=i.charCodeAt(d),r===void 0)if(h===-1&&Rs[l]===1)c===-1&&(c=d);else if(d!==0&&(l===32||l===9))h===-1&&c!==-1&&(h=d);else if(l===59||l===44){if(c===-1)throw new SyntaxError(`Unexpected character at index ${d}`);h===-1&&(h=d);let p=i.slice(c,h);l===44?(pt(t,p,e),e=Object.create(null)):r=p,c=h=-1}else throw new SyntaxError(`Unexpected character at index ${d}`);else if(o===void 0)if(h===-1&&Rs[l]===1)c===-1&&(c=d);else if(l===32||l===9)h===-1&&c!==-1&&(h=d);else if(l===59||l===44){if(c===-1)throw new SyntaxError(`Unexpected character at index ${d}`);h===-1&&(h=d),pt(e,i.slice(c,h),!0),l===44&&(pt(t,r,e),e=Object.create(null),r=void 0),c=h=-1}else if(l===61&&c!==-1&&h===-1)o=i.slice(c,d),c=h=-1;else throw new SyntaxError(`Unexpected character at index ${d}`);else if(n){if(Rs[l]!==1)throw new SyntaxError(`Unexpected character at index ${d}`);c===-1?c=d:s||(s=!0),n=!1}else if(a)if(Rs[l]===1)c===-1&&(c=d);else if(l===34&&c!==-1)a=!1,h=d;else if(l===92)n=!0;else throw new SyntaxError(`Unexpected character at index ${d}`);else if(l===34&&i.charCodeAt(d-1)===61)a=!0;else if(h===-1&&Rs[l]===1)c===-1&&(c=d);else if(c!==-1&&(l===32||l===9))h===-1&&(h=d);else if(l===59||l===44){if(c===-1)throw new SyntaxError(`Unexpected character at index ${d}`);h===-1&&(h=d);let p=i.slice(c,h);s&&(p=p.replace(/\\/g,""),s=!1),pt(e,o,p),l===44&&(pt(t,r,e),e=Object.create(null),r=void 0),o=void 0,c=h=-1}else throw new SyntaxError(`Unexpected character at index ${d}`);if(c===-1||a||l===32||l===9)throw new SyntaxError("Unexpected end of input");h===-1&&(h=d);let u=i.slice(c,h);return r===void 0?pt(t,u,e):(o===void 0?pt(e,u,!0):s?pt(e,o,u.replace(/\\/g,"")):pt(e,o,u),pt(t,r,e)),t}function Qc(i){return Object.keys(i).map(t=>{let e=i[t];return Array.isArray(e)||(e=[e]),e.map(s=>[t].concat(Object.keys(s).map(n=>{let a=s[n];return Array.isArray(a)||(a=[a]),a.map(r=>r===!0?n:`${n}=${r}`).join("; ")})).join("; ")).join(", ")}).join(", ")}ao.exports={format:Qc,parse:Xc}});var Ln=Ve((Dp,yo)=>{"use strict";var Zc=require("events"),ed=require("https"),td=require("http"),oo=require("net"),sd=require("tls"),{randomBytes:nd,createHash:ad}=require("crypto"),{Duplex:Pp,Readable:Rp}=require("stream"),{URL:Ua}=require("url"),Ct=Xt(),id=La(),rd=Na(),{isBlob:od}=Qt(),{BINARY_TYPES:io,CLOSE_TIMEOUT:ld,EMPTY_BUFFER:Dn,GUID:cd,kForOnEventAttribute:$a,kListener:dd,kStatusCode:hd,kWebSocket:De,NOOP:lo}=gt(),{EventTarget:{addEventListener:ud,removeEventListener:pd}}=no(),{format:md,parse:fd}=Rn(),{toBuffer:gd}=_s(),co=Symbol("kAborted"),ja=[8,13],wt=["CONNECTING","OPEN","CLOSING","CLOSED"],yd=/^[!#$%&'*+\-.0-9A-Z^_`|a-z~]+$/,be=class i extends Zc{constructor(t,e,s){super(),this._binaryType=io[0],this._closeCode=1006,this._closeFrameReceived=!1,this._closeFrameSent=!1,this._closeMessage=Dn,this._closeTimer=null,this._errorEmitted=!1,this._extensions={},this._paused=!1,this._protocol="",this._readyState=i.CONNECTING,this._receiver=null,this._sender=null,this._socket=null,t!==null?(this._bufferedAmount=0,this._isServer=!1,this._redirects=0,e===void 0?e=[]:Array.isArray(e)||(typeof e=="object"&&e!==null?(s=e,e=[]):e=[e]),ho(this,t,e,s)):(this._autoPong=s.autoPong,this._closeTimeout=s.closeTimeout,this._isServer=!0)}get binaryType(){return this._binaryType}set binaryType(t){io.includes(t)&&(this._binaryType=t,this._receiver&&(this._receiver._binaryType=t))}get bufferedAmount(){return this._socket?this._socket._writableState.length+this._sender._bufferedBytes:this._bufferedAmount}get extensions(){return Object.keys(this._extensions).join()}get isPaused(){return this._paused}get onclose(){return null}get onerror(){return null}get onopen(){return null}get onmessage(){return null}get protocol(){return this._protocol}get readyState(){return this._readyState}get url(){return this._url}setSocket(t,e,s){let n=new id({allowSynchronousEvents:s.allowSynchronousEvents,binaryType:this.binaryType,extensions:this._extensions,isServer:this._isServer,maxBufferedChunks:s.maxBufferedChunks,maxFragments:s.maxFragments,maxPayload:s.maxPayload,skipUTF8Validation:s.skipUTF8Validation}),a=new rd(t,this._extensions,s.generateMask);this._receiver=n,this._sender=a,this._socket=t,n[De]=this,a[De]=this,t[De]=this,n.on("conclude",bd),n.on("drain",kd),n.on("error",xd),n.on("message",Sd),n.on("ping",Cd),n.on("pong",Td),a.onerror=_d,t.setTimeout&&t.setTimeout(0),t.setNoDelay&&t.setNoDelay(),e.length>0&&t.unshift(e),t.on("close",mo),t.on("data",Mn),t.on("end",fo),t.on("error",go),this._readyState=i.OPEN,this.emit("open")}emitClose(){if(!this._socket){this._readyState=i.CLOSED,this.emit("close",this._closeCode,this._closeMessage);return}this._extensions[Ct.extensionName]&&this._extensions[Ct.extensionName].cleanup(),this._receiver.removeAllListeners(),this._readyState=i.CLOSED,this.emit("close",this._closeCode,this._closeMessage)}close(t,e){if(this.readyState!==i.CLOSED){if(this.readyState===i.CONNECTING){Qe(this,this._req,"WebSocket was closed before the connection was established");return}if(this.readyState===i.CLOSING){this._closeFrameSent&&(this._closeFrameReceived||this._receiver._writableState.errorEmitted)&&this._socket.end();return}this._readyState=i.CLOSING,this._sender.close(t,e,!this._isServer,s=>{s||(this._closeFrameSent=!0,(this._closeFrameReceived||this._receiver._writableState.errorEmitted)&&this._socket.end())}),po(this)}}pause(){this.readyState===i.CONNECTING||this.readyState===i.CLOSED||(this._paused=!0,this._socket.pause())}ping(t,e,s){if(this.readyState===i.CONNECTING)throw new Error("WebSocket is not open: readyState 0 (CONNECTING)");if(typeof t=="function"?(s=t,t=e=void 0):typeof e=="function"&&(s=e,e=void 0),typeof t=="number"&&(t=t.toString()),this.readyState!==i.OPEN){Wa(this,t,s);return}e===void 0&&(e=!this._isServer),this._sender.ping(t||Dn,e,s)}pong(t,e,s){if(this.readyState===i.CONNECTING)throw new Error("WebSocket is not open: readyState 0 (CONNECTING)");if(typeof t=="function"?(s=t,t=e=void 0):typeof e=="function"&&(s=e,e=void 0),typeof t=="number"&&(t=t.toString()),this.readyState!==i.OPEN){Wa(this,t,s);return}e===void 0&&(e=!this._isServer),this._sender.pong(t||Dn,e,s)}resume(){this.readyState===i.CONNECTING||this.readyState===i.CLOSED||(this._paused=!1,this._receiver._writableState.needDrain||this._socket.resume())}send(t,e,s){if(this.readyState===i.CONNECTING)throw new Error("WebSocket is not open: readyState 0 (CONNECTING)");if(typeof e=="function"&&(s=e,e={}),typeof t=="number"&&(t=t.toString()),this.readyState!==i.OPEN){Wa(this,t,s);return}let n={binary:typeof t!="string",mask:!this._isServer,compress:!0,fin:!0,...e};this._extensions[Ct.extensionName]||(n.compress=!1),this._sender.send(t||Dn,n,s)}terminate(){if(this.readyState!==i.CLOSED){if(this.readyState===i.CONNECTING){Qe(this,this._req,"WebSocket was closed before the connection was established");return}this._socket&&(this._readyState=i.CLOSING,this._socket.destroy())}}};Object.defineProperty(be,"CONNECTING",{enumerable:!0,value:wt.indexOf("CONNECTING")});Object.defineProperty(be.prototype,"CONNECTING",{enumerable:!0,value:wt.indexOf("CONNECTING")});Object.defineProperty(be,"OPEN",{enumerable:!0,value:wt.indexOf("OPEN")});Object.defineProperty(be.prototype,"OPEN",{enumerable:!0,value:wt.indexOf("OPEN")});Object.defineProperty(be,"CLOSING",{enumerable:!0,value:wt.indexOf("CLOSING")});Object.defineProperty(be.prototype,"CLOSING",{enumerable:!0,value:wt.indexOf("CLOSING")});Object.defineProperty(be,"CLOSED",{enumerable:!0,value:wt.indexOf("CLOSED")});Object.defineProperty(be.prototype,"CLOSED",{enumerable:!0,value:wt.indexOf("CLOSED")});["binaryType","bufferedAmount","extensions","isPaused","protocol","readyState","url"].forEach(i=>{Object.defineProperty(be.prototype,i,{enumerable:!0})});["open","error","close","message"].forEach(i=>{Object.defineProperty(be.prototype,`on${i}`,{enumerable:!0,get(){for(let t of this.listeners(i))if(t[$a])return t[dd];return null},set(t){for(let e of this.listeners(i))if(e[$a]){this.removeListener(i,e);break}typeof t=="function"&&this.addEventListener(i,t,{[$a]:!0})}})});be.prototype.addEventListener=ud;be.prototype.removeEventListener=pd;yo.exports=be;function ho(i,t,e,s){let n={allowSynchronousEvents:!0,autoPong:!0,closeTimeout:ld,protocolVersion:ja[1],maxBufferedChunks:1048576,maxFragments:131072,maxPayload:104857600,skipUTF8Validation:!1,perMessageDeflate:!0,followRedirects:!1,maxRedirects:10,...s,socketPath:void 0,hostname:void 0,protocol:void 0,timeout:void 0,method:"GET",host:void 0,path:void 0,port:void 0};if(i._autoPong=n.autoPong,i._closeTimeout=n.closeTimeout,!ja.includes(n.protocolVersion))throw new RangeError(`Unsupported protocol version: ${n.protocolVersion} (supported versions: ${ja.join(", ")})`);let a;if(t instanceof Ua)a=t;else try{a=new Ua(t)}catch{throw new SyntaxError(`Invalid URL: ${t}`)}a.protocol==="http:"?a.protocol="ws:":a.protocol==="https:"&&(a.protocol="wss:"),i._url=a.href;let r=a.protocol==="wss:",o=a.protocol==="ws+unix:",c;if(a.protocol!=="ws:"&&!r&&!o?c=`The URL's protocol must be one of "ws:", "wss:", "http:", "https:", or "ws+unix:"`:o&&!a.pathname?c="The URL's pathname is empty":a.hash&&(c="The URL contains a fragment identifier"),c){let f=new SyntaxError(c);if(i._redirects===0)throw f;In(i,f);return}let l=r?443:80,h=nd(16).toString("base64"),d=r?ed.request:td.request,u=new Set,p;if(n.createConnection=n.createConnection||(r?wd:vd),n.defaultPort=n.defaultPort||l,n.port=a.port||l,n.host=a.hostname.startsWith("[")?a.hostname.slice(1,-1):a.hostname,n.headers={...n.headers,"Sec-WebSocket-Version":n.protocolVersion,"Sec-WebSocket-Key":h,Connection:"Upgrade",Upgrade:"websocket"},n.path=a.pathname+a.search,n.timeout=n.handshakeTimeout,n.perMessageDeflate&&(p=new Ct({...n.perMessageDeflate,isServer:!1,maxPayload:n.maxPayload}),n.headers["Sec-WebSocket-Extensions"]=md({[Ct.extensionName]:p.offer()})),e.length){for(let f of e){if(typeof f!="string"||!yd.test(f)||u.has(f))throw new SyntaxError("An invalid or duplicated subprotocol was specified");u.add(f)}n.headers["Sec-WebSocket-Protocol"]=e.join(",")}if(n.origin&&(n.protocolVersion<13?n.headers["Sec-WebSocket-Origin"]=n.origin:n.headers.Origin=n.origin),(a.username||a.password)&&(n.auth=`${a.username}:${a.password}`),o){let f=n.path.split(":");n.socketPath=f[0],n.path=f[1]}let m;if(n.followRedirects){if(i._redirects===0){i._originalIpc=o,i._originalSecure=r,i._originalHostOrSocketPath=o?n.socketPath:a.host;let f=s&&s.headers;if(s={...s,headers:{}},f)for(let[g,y]of Object.entries(f))s.headers[g.toLowerCase()]=y}else if(i.listenerCount("redirect")===0){let f=o?i._originalIpc?n.socketPath===i._originalHostOrSocketPath:!1:i._originalIpc?!1:a.host===i._originalHostOrSocketPath;(!f||i._originalSecure&&!r)&&(delete n.headers.authorization,delete n.headers.cookie,f||delete n.headers.host,n.auth=void 0)}n.auth&&!s.headers.authorization&&(s.headers.authorization="Basic "+Buffer.from(n.auth).toString("base64")),m=i._req=d(n),i._redirects&&i.emit("redirect",i.url,m)}else m=i._req=d(n);n.timeout&&m.on("timeout",()=>{Qe(i,m,"Opening handshake has timed out")}),m.on("error",f=>{m===null||m[co]||(m=i._req=null,In(i,f))}),m.on("response",f=>{let g=f.headers.location,y=f.statusCode;if(g&&n.followRedirects&&y>=300&&y<400){if(++i._redirects>n.maxRedirects){Qe(i,m,"Maximum redirects exceeded");return}m.abort();let v;try{v=new Ua(g,t)}catch{let w=new SyntaxError(`Invalid URL: ${g}`);In(i,w);return}ho(i,v,e,s)}else i.emit("unexpected-response",m,f)||Qe(i,m,`Unexpected server response: ${f.statusCode}`)}),m.on("upgrade",(f,g,y)=>{if(i.emit("upgrade",f),i.readyState!==be.CONNECTING)return;m=i._req=null;let v=f.headers.upgrade;if(v===void 0||v.toLowerCase()!=="websocket"){Qe(i,g,"Invalid Upgrade header");return}let k=ad("sha1").update(h+cd).digest("base64");if(f.headers["sec-websocket-accept"]!==k){Qe(i,g,"Invalid Sec-WebSocket-Accept header");return}let w=f.headers["sec-websocket-protocol"],S;if(w!==void 0?u.size?u.has(w)||(S="Server sent an invalid subprotocol"):S="Server sent a subprotocol but none was requested":u.size&&(S="Server sent no subprotocol"),S){Qe(i,g,S);return}w&&(i._protocol=w);let T=f.headers["sec-websocket-extensions"];if(T!==void 0){if(!p){Qe(i,g,"Server sent a Sec-WebSocket-Extensions header but no extension was requested");return}let _;try{_=fd(T)}catch{Qe(i,g,"Invalid Sec-WebSocket-Extensions header");return}let D=Object.keys(_);if(D.length!==1||D[0]!==Ct.extensionName){Qe(i,g,"Server indicated an extension that was not requested");return}try{p.accept(_[Ct.extensionName])}catch{Qe(i,g,"Invalid Sec-WebSocket-Extensions header");return}i._extensions[Ct.extensionName]=p}i.setSocket(g,y,{allowSynchronousEvents:n.allowSynchronousEvents,generateMask:n.generateMask,maxBufferedChunks:n.maxBufferedChunks,maxFragments:n.maxFragments,maxPayload:n.maxPayload,skipUTF8Validation:n.skipUTF8Validation})}),n.finishRequest?n.finishRequest(m,i):m.end()}function In(i,t){i._readyState=be.CLOSING,i._errorEmitted=!0,i.emit("error",t),i.emitClose()}function vd(i){return i.path=i.socketPath,oo.connect(i)}function wd(i){return i.path=void 0,!i.servername&&i.servername!==""&&(i.servername=oo.isIP(i.host)?"":i.host),sd.connect(i)}function Qe(i,t,e){i._readyState=be.CLOSING;let s=new Error(e);Error.captureStackTrace(s,Qe),t.setHeader?(t[co]=!0,t.abort(),t.socket&&!t.socket.destroyed&&t.socket.destroy(),process.nextTick(In,i,s)):(t.destroy(s),t.once("error",i.emit.bind(i,"error")),t.once("close",i.emitClose.bind(i)))}function Wa(i,t,e){if(t){let s=od(t)?t.size:gd(t).length;i._socket?i._sender._bufferedBytes+=s:i._bufferedAmount+=s}if(e){let s=new Error(`WebSocket is not open: readyState ${i.readyState} (${wt[i.readyState]})`);process.nextTick(e,s)}}function bd(i,t){let e=this[De];e._closeFrameReceived=!0,e._closeMessage=t,e._closeCode=i,e._socket[De]!==void 0&&(e._socket.removeListener("data",Mn),process.nextTick(uo,e._socket),i===1005?e.close():e.close(i,t))}function kd(){let i=this[De];i.isPaused||i._socket.resume()}function xd(i){let t=this[De];t._socket[De]!==void 0&&(t._socket.removeListener("data",Mn),process.nextTick(uo,t._socket),t.close(i[hd])),t._errorEmitted||(t._errorEmitted=!0,t.emit("error",i))}function ro(){this[De].emitClose()}function Sd(i,t){this[De].emit("message",i,t)}function Cd(i){let t=this[De];t._autoPong&&t.pong(i,!this._isServer,lo),t.emit("ping",i)}function Td(i){this[De].emit("pong",i)}function uo(i){i.resume()}function _d(i){let t=this[De];t.readyState!==be.CLOSED&&(t.readyState===be.OPEN&&(t._readyState=be.CLOSING,po(t)),this._socket.end(),t._errorEmitted||(t._errorEmitted=!0,t.emit("error",i)))}function po(i){i._closeTimer=setTimeout(i._socket.destroy.bind(i._socket),i._closeTimeout)}function mo(){let i=this[De];if(this.removeListener("close",mo),this.removeListener("data",Mn),this.removeListener("end",fo),i._readyState=be.CLOSING,!this._readableState.endEmitted&&!i._closeFrameReceived&&!i._receiver._writableState.errorEmitted&&this._readableState.length!==0){let t=this.read(this._readableState.length);i._receiver.write(t)}i._receiver.end(),this[De]=void 0,clearTimeout(i._closeTimer),i._receiver._writableState.finished||i._receiver._writableState.errorEmitted?i.emitClose():(i._receiver.on("error",ro),i._receiver.on("finish",ro))}function Mn(i){this[De]._receiver.write(i)||this.pause()}function fo(){let i=this[De];i._readyState=be.CLOSING,i._receiver.end(),this.end()}function go(){let i=this[De];this.removeListener("error",go),this.on("error",lo),i&&(i._readyState=be.CLOSING,this.destroy())}});var ko=Ve((Mp,bo)=>{"use strict";var Ip=Ln(),{Duplex:Ad}=require("stream");function vo(i){i.emit("close")}function Ed(){!this.destroyed&&this._writableState.finished&&this.destroy()}function wo(i){this.removeListener("error",wo),this.destroy(),this.listenerCount("error")===0&&this.emit("error",i)}function Pd(i,t){let e=!0,s=new Ad({...t,autoDestroy:!1,emitClose:!1,objectMode:!1,writableObjectMode:!1});return i.on("message",function(a,r){let o=!r&&s._readableState.objectMode?a.toString():a;s.push(o)||i.pause()}),i.once("error",function(a){s.destroyed||(e=!1,s.destroy(a))}),i.once("close",function(){s.destroyed||s.push(null)}),s._destroy=function(n,a){if(i.readyState===i.CLOSED){a(n),process.nextTick(vo,s);return}let r=!1;i.once("error",function(c){r=!0,a(c)}),i.once("close",function(){r||a(n),process.nextTick(vo,s)}),e&&i.terminate()},s._final=function(n){if(i.readyState===i.CONNECTING){i.once("open",function(){s._final(n)});return}i._socket!==null&&(i._socket._writableState.finished?(n(),s._readableState.endEmitted&&s.destroy()):(i._socket.once("finish",function(){n()}),i.close()))},s._read=function(){i.isPaused&&i.resume()},s._write=function(n,a,r){if(i.readyState===i.CONNECTING){i.once("open",function(){s._write(n,a,r)});return}i.send(n,r)},s.on("end",Ed),s.on("error",wo),s}bo.exports=Pd});var Ha=Ve((Lp,xo)=>{"use strict";var{tokenChars:Rd}=Qt();function Dd(i){let t=new Set,e=-1,s=-1,n=0;for(n;n<i.length;n++){let r=i.charCodeAt(n);if(s===-1&&Rd[r]===1)e===-1&&(e=n);else if(n!==0&&(r===32||r===9))s===-1&&e!==-1&&(s=n);else if(r===44){if(e===-1)throw new SyntaxError(`Unexpected character at index ${n}`);s===-1&&(s=n);let o=i.slice(e,s);if(t.has(o))throw new SyntaxError(`The "${o}" subprotocol is duplicated`);t.add(o),e=s=-1}else throw new SyntaxError(`Unexpected character at index ${n}`)}if(e===-1||s!==-1)throw new SyntaxError("Unexpected end of input");let a=i.slice(e,n);if(t.has(a))throw new SyntaxError(`The "${a}" subprotocol is duplicated`);return t.add(a),t}xo.exports={parse:Dd}});var Po=Ve((Op,Eo)=>{"use strict";var Id=require("events"),Fn=require("http"),{Duplex:Fp}=require("stream"),{createHash:Md}=require("crypto"),So=Rn(),Ft=Xt(),Ld=Ha(),Fd=Ln(),{CLOSE_TIMEOUT:Od,GUID:Nd,kWebSocket:Bd}=gt(),Ud=/^[+/0-9A-Za-z]{22}==$/,Co=0,To=1,Ao=2,qa=class extends Id{constructor(t,e){if(super(),t={allowSynchronousEvents:!0,autoPong:!0,maxBufferedChunks:1024*1024,maxFragments:128*1024,maxPayload:100*1024*1024,skipUTF8Validation:!1,perMessageDeflate:!1,handleProtocols:null,clientTracking:!0,closeTimeout:Od,verifyClient:null,noServer:!1,backlog:null,server:null,host:null,path:null,port:null,WebSocket:Fd,...t},t.port==null&&!t.server&&!t.noServer||t.port!=null&&(t.server||t.noServer)||t.server&&t.noServer)throw new TypeError('One and only one of the "port", "server", or "noServer" options must be specified');if(t.port!=null?(this._server=Fn.createServer((s,n)=>{let a=Fn.STATUS_CODES[426];n.writeHead(426,{"Content-Length":a.length,"Content-Type":"text/plain"}),n.end(a)}),this._server.listen(t.port,t.host,t.backlog,e)):t.server&&(this._server=t.server),this._server){let s=this.emit.bind(this,"connection");this._removeListeners=$d(this._server,{listening:this.emit.bind(this,"listening"),error:this.emit.bind(this,"error"),upgrade:(n,a,r)=>{this.handleUpgrade(n,a,r,s)}})}t.perMessageDeflate===!0&&(t.perMessageDeflate={}),t.clientTracking&&(this.clients=new Set,this._shouldEmitClose=!1),this.options=t,this._state=Co}address(){if(this.options.noServer)throw new Error('The server is operating in "noServer" mode');return this._server?this._server.address():null}close(t){if(this._state===Ao){t&&this.once("close",()=>{t(new Error("The server is not running"))}),process.nextTick(Ds,this);return}if(t&&this.once("close",t),this._state!==To)if(this._state=To,this.options.noServer||this.options.server)this._server&&(this._removeListeners(),this._removeListeners=this._server=null),this.clients?this.clients.size?this._shouldEmitClose=!0:process.nextTick(Ds,this):process.nextTick(Ds,this);else{let e=this._server;this._removeListeners(),this._removeListeners=this._server=null,e.close(()=>{Ds(this)})}}shouldHandle(t){if(this.options.path){let e=t.url.indexOf("?");if((e!==-1?t.url.slice(0,e):t.url)!==this.options.path)return!1}return!0}handleUpgrade(t,e,s,n){e.on("error",_o);let a=t.headers["sec-websocket-key"],r=t.headers.upgrade,o=+t.headers["sec-websocket-version"];if(t.method!=="GET"){Ot(this,t,e,405,"Invalid HTTP method");return}if(r===void 0||r.toLowerCase()!=="websocket"){Ot(this,t,e,400,"Invalid Upgrade header");return}if(a===void 0||!Ud.test(a)){Ot(this,t,e,400,"Missing or invalid Sec-WebSocket-Key header");return}if(o!==13&&o!==8){Ot(this,t,e,400,"Missing or invalid Sec-WebSocket-Version header",{"Sec-WebSocket-Version":"13, 8"});return}if(!this.shouldHandle(t)){Is(e,400);return}let c=t.headers["sec-websocket-protocol"],l=new Set;if(c!==void 0)try{l=Ld.parse(c)}catch{Ot(this,t,e,400,"Invalid Sec-WebSocket-Protocol header");return}let h=t.headers["sec-websocket-extensions"],d={};if(this.options.perMessageDeflate&&h!==void 0){let u=new Ft({...this.options.perMessageDeflate,isServer:!0,maxPayload:this.options.maxPayload});try{let p=So.parse(h);p[Ft.extensionName]&&(u.accept(p[Ft.extensionName]),d[Ft.extensionName]=u)}catch{Ot(this,t,e,400,"Invalid or unacceptable Sec-WebSocket-Extensions header");return}}if(this.options.verifyClient){let u={origin:t.headers[`${o===8?"sec-websocket-origin":"origin"}`],secure:!!(t.socket.authorized||t.socket.encrypted),req:t};if(this.options.verifyClient.length===2){this.options.verifyClient(u,(p,m,f,g)=>{if(!p)return Is(e,m||401,f,g);this.completeUpgrade(d,a,l,t,e,s,n)});return}if(!this.options.verifyClient(u))return Is(e,401)}this.completeUpgrade(d,a,l,t,e,s,n)}completeUpgrade(t,e,s,n,a,r,o){if(!a.readable||!a.writable)return a.destroy();if(a[Bd])throw new Error("server.handleUpgrade() was called more than once with the same socket, possibly due to a misconfiguration");if(this._state>Co)return Is(a,503);let l=["HTTP/1.1 101 Switching Protocols","Upgrade: websocket","Connection: Upgrade",`Sec-WebSocket-Accept: ${Md("sha1").update(e+Nd).digest("base64")}`],h=new this.options.WebSocket(null,void 0,this.options);if(s.size){let d=this.options.handleProtocols?this.options.handleProtocols(s,n):s.values().next().value;d&&(l.push(`Sec-WebSocket-Protocol: ${d}`),h._protocol=d)}if(t[Ft.extensionName]){let d=t[Ft.extensionName].params,u=So.format({[Ft.extensionName]:[d]});l.push(`Sec-WebSocket-Extensions: ${u}`),h._extensions=t}this.emit("headers",l,n),a.write(l.concat(`\r
1
+ "use strict";var rl=Object.create;var $s=Object.defineProperty;var il=Object.getOwnPropertyDescriptor;var ol=Object.getOwnPropertyNames;var ll=Object.getPrototypeOf,cl=Object.prototype.hasOwnProperty;var Ve=(r,t)=>()=>(t||r((t={exports:{}}).exports,t),t.exports),dl=(r,t)=>{for(var e in t)$s(r,e,{get:t[e],enumerable:!0})},ar=(r,t,e,s)=>{if(t&&typeof t=="object"||typeof t=="function")for(let n of ol(t))!cl.call(r,n)&&n!==e&&$s(r,n,{get:()=>t[n],enumerable:!(s=il(t,n))||s.enumerable});return r};var Ye=(r,t,e)=>(e=r!=null?rl(ll(r)):{},ar(t||!r||!r.__esModule?$s(e,"default",{value:r,enumerable:!0}):e,r)),hl=r=>ar($s({},"__esModule",{value:!0}),r);var gt=Ve((Sp,Pi)=>{"use strict";var Ai=["nodebuffer","arraybuffer","fragments"],Ei=typeof Blob<"u";Ei&&Ai.push("blob");Pi.exports={BINARY_TYPES:Ai,CLOSE_TIMEOUT:3e4,EMPTY_BUFFER:Buffer.alloc(0),GUID:"258EAFA5-E914-47DA-95CA-C5AB0DC85B11",hasBlob:Ei,kForOnEventAttribute:Symbol("kIsForOnEventAttribute"),kListener:Symbol("kListener"),kStatusCode:Symbol("status-code"),kWebSocket:Symbol("websocket"),NOOP:()=>{}}});var _s=Ve((Cp,xn)=>{"use strict";var{EMPTY_BUFFER:xc}=gt(),Ca=Buffer[Symbol.species];function Sc(r,t){if(r.length===0)return xc;if(r.length===1)return r[0];let e=Buffer.allocUnsafe(t),s=0;for(let n=0;n<r.length;n++){let a=r[n];e.set(a,s),s+=a.length}return s<t?new Ca(e.buffer,e.byteOffset,s):e}function Ri(r,t,e,s,n){for(let a=0;a<n;a++)e[s+a]=r[a]^t[a&3]}function Di(r,t){for(let e=0;e<r.length;e++)r[e]^=t[e&3]}function Cc(r){return r.length===r.buffer.byteLength?r.buffer:r.buffer.slice(r.byteOffset,r.byteOffset+r.length)}function Ta(r){if(Ta.readOnly=!0,Buffer.isBuffer(r))return r;let t;return r instanceof ArrayBuffer?t=new Ca(r):ArrayBuffer.isView(r)?t=new Ca(r.buffer,r.byteOffset,r.byteLength):(t=Buffer.from(r),Ta.readOnly=!1),t}xn.exports={concat:Sc,mask:Ri,toArrayBuffer:Cc,toBuffer:Ta,unmask:Di};if(!process.env.WS_NO_BUFFER_UTIL)try{let r=require("bufferutil");xn.exports.mask=function(t,e,s,n,a){a<48?Ri(t,e,s,n,a):r.mask(t,e,s,n,a)},xn.exports.unmask=function(t,e){t.length<32?Di(t,e):r.unmask(t,e)}}catch{}});var Li=Ve((Tp,Mi)=>{"use strict";var Ii=Symbol("kDone"),_a=Symbol("kRun"),Aa=class{constructor(t){this[Ii]=()=>{this.pending--,this[_a]()},this.concurrency=t||1/0,this.jobs=[],this.pending=0}add(t){this.jobs.push(t),this[_a]()}[_a](){if(this.pending!==this.concurrency&&this.jobs.length){let t=this.jobs.shift();this.pending++,t(this[Ii])}}};Mi.exports=Aa});var Qt=Ve((_p,Bi)=>{"use strict";var As=require("zlib"),Fi=_s(),Tc=Li(),{kStatusCode:Oi}=gt(),_c=Buffer[Symbol.species],Ac=Buffer.from([0,0,255,255]),Cn=Symbol("permessage-deflate"),yt=Symbol("total-length"),Jt=Symbol("callback"),St=Symbol("buffers"),Xt=Symbol("error"),Sn,Ea=class{constructor(t){if(this._options=t||{},this._threshold=this._options.threshold!==void 0?this._options.threshold:1024,this._maxPayload=this._options.maxPayload|0,this._isServer=!!this._options.isServer,this._deflate=null,this._inflate=null,this.params=null,!Sn){let e=this._options.concurrencyLimit!==void 0?this._options.concurrencyLimit:10;Sn=new Tc(e)}}static get extensionName(){return"permessage-deflate"}offer(){let t={};return this._options.serverNoContextTakeover&&(t.server_no_context_takeover=!0),this._options.clientNoContextTakeover&&(t.client_no_context_takeover=!0),this._options.serverMaxWindowBits&&(t.server_max_window_bits=this._options.serverMaxWindowBits),this._options.clientMaxWindowBits?t.client_max_window_bits=this._options.clientMaxWindowBits:this._options.clientMaxWindowBits==null&&(t.client_max_window_bits=!0),t}accept(t){return t=this.normalizeParams(t),this.params=this._isServer?this.acceptAsServer(t):this.acceptAsClient(t),this.params}cleanup(){if(this._inflate&&(this._inflate.close(),this._inflate=null),this._deflate){let t=this._deflate[Jt];this._deflate.close(),this._deflate=null,t&&t(new Error("The deflate stream was closed while data was being processed"))}}acceptAsServer(t){let e=this._options,s=t.find(n=>!(e.serverNoContextTakeover===!1&&n.server_no_context_takeover||n.server_max_window_bits&&(e.serverMaxWindowBits===!1||typeof e.serverMaxWindowBits=="number"&&e.serverMaxWindowBits>n.server_max_window_bits)||typeof e.clientMaxWindowBits=="number"&&!n.client_max_window_bits));if(!s)throw new Error("None of the extension offers can be accepted");return e.serverNoContextTakeover&&(s.server_no_context_takeover=!0),e.clientNoContextTakeover&&(s.client_no_context_takeover=!0),typeof e.serverMaxWindowBits=="number"&&(s.server_max_window_bits=e.serverMaxWindowBits),typeof e.clientMaxWindowBits=="number"?s.client_max_window_bits=e.clientMaxWindowBits:(s.client_max_window_bits===!0||e.clientMaxWindowBits===!1)&&delete s.client_max_window_bits,s}acceptAsClient(t){let e=t[0];if(this._options.clientNoContextTakeover===!1&&e.client_no_context_takeover)throw new Error('Unexpected parameter "client_no_context_takeover"');if(!e.client_max_window_bits)typeof this._options.clientMaxWindowBits=="number"&&(e.client_max_window_bits=this._options.clientMaxWindowBits);else if(this._options.clientMaxWindowBits===!1||typeof this._options.clientMaxWindowBits=="number"&&e.client_max_window_bits>this._options.clientMaxWindowBits)throw new Error('Unexpected or invalid parameter "client_max_window_bits"');return e}normalizeParams(t){return t.forEach(e=>{Object.keys(e).forEach(s=>{let n=e[s];if(n.length>1)throw new Error(`Parameter "${s}" must have only a single value`);if(n=n[0],s==="client_max_window_bits"){if(n!==!0){let a=+n;if(!Number.isInteger(a)||a<8||a>15)throw new TypeError(`Invalid value for parameter "${s}": ${n}`);n=a}else if(!this._isServer)throw new TypeError(`Invalid value for parameter "${s}": ${n}`)}else if(s==="server_max_window_bits"){let a=+n;if(!Number.isInteger(a)||a<8||a>15)throw new TypeError(`Invalid value for parameter "${s}": ${n}`);n=a}else if(s==="client_no_context_takeover"||s==="server_no_context_takeover"){if(n!==!0)throw new TypeError(`Invalid value for parameter "${s}": ${n}`)}else throw new Error(`Unknown parameter "${s}"`);e[s]=n})}),t}decompress(t,e,s){Sn.add(n=>{this._decompress(t,e,(a,i)=>{n(),s(a,i)})})}compress(t,e,s){Sn.add(n=>{this._compress(t,e,(a,i)=>{n(),s(a,i)})})}_decompress(t,e,s){let n=this._isServer?"client":"server";if(!this._inflate){let a=`${n}_max_window_bits`,i=typeof this.params[a]!="number"?As.Z_DEFAULT_WINDOWBITS:this.params[a];this._inflate=As.createInflateRaw({...this._options.zlibInflateOptions,windowBits:i}),this._inflate[Cn]=this,this._inflate[yt]=0,this._inflate[St]=[],this._inflate.on("error",Pc),this._inflate.on("data",Ni)}this._inflate[Jt]=s,this._inflate.write(t),e&&this._inflate.write(Ac),this._inflate.flush(()=>{let a=this._inflate[Xt];if(a){this._inflate.close(),this._inflate=null,s(a);return}let i=Fi.concat(this._inflate[St],this._inflate[yt]);this._inflate._readableState.endEmitted?(this._inflate.close(),this._inflate=null):(this._inflate[yt]=0,this._inflate[St]=[],e&&this.params[`${n}_no_context_takeover`]&&this._inflate.reset()),s(null,i)})}_compress(t,e,s){let n=this._isServer?"server":"client";if(!this._deflate){let a=`${n}_max_window_bits`,i=typeof this.params[a]!="number"?As.Z_DEFAULT_WINDOWBITS:this.params[a];this._deflate=As.createDeflateRaw({...this._options.zlibDeflateOptions,windowBits:i}),this._deflate[yt]=0,this._deflate[St]=[],this._deflate.on("data",Ec)}this._deflate[Jt]=s,this._deflate.write(t),this._deflate.flush(As.Z_SYNC_FLUSH,()=>{if(!this._deflate)return;let a=Fi.concat(this._deflate[St],this._deflate[yt]);e&&(a=new _c(a.buffer,a.byteOffset,a.length-4)),this._deflate[Jt]=null,this._deflate[yt]=0,this._deflate[St]=[],e&&this.params[`${n}_no_context_takeover`]&&this._deflate.reset(),s(null,a)})}};Bi.exports=Ea;function Ec(r){this[St].push(r),this[yt]+=r.length}function Ni(r){if(this[yt]+=r.length,this[Cn]._maxPayload<1||this[yt]<=this[Cn]._maxPayload){this[St].push(r);return}this[Xt]=new RangeError("Max payload size exceeded"),this[Xt].code="WS_ERR_UNSUPPORTED_MESSAGE_LENGTH",this[Xt][Oi]=1009,this.removeListener("data",Ni),this.reset()}function Pc(r){if(this[Cn]._inflate=null,this[Xt]){this[Jt](this[Xt]);return}r[Oi]=1007,this[Jt](r)}});var Zt=Ve((Ap,Tn)=>{"use strict";var{isUtf8:Ui}=require("buffer"),{hasBlob:Rc}=gt(),Dc=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1,0,1,1,1,1,1,0,0,1,1,0,1,1,0,1,1,1,1,1,1,1,1,1,1,0,0,0,0,0,0,0,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,0,0,0,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,0,1,0,1,0];function Ic(r){return r>=1e3&&r<=1014&&r!==1004&&r!==1005&&r!==1006||r>=3e3&&r<=4999}function Pa(r){let t=r.length,e=0;for(;e<t;)if((r[e]&128)===0)e++;else if((r[e]&224)===192){if(e+1===t||(r[e+1]&192)!==128||(r[e]&254)===192)return!1;e+=2}else if((r[e]&240)===224){if(e+2>=t||(r[e+1]&192)!==128||(r[e+2]&192)!==128||r[e]===224&&(r[e+1]&224)===128||r[e]===237&&(r[e+1]&224)===160)return!1;e+=3}else if((r[e]&248)===240){if(e+3>=t||(r[e+1]&192)!==128||(r[e+2]&192)!==128||(r[e+3]&192)!==128||r[e]===240&&(r[e+1]&240)===128||r[e]===244&&r[e+1]>143||r[e]>244)return!1;e+=4}else return!1;return!0}function Mc(r){return Rc&&typeof r=="object"&&typeof r.arrayBuffer=="function"&&typeof r.type=="string"&&typeof r.stream=="function"&&(r[Symbol.toStringTag]==="Blob"||r[Symbol.toStringTag]==="File")}Tn.exports={isBlob:Mc,isValidStatusCode:Ic,isValidUTF8:Pa,tokenChars:Dc};if(Ui)Tn.exports.isValidUTF8=function(r){return r.length<24?Pa(r):Ui(r)};else if(!process.env.WS_NO_UTF_8_VALIDATE)try{let r=require("utf-8-validate");Tn.exports.isValidUTF8=function(t){return t.length<32?Pa(t):r(t)}}catch{}});var La=Ve((Ep,Gi)=>{"use strict";var{Writable:Lc}=require("stream"),$i=Qt(),{BINARY_TYPES:Fc,EMPTY_BUFFER:ji,kStatusCode:Oc,kWebSocket:Nc}=gt(),{concat:Ra,toArrayBuffer:Bc,unmask:Uc}=_s(),{isValidStatusCode:$c,isValidUTF8:Wi}=Zt(),_n=Buffer[Symbol.species],et=0,Hi=1,qi=2,zi=3,Da=4,Ia=5,An=6,Ma=class extends Lc{constructor(t={}){super(),this._allowSynchronousEvents=t.allowSynchronousEvents!==void 0?t.allowSynchronousEvents:!0,this._binaryType=t.binaryType||Fc[0],this._extensions=t.extensions||{},this._isServer=!!t.isServer,this._maxBufferedChunks=t.maxBufferedChunks|0,this._maxFragments=t.maxFragments|0,this._maxPayload=t.maxPayload|0,this._skipUTF8Validation=!!t.skipUTF8Validation,this[Nc]=void 0,this._bufferedBytes=0,this._buffers=[],this._compressed=!1,this._payloadLength=0,this._mask=void 0,this._fragmented=0,this._masked=!1,this._fin=!1,this._opcode=0,this._totalPayloadLength=0,this._messageLength=0,this._fragments=[],this._errored=!1,this._loop=!1,this._state=et}_write(t,e,s){if(this._opcode===8&&this._state==et)return s();if(this._maxBufferedChunks>0&&this._buffers.length>=this._maxBufferedChunks){s(this.createError(RangeError,"Too many buffered chunks",!1,1008,"WS_ERR_TOO_MANY_BUFFERED_PARTS"));return}this._bufferedBytes+=t.length,this._buffers.push(t),this.startLoop(s)}consume(t){if(this._bufferedBytes-=t,t===this._buffers[0].length)return this._buffers.shift();if(t<this._buffers[0].length){let s=this._buffers[0];return this._buffers[0]=new _n(s.buffer,s.byteOffset+t,s.length-t),new _n(s.buffer,s.byteOffset,t)}let e=Buffer.allocUnsafe(t);do{let s=this._buffers[0],n=e.length-t;t>=s.length?e.set(this._buffers.shift(),n):(e.set(new Uint8Array(s.buffer,s.byteOffset,t),n),this._buffers[0]=new _n(s.buffer,s.byteOffset+t,s.length-t)),t-=s.length}while(t>0);return e}startLoop(t){this._loop=!0;do switch(this._state){case et:this.getInfo(t);break;case Hi:this.getPayloadLength16(t);break;case qi:this.getPayloadLength64(t);break;case zi:this.getMask();break;case Da:this.getData(t);break;case Ia:case An:this._loop=!1;return}while(this._loop);this._errored||t()}getInfo(t){if(this._bufferedBytes<2){this._loop=!1;return}let e=this.consume(2);if((e[0]&48)!==0){let n=this.createError(RangeError,"RSV2 and RSV3 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_2_3");t(n);return}let s=(e[0]&64)===64;if(s&&!this._extensions[$i.extensionName]){let n=this.createError(RangeError,"RSV1 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_1");t(n);return}if(this._fin=(e[0]&128)===128,this._opcode=e[0]&15,this._payloadLength=e[1]&127,this._opcode===0){if(s){let n=this.createError(RangeError,"RSV1 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_1");t(n);return}if(!this._fragmented){let n=this.createError(RangeError,"invalid opcode 0",!0,1002,"WS_ERR_INVALID_OPCODE");t(n);return}this._opcode=this._fragmented}else if(this._opcode===1||this._opcode===2){if(this._fragmented){let n=this.createError(RangeError,`invalid opcode ${this._opcode}`,!0,1002,"WS_ERR_INVALID_OPCODE");t(n);return}this._compressed=s}else if(this._opcode>7&&this._opcode<11){if(!this._fin){let n=this.createError(RangeError,"FIN must be set",!0,1002,"WS_ERR_EXPECTED_FIN");t(n);return}if(s){let n=this.createError(RangeError,"RSV1 must be clear",!0,1002,"WS_ERR_UNEXPECTED_RSV_1");t(n);return}if(this._payloadLength>125||this._opcode===8&&this._payloadLength===1){let n=this.createError(RangeError,`invalid payload length ${this._payloadLength}`,!0,1002,"WS_ERR_INVALID_CONTROL_PAYLOAD_LENGTH");t(n);return}}else{let n=this.createError(RangeError,`invalid opcode ${this._opcode}`,!0,1002,"WS_ERR_INVALID_OPCODE");t(n);return}if(!this._fin&&!this._fragmented&&(this._fragmented=this._opcode),this._masked=(e[1]&128)===128,this._isServer){if(!this._masked){let n=this.createError(RangeError,"MASK must be set",!0,1002,"WS_ERR_EXPECTED_MASK");t(n);return}}else if(this._masked){let n=this.createError(RangeError,"MASK must be clear",!0,1002,"WS_ERR_UNEXPECTED_MASK");t(n);return}this._payloadLength===126?this._state=Hi:this._payloadLength===127?this._state=qi:this.haveLength(t)}getPayloadLength16(t){if(this._bufferedBytes<2){this._loop=!1;return}this._payloadLength=this.consume(2).readUInt16BE(0),this.haveLength(t)}getPayloadLength64(t){if(this._bufferedBytes<8){this._loop=!1;return}let e=this.consume(8),s=e.readUInt32BE(0);if(s>Math.pow(2,21)-1){let n=this.createError(RangeError,"Unsupported WebSocket frame: payload length > 2^53 - 1",!1,1009,"WS_ERR_UNSUPPORTED_DATA_PAYLOAD_LENGTH");t(n);return}this._payloadLength=s*Math.pow(2,32)+e.readUInt32BE(4),this.haveLength(t)}haveLength(t){if(this._payloadLength&&this._opcode<8&&(this._totalPayloadLength+=this._payloadLength,this._totalPayloadLength>this._maxPayload&&this._maxPayload>0)){let e=this.createError(RangeError,"Max payload size exceeded",!1,1009,"WS_ERR_UNSUPPORTED_MESSAGE_LENGTH");t(e);return}this._masked?this._state=zi:this._state=Da}getMask(){if(this._bufferedBytes<4){this._loop=!1;return}this._mask=this.consume(4),this._state=Da}getData(t){let e=ji;if(this._payloadLength){if(this._bufferedBytes<this._payloadLength){this._loop=!1;return}e=this.consume(this._payloadLength),this._masked&&(this._mask[0]|this._mask[1]|this._mask[2]|this._mask[3])!==0&&Uc(e,this._mask)}if(this._opcode>7){this.controlMessage(e,t);return}if(this._compressed){this._state=Ia,this.decompress(e,t);return}if(e.length){if(this._maxFragments>0&&this._fragments.length>=this._maxFragments){let s=this.createError(RangeError,"Too many message fragments",!1,1008,"WS_ERR_TOO_MANY_BUFFERED_PARTS");t(s);return}this._messageLength=this._totalPayloadLength,this._fragments.push(e)}this.dataMessage(t)}decompress(t,e){this._extensions[$i.extensionName].decompress(t,this._fin,(n,a)=>{if(n)return e(n);if(a.length){if(this._messageLength+=a.length,this._messageLength>this._maxPayload&&this._maxPayload>0){let i=this.createError(RangeError,"Max payload size exceeded",!1,1009,"WS_ERR_UNSUPPORTED_MESSAGE_LENGTH");e(i);return}if(this._maxFragments>0&&this._fragments.length>=this._maxFragments){let i=this.createError(RangeError,"Too many message fragments",!1,1008,"WS_ERR_TOO_MANY_BUFFERED_PARTS");e(i);return}this._fragments.push(a)}this.dataMessage(e),this._state===et&&this.startLoop(e)})}dataMessage(t){if(!this._fin){this._state=et;return}let e=this._messageLength,s=this._fragments;if(this._totalPayloadLength=0,this._messageLength=0,this._fragmented=0,this._fragments=[],this._opcode===2){let n;this._binaryType==="nodebuffer"?n=Ra(s,e):this._binaryType==="arraybuffer"?n=Bc(Ra(s,e)):this._binaryType==="blob"?n=new Blob(s):n=s,this._allowSynchronousEvents?(this.emit("message",n,!0),this._state=et):(this._state=An,setImmediate(()=>{this.emit("message",n,!0),this._state=et,this.startLoop(t)}))}else{let n=Ra(s,e);if(!this._skipUTF8Validation&&!Wi(n)){let a=this.createError(Error,"invalid UTF-8 sequence",!0,1007,"WS_ERR_INVALID_UTF8");t(a);return}this._state===Ia||this._allowSynchronousEvents?(this.emit("message",n,!1),this._state=et):(this._state=An,setImmediate(()=>{this.emit("message",n,!1),this._state=et,this.startLoop(t)}))}}controlMessage(t,e){if(this._opcode===8){if(t.length===0)this._loop=!1,this.emit("conclude",1005,ji),this.end();else{let s=t.readUInt16BE(0);if(!$c(s)){let a=this.createError(RangeError,`invalid status code ${s}`,!0,1002,"WS_ERR_INVALID_CLOSE_CODE");e(a);return}let n=new _n(t.buffer,t.byteOffset+2,t.length-2);if(!this._skipUTF8Validation&&!Wi(n)){let a=this.createError(Error,"invalid UTF-8 sequence",!0,1007,"WS_ERR_INVALID_UTF8");e(a);return}this._loop=!1,this.emit("conclude",s,n),this.end()}this._state=et;return}this._allowSynchronousEvents?(this.emit(this._opcode===9?"ping":"pong",t),this._state=et):(this._state=An,setImmediate(()=>{this.emit(this._opcode===9?"ping":"pong",t),this._state=et,this.startLoop(e)}))}createError(t,e,s,n,a){this._loop=!1,this._errored=!0;let i=new t(s?`Invalid WebSocket frame: ${e}`:e);return Error.captureStackTrace(i,this.createError),i.code=a,i[Oc]=n,i}};Gi.exports=Ma});var Na=Ve((Rp,Ki)=>{"use strict";var{Duplex:Pp}=require("stream"),{randomFillSync:jc}=require("crypto"),{types:{isUint8Array:Wc}}=require("util"),Vi=Qt(),{EMPTY_BUFFER:Hc,kWebSocket:qc,NOOP:zc}=gt(),{isBlob:es,isValidStatusCode:Gc}=Zt(),{mask:Yi,toBuffer:Mt}=_s(),tt=Symbol("kByteLength"),Vc=Buffer.alloc(4),En=8*1024,Lt,ts=En,lt=0,Yc=1,Kc=2,Fa=class r{constructor(t,e,s){this._extensions=e||{},s&&(this._generateMask=s,this._maskBuffer=Buffer.alloc(4)),this._socket=t,this._firstFragment=!0,this._compress=!1,this._bufferedBytes=0,this._queue=[],this._state=lt,this.onerror=zc,this[qc]=void 0}static frame(t,e){let s,n=!1,a=2,i=!1;e.mask&&(s=e.maskBuffer||Vc,e.generateMask?e.generateMask(s):(ts===En&&(Lt===void 0&&(Lt=Buffer.alloc(En)),jc(Lt,0,En),ts=0),s[0]=Lt[ts++],s[1]=Lt[ts++],s[2]=Lt[ts++],s[3]=Lt[ts++]),i=(s[0]|s[1]|s[2]|s[3])===0,a=6);let o;typeof t=="string"?(!e.mask||i)&&e[tt]!==void 0?o=e[tt]:(t=Buffer.from(t),o=t.length):(o=t.length,n=e.mask&&e.readOnly&&!i);let c=o;o>=65536?(a+=8,c=127):o>125&&(a+=2,c=126);let l=Buffer.allocUnsafe(n?o+a:a);return l[0]=e.fin?e.opcode|128:e.opcode,e.rsv1&&(l[0]|=64),l[1]=c,c===126?l.writeUInt16BE(o,2):c===127&&(l[2]=l[3]=0,l.writeUIntBE(o,4,6)),e.mask?(l[1]|=128,l[a-4]=s[0],l[a-3]=s[1],l[a-2]=s[2],l[a-1]=s[3],i?[l,t]:n?(Yi(t,s,l,a,o),[l]):(Yi(t,s,t,0,o),[l,t])):[l,t]}close(t,e,s,n){let a;if(t===void 0)a=Hc;else{if(typeof t!="number"||!Gc(t))throw new TypeError("First argument must be a valid error code number");if(e===void 0||!e.length)a=Buffer.allocUnsafe(2),a.writeUInt16BE(t,0);else{let o=Buffer.byteLength(e);if(o>123)throw new RangeError("The message must not be greater than 123 bytes");if(a=Buffer.allocUnsafe(2+o),a.writeUInt16BE(t,0),typeof e=="string")a.write(e,2);else if(Wc(e))a.set(e,2);else throw new TypeError("Second argument must be a string or a Uint8Array")}}let i={[tt]:a.length,fin:!0,generateMask:this._generateMask,mask:s,maskBuffer:this._maskBuffer,opcode:8,readOnly:!1,rsv1:!1};this._state!==lt?this.enqueue([this.dispatch,a,!1,i,n]):this.sendFrame(r.frame(a,i),n)}ping(t,e,s){let n,a;if(typeof t=="string"?(n=Buffer.byteLength(t),a=!1):es(t)?(n=t.size,a=!1):(t=Mt(t),n=t.length,a=Mt.readOnly),n>125)throw new RangeError("The data size must not be greater than 125 bytes");let i={[tt]:n,fin:!0,generateMask:this._generateMask,mask:e,maskBuffer:this._maskBuffer,opcode:9,readOnly:a,rsv1:!1};es(t)?this._state!==lt?this.enqueue([this.getBlobData,t,!1,i,s]):this.getBlobData(t,!1,i,s):this._state!==lt?this.enqueue([this.dispatch,t,!1,i,s]):this.sendFrame(r.frame(t,i),s)}pong(t,e,s){let n,a;if(typeof t=="string"?(n=Buffer.byteLength(t),a=!1):es(t)?(n=t.size,a=!1):(t=Mt(t),n=t.length,a=Mt.readOnly),n>125)throw new RangeError("The data size must not be greater than 125 bytes");let i={[tt]:n,fin:!0,generateMask:this._generateMask,mask:e,maskBuffer:this._maskBuffer,opcode:10,readOnly:a,rsv1:!1};es(t)?this._state!==lt?this.enqueue([this.getBlobData,t,!1,i,s]):this.getBlobData(t,!1,i,s):this._state!==lt?this.enqueue([this.dispatch,t,!1,i,s]):this.sendFrame(r.frame(t,i),s)}send(t,e,s){let n=this._extensions[Vi.extensionName],a=e.binary?2:1,i=e.compress,o,c;typeof t=="string"?(o=Buffer.byteLength(t),c=!1):es(t)?(o=t.size,c=!1):(t=Mt(t),o=t.length,c=Mt.readOnly),this._firstFragment?(this._firstFragment=!1,i&&n&&n.params[n._isServer?"server_no_context_takeover":"client_no_context_takeover"]&&(i=o>=n._threshold),this._compress=i):(i=!1,a=0),e.fin&&(this._firstFragment=!0);let l={[tt]:o,fin:e.fin,generateMask:this._generateMask,mask:e.mask,maskBuffer:this._maskBuffer,opcode:a,readOnly:c,rsv1:i};es(t)?this._state!==lt?this.enqueue([this.getBlobData,t,this._compress,l,s]):this.getBlobData(t,this._compress,l,s):this._state!==lt?this.enqueue([this.dispatch,t,this._compress,l,s]):this.dispatch(t,this._compress,l,s)}getBlobData(t,e,s,n){this._bufferedBytes+=s[tt],this._state=Kc,t.arrayBuffer().then(a=>{if(this._socket.destroyed){let o=new Error("The socket was closed while the blob was being read");process.nextTick(Oa,this,o,n);return}this._bufferedBytes-=s[tt];let i=Mt(a);e?this.dispatch(i,e,s,n):(this._state=lt,this.sendFrame(r.frame(i,s),n),this.dequeue())}).catch(a=>{process.nextTick(Jc,this,a,n)})}dispatch(t,e,s,n){if(!e){this.sendFrame(r.frame(t,s),n);return}let a=this._extensions[Vi.extensionName];this._bufferedBytes+=s[tt],this._state=Yc,a.compress(t,s.fin,(i,o)=>{if(this._socket.destroyed){let c=new Error("The socket was closed while data was being compressed");Oa(this,c,n);return}this._bufferedBytes-=s[tt],this._state=lt,s.readOnly=!1,this.sendFrame(r.frame(o,s),n),this.dequeue()})}dequeue(){for(;this._state===lt&&this._queue.length;){let t=this._queue.shift();this._bufferedBytes-=t[3][tt],Reflect.apply(t[0],this,t.slice(1))}}enqueue(t){this._bufferedBytes+=t[3][tt],this._queue.push(t)}sendFrame(t,e){t.length===2?(this._socket.cork(),this._socket.write(t[0]),this._socket.write(t[1],e),this._socket.uncork()):this._socket.write(t[0],e)}};Ki.exports=Fa;function Oa(r,t,e){typeof e=="function"&&e(t);for(let s=0;s<r._queue.length;s++){let n=r._queue[s],a=n[n.length-1];typeof a=="function"&&a(t)}}function Jc(r,t,e){Oa(r,t,e),r.onerror(t)}});var ao=Ve((Dp,no)=>{"use strict";var{kForOnEventAttribute:Es,kListener:Ba}=gt(),Ji=Symbol("kCode"),Xi=Symbol("kData"),Qi=Symbol("kError"),Zi=Symbol("kMessage"),eo=Symbol("kReason"),ss=Symbol("kTarget"),to=Symbol("kType"),so=Symbol("kWasClean"),vt=class{constructor(t){this[ss]=null,this[to]=t}get target(){return this[ss]}get type(){return this[to]}};Object.defineProperty(vt.prototype,"target",{enumerable:!0});Object.defineProperty(vt.prototype,"type",{enumerable:!0});var Ft=class extends vt{constructor(t,e={}){super(t),this[Ji]=e.code===void 0?0:e.code,this[eo]=e.reason===void 0?"":e.reason,this[so]=e.wasClean===void 0?!1:e.wasClean}get code(){return this[Ji]}get reason(){return this[eo]}get wasClean(){return this[so]}};Object.defineProperty(Ft.prototype,"code",{enumerable:!0});Object.defineProperty(Ft.prototype,"reason",{enumerable:!0});Object.defineProperty(Ft.prototype,"wasClean",{enumerable:!0});var ns=class extends vt{constructor(t,e={}){super(t),this[Qi]=e.error===void 0?null:e.error,this[Zi]=e.message===void 0?"":e.message}get error(){return this[Qi]}get message(){return this[Zi]}};Object.defineProperty(ns.prototype,"error",{enumerable:!0});Object.defineProperty(ns.prototype,"message",{enumerable:!0});var Ps=class extends vt{constructor(t,e={}){super(t),this[Xi]=e.data===void 0?null:e.data}get data(){return this[Xi]}};Object.defineProperty(Ps.prototype,"data",{enumerable:!0});var Xc={addEventListener(r,t,e={}){for(let n of this.listeners(r))if(!e[Es]&&n[Ba]===t&&!n[Es])return;let s;if(r==="message")s=function(a,i){let o=new Ps("message",{data:i?a:a.toString()});o[ss]=this,Pn(t,this,o)};else if(r==="close")s=function(a,i){let o=new Ft("close",{code:a,reason:i.toString(),wasClean:this._closeFrameReceived&&this._closeFrameSent});o[ss]=this,Pn(t,this,o)};else if(r==="error")s=function(a){let i=new ns("error",{error:a,message:a.message});i[ss]=this,Pn(t,this,i)};else if(r==="open")s=function(){let a=new vt("open");a[ss]=this,Pn(t,this,a)};else return;s[Es]=!!e[Es],s[Ba]=t,e.once?this.once(r,s):this.on(r,s)},removeEventListener(r,t){for(let e of this.listeners(r))if(e[Ba]===t&&!e[Es]){this.removeListener(r,e);break}}};no.exports={CloseEvent:Ft,ErrorEvent:ns,Event:vt,EventTarget:Xc,MessageEvent:Ps};function Pn(r,t,e){typeof r=="object"&&r.handleEvent?r.handleEvent.call(r,e):r.call(t,e)}});var Rn=Ve((Ip,ro)=>{"use strict";var{tokenChars:Rs}=Zt();function pt(r,t,e){r[t]===void 0?r[t]=[e]:r[t].push(e)}function Qc(r){let t=Object.create(null),e=Object.create(null),s=!1,n=!1,a=!1,i,o,c=-1,l=-1,h=-1,d=0;for(;d<r.length;d++)if(l=r.charCodeAt(d),i===void 0)if(h===-1&&Rs[l]===1)c===-1&&(c=d);else if(d!==0&&(l===32||l===9))h===-1&&c!==-1&&(h=d);else if(l===59||l===44){if(c===-1)throw new SyntaxError(`Unexpected character at index ${d}`);h===-1&&(h=d);let p=r.slice(c,h);l===44?(pt(t,p,e),e=Object.create(null)):i=p,c=h=-1}else throw new SyntaxError(`Unexpected character at index ${d}`);else if(o===void 0)if(h===-1&&Rs[l]===1)c===-1&&(c=d);else if(l===32||l===9)h===-1&&c!==-1&&(h=d);else if(l===59||l===44){if(c===-1)throw new SyntaxError(`Unexpected character at index ${d}`);h===-1&&(h=d),pt(e,r.slice(c,h),!0),l===44&&(pt(t,i,e),e=Object.create(null),i=void 0),c=h=-1}else if(l===61&&c!==-1&&h===-1)o=r.slice(c,d),c=h=-1;else throw new SyntaxError(`Unexpected character at index ${d}`);else if(n){if(Rs[l]!==1)throw new SyntaxError(`Unexpected character at index ${d}`);c===-1?c=d:s||(s=!0),n=!1}else if(a)if(Rs[l]===1)c===-1&&(c=d);else if(l===34&&c!==-1)a=!1,h=d;else if(l===92)n=!0;else throw new SyntaxError(`Unexpected character at index ${d}`);else if(l===34&&r.charCodeAt(d-1)===61)a=!0;else if(h===-1&&Rs[l]===1)c===-1&&(c=d);else if(c!==-1&&(l===32||l===9))h===-1&&(h=d);else if(l===59||l===44){if(c===-1)throw new SyntaxError(`Unexpected character at index ${d}`);h===-1&&(h=d);let p=r.slice(c,h);s&&(p=p.replace(/\\/g,""),s=!1),pt(e,o,p),l===44&&(pt(t,i,e),e=Object.create(null),i=void 0),o=void 0,c=h=-1}else throw new SyntaxError(`Unexpected character at index ${d}`);if(c===-1||a||l===32||l===9)throw new SyntaxError("Unexpected end of input");h===-1&&(h=d);let u=r.slice(c,h);return i===void 0?pt(t,u,e):(o===void 0?pt(e,u,!0):s?pt(e,o,u.replace(/\\/g,"")):pt(e,o,u),pt(t,i,e)),t}function Zc(r){return Object.keys(r).map(t=>{let e=r[t];return Array.isArray(e)||(e=[e]),e.map(s=>[t].concat(Object.keys(s).map(n=>{let a=s[n];return Array.isArray(a)||(a=[a]),a.map(i=>i===!0?n:`${n}=${i}`).join("; ")})).join("; ")).join(", ")}).join(", ")}ro.exports={format:Zc,parse:Qc}});var Ln=Ve((Fp,vo)=>{"use strict";var ed=require("events"),td=require("https"),sd=require("http"),lo=require("net"),nd=require("tls"),{randomBytes:ad,createHash:rd}=require("crypto"),{Duplex:Mp,Readable:Lp}=require("stream"),{URL:Ua}=require("url"),Ct=Qt(),id=La(),od=Na(),{isBlob:ld}=Zt(),{BINARY_TYPES:io,CLOSE_TIMEOUT:cd,EMPTY_BUFFER:Dn,GUID:dd,kForOnEventAttribute:$a,kListener:hd,kStatusCode:ud,kWebSocket:De,NOOP:co}=gt(),{EventTarget:{addEventListener:pd,removeEventListener:md}}=ao(),{format:fd,parse:gd}=Rn(),{toBuffer:yd}=_s(),ho=Symbol("kAborted"),ja=[8,13],wt=["CONNECTING","OPEN","CLOSING","CLOSED"],vd=/^[!#$%&'*+\-.0-9A-Z^_`|a-z~]+$/,be=class r extends ed{constructor(t,e,s){super(),this._binaryType=io[0],this._closeCode=1006,this._closeFrameReceived=!1,this._closeFrameSent=!1,this._closeMessage=Dn,this._closeTimer=null,this._errorEmitted=!1,this._extensions={},this._paused=!1,this._protocol="",this._readyState=r.CONNECTING,this._receiver=null,this._sender=null,this._socket=null,t!==null?(this._bufferedAmount=0,this._isServer=!1,this._redirects=0,e===void 0?e=[]:Array.isArray(e)||(typeof e=="object"&&e!==null?(s=e,e=[]):e=[e]),uo(this,t,e,s)):(this._autoPong=s.autoPong,this._closeTimeout=s.closeTimeout,this._isServer=!0)}get binaryType(){return this._binaryType}set binaryType(t){io.includes(t)&&(this._binaryType=t,this._receiver&&(this._receiver._binaryType=t))}get bufferedAmount(){return this._socket?this._socket._writableState.length+this._sender._bufferedBytes:this._bufferedAmount}get extensions(){return Object.keys(this._extensions).join()}get isPaused(){return this._paused}get onclose(){return null}get onerror(){return null}get onopen(){return null}get onmessage(){return null}get protocol(){return this._protocol}get readyState(){return this._readyState}get url(){return this._url}setSocket(t,e,s){let n=new id({allowSynchronousEvents:s.allowSynchronousEvents,binaryType:this.binaryType,extensions:this._extensions,isServer:this._isServer,maxBufferedChunks:s.maxBufferedChunks,maxFragments:s.maxFragments,maxPayload:s.maxPayload,skipUTF8Validation:s.skipUTF8Validation}),a=new od(t,this._extensions,s.generateMask);this._receiver=n,this._sender=a,this._socket=t,n[De]=this,a[De]=this,t[De]=this,n.on("conclude",kd),n.on("drain",xd),n.on("error",Sd),n.on("message",Cd),n.on("ping",Td),n.on("pong",_d),a.onerror=Ad,t.setTimeout&&t.setTimeout(0),t.setNoDelay&&t.setNoDelay(),e.length>0&&t.unshift(e),t.on("close",fo),t.on("data",Mn),t.on("end",go),t.on("error",yo),this._readyState=r.OPEN,this.emit("open")}emitClose(){if(!this._socket){this._readyState=r.CLOSED,this.emit("close",this._closeCode,this._closeMessage);return}this._extensions[Ct.extensionName]&&this._extensions[Ct.extensionName].cleanup(),this._receiver.removeAllListeners(),this._readyState=r.CLOSED,this.emit("close",this._closeCode,this._closeMessage)}close(t,e){if(this.readyState!==r.CLOSED){if(this.readyState===r.CONNECTING){Qe(this,this._req,"WebSocket was closed before the connection was established");return}if(this.readyState===r.CLOSING){this._closeFrameSent&&(this._closeFrameReceived||this._receiver._writableState.errorEmitted)&&this._socket.end();return}this._readyState=r.CLOSING,this._sender.close(t,e,!this._isServer,s=>{s||(this._closeFrameSent=!0,(this._closeFrameReceived||this._receiver._writableState.errorEmitted)&&this._socket.end())}),mo(this)}}pause(){this.readyState===r.CONNECTING||this.readyState===r.CLOSED||(this._paused=!0,this._socket.pause())}ping(t,e,s){if(this.readyState===r.CONNECTING)throw new Error("WebSocket is not open: readyState 0 (CONNECTING)");if(typeof t=="function"?(s=t,t=e=void 0):typeof e=="function"&&(s=e,e=void 0),typeof t=="number"&&(t=t.toString()),this.readyState!==r.OPEN){Wa(this,t,s);return}e===void 0&&(e=!this._isServer),this._sender.ping(t||Dn,e,s)}pong(t,e,s){if(this.readyState===r.CONNECTING)throw new Error("WebSocket is not open: readyState 0 (CONNECTING)");if(typeof t=="function"?(s=t,t=e=void 0):typeof e=="function"&&(s=e,e=void 0),typeof t=="number"&&(t=t.toString()),this.readyState!==r.OPEN){Wa(this,t,s);return}e===void 0&&(e=!this._isServer),this._sender.pong(t||Dn,e,s)}resume(){this.readyState===r.CONNECTING||this.readyState===r.CLOSED||(this._paused=!1,this._receiver._writableState.needDrain||this._socket.resume())}send(t,e,s){if(this.readyState===r.CONNECTING)throw new Error("WebSocket is not open: readyState 0 (CONNECTING)");if(typeof e=="function"&&(s=e,e={}),typeof t=="number"&&(t=t.toString()),this.readyState!==r.OPEN){Wa(this,t,s);return}let n={binary:typeof t!="string",mask:!this._isServer,compress:!0,fin:!0,...e};this._extensions[Ct.extensionName]||(n.compress=!1),this._sender.send(t||Dn,n,s)}terminate(){if(this.readyState!==r.CLOSED){if(this.readyState===r.CONNECTING){Qe(this,this._req,"WebSocket was closed before the connection was established");return}this._socket&&(this._readyState=r.CLOSING,this._socket.destroy())}}};Object.defineProperty(be,"CONNECTING",{enumerable:!0,value:wt.indexOf("CONNECTING")});Object.defineProperty(be.prototype,"CONNECTING",{enumerable:!0,value:wt.indexOf("CONNECTING")});Object.defineProperty(be,"OPEN",{enumerable:!0,value:wt.indexOf("OPEN")});Object.defineProperty(be.prototype,"OPEN",{enumerable:!0,value:wt.indexOf("OPEN")});Object.defineProperty(be,"CLOSING",{enumerable:!0,value:wt.indexOf("CLOSING")});Object.defineProperty(be.prototype,"CLOSING",{enumerable:!0,value:wt.indexOf("CLOSING")});Object.defineProperty(be,"CLOSED",{enumerable:!0,value:wt.indexOf("CLOSED")});Object.defineProperty(be.prototype,"CLOSED",{enumerable:!0,value:wt.indexOf("CLOSED")});["binaryType","bufferedAmount","extensions","isPaused","protocol","readyState","url"].forEach(r=>{Object.defineProperty(be.prototype,r,{enumerable:!0})});["open","error","close","message"].forEach(r=>{Object.defineProperty(be.prototype,`on${r}`,{enumerable:!0,get(){for(let t of this.listeners(r))if(t[$a])return t[hd];return null},set(t){for(let e of this.listeners(r))if(e[$a]){this.removeListener(r,e);break}typeof t=="function"&&this.addEventListener(r,t,{[$a]:!0})}})});be.prototype.addEventListener=pd;be.prototype.removeEventListener=md;vo.exports=be;function uo(r,t,e,s){let n={allowSynchronousEvents:!0,autoPong:!0,closeTimeout:cd,protocolVersion:ja[1],maxBufferedChunks:1048576,maxFragments:131072,maxPayload:104857600,skipUTF8Validation:!1,perMessageDeflate:!0,followRedirects:!1,maxRedirects:10,...s,socketPath:void 0,hostname:void 0,protocol:void 0,timeout:void 0,method:"GET",host:void 0,path:void 0,port:void 0};if(r._autoPong=n.autoPong,r._closeTimeout=n.closeTimeout,!ja.includes(n.protocolVersion))throw new RangeError(`Unsupported protocol version: ${n.protocolVersion} (supported versions: ${ja.join(", ")})`);let a;if(t instanceof Ua)a=t;else try{a=new Ua(t)}catch{throw new SyntaxError(`Invalid URL: ${t}`)}a.protocol==="http:"?a.protocol="ws:":a.protocol==="https:"&&(a.protocol="wss:"),r._url=a.href;let i=a.protocol==="wss:",o=a.protocol==="ws+unix:",c;if(a.protocol!=="ws:"&&!i&&!o?c=`The URL's protocol must be one of "ws:", "wss:", "http:", "https:", or "ws+unix:"`:o&&!a.pathname?c="The URL's pathname is empty":a.hash&&(c="The URL contains a fragment identifier"),c){let f=new SyntaxError(c);if(r._redirects===0)throw f;In(r,f);return}let l=i?443:80,h=ad(16).toString("base64"),d=i?td.request:sd.request,u=new Set,p;if(n.createConnection=n.createConnection||(i?bd:wd),n.defaultPort=n.defaultPort||l,n.port=a.port||l,n.host=a.hostname.startsWith("[")?a.hostname.slice(1,-1):a.hostname,n.headers={...n.headers,"Sec-WebSocket-Version":n.protocolVersion,"Sec-WebSocket-Key":h,Connection:"Upgrade",Upgrade:"websocket"},n.path=a.pathname+a.search,n.timeout=n.handshakeTimeout,n.perMessageDeflate&&(p=new Ct({...n.perMessageDeflate,isServer:!1,maxPayload:n.maxPayload}),n.headers["Sec-WebSocket-Extensions"]=fd({[Ct.extensionName]:p.offer()})),e.length){for(let f of e){if(typeof f!="string"||!vd.test(f)||u.has(f))throw new SyntaxError("An invalid or duplicated subprotocol was specified");u.add(f)}n.headers["Sec-WebSocket-Protocol"]=e.join(",")}if(n.origin&&(n.protocolVersion<13?n.headers["Sec-WebSocket-Origin"]=n.origin:n.headers.Origin=n.origin),(a.username||a.password)&&(n.auth=`${a.username}:${a.password}`),o){let f=n.path.split(":");n.socketPath=f[0],n.path=f[1]}let m;if(n.followRedirects){if(r._redirects===0){r._originalIpc=o,r._originalSecure=i,r._originalHostOrSocketPath=o?n.socketPath:a.host;let f=s&&s.headers;if(s={...s,headers:{}},f)for(let[g,y]of Object.entries(f))s.headers[g.toLowerCase()]=y}else if(r.listenerCount("redirect")===0){let f=o?r._originalIpc?n.socketPath===r._originalHostOrSocketPath:!1:r._originalIpc?!1:a.host===r._originalHostOrSocketPath;(!f||r._originalSecure&&!i)&&(delete n.headers.authorization,delete n.headers.cookie,f||delete n.headers.host,n.auth=void 0)}n.auth&&!s.headers.authorization&&(s.headers.authorization="Basic "+Buffer.from(n.auth).toString("base64")),m=r._req=d(n),r._redirects&&r.emit("redirect",r.url,m)}else m=r._req=d(n);n.timeout&&m.on("timeout",()=>{Qe(r,m,"Opening handshake has timed out")}),m.on("error",f=>{m===null||m[ho]||(m=r._req=null,In(r,f))}),m.on("response",f=>{let g=f.headers.location,y=f.statusCode;if(g&&n.followRedirects&&y>=300&&y<400){if(++r._redirects>n.maxRedirects){Qe(r,m,"Maximum redirects exceeded");return}m.abort();let v;try{v=new Ua(g,t)}catch{let w=new SyntaxError(`Invalid URL: ${g}`);In(r,w);return}uo(r,v,e,s)}else r.emit("unexpected-response",m,f)||Qe(r,m,`Unexpected server response: ${f.statusCode}`)}),m.on("upgrade",(f,g,y)=>{if(r.emit("upgrade",f),r.readyState!==be.CONNECTING)return;m=r._req=null;let v=f.headers.upgrade;if(v===void 0||v.toLowerCase()!=="websocket"){Qe(r,g,"Invalid Upgrade header");return}let k=rd("sha1").update(h+dd).digest("base64");if(f.headers["sec-websocket-accept"]!==k){Qe(r,g,"Invalid Sec-WebSocket-Accept header");return}let w=f.headers["sec-websocket-protocol"],S;if(w!==void 0?u.size?u.has(w)||(S="Server sent an invalid subprotocol"):S="Server sent a subprotocol but none was requested":u.size&&(S="Server sent no subprotocol"),S){Qe(r,g,S);return}w&&(r._protocol=w);let T=f.headers["sec-websocket-extensions"];if(T!==void 0){if(!p){Qe(r,g,"Server sent a Sec-WebSocket-Extensions header but no extension was requested");return}let _;try{_=gd(T)}catch{Qe(r,g,"Invalid Sec-WebSocket-Extensions header");return}let D=Object.keys(_);if(D.length!==1||D[0]!==Ct.extensionName){Qe(r,g,"Server indicated an extension that was not requested");return}try{p.accept(_[Ct.extensionName])}catch{Qe(r,g,"Invalid Sec-WebSocket-Extensions header");return}r._extensions[Ct.extensionName]=p}r.setSocket(g,y,{allowSynchronousEvents:n.allowSynchronousEvents,generateMask:n.generateMask,maxBufferedChunks:n.maxBufferedChunks,maxFragments:n.maxFragments,maxPayload:n.maxPayload,skipUTF8Validation:n.skipUTF8Validation})}),n.finishRequest?n.finishRequest(m,r):m.end()}function In(r,t){r._readyState=be.CLOSING,r._errorEmitted=!0,r.emit("error",t),r.emitClose()}function wd(r){return r.path=r.socketPath,lo.connect(r)}function bd(r){return r.path=void 0,!r.servername&&r.servername!==""&&(r.servername=lo.isIP(r.host)?"":r.host),nd.connect(r)}function Qe(r,t,e){r._readyState=be.CLOSING;let s=new Error(e);Error.captureStackTrace(s,Qe),t.setHeader?(t[ho]=!0,t.abort(),t.socket&&!t.socket.destroyed&&t.socket.destroy(),process.nextTick(In,r,s)):(t.destroy(s),t.once("error",r.emit.bind(r,"error")),t.once("close",r.emitClose.bind(r)))}function Wa(r,t,e){if(t){let s=ld(t)?t.size:yd(t).length;r._socket?r._sender._bufferedBytes+=s:r._bufferedAmount+=s}if(e){let s=new Error(`WebSocket is not open: readyState ${r.readyState} (${wt[r.readyState]})`);process.nextTick(e,s)}}function kd(r,t){let e=this[De];e._closeFrameReceived=!0,e._closeMessage=t,e._closeCode=r,e._socket[De]!==void 0&&(e._socket.removeListener("data",Mn),process.nextTick(po,e._socket),r===1005?e.close():e.close(r,t))}function xd(){let r=this[De];r.isPaused||r._socket.resume()}function Sd(r){let t=this[De];t._socket[De]!==void 0&&(t._socket.removeListener("data",Mn),process.nextTick(po,t._socket),t.close(r[ud])),t._errorEmitted||(t._errorEmitted=!0,t.emit("error",r))}function oo(){this[De].emitClose()}function Cd(r,t){this[De].emit("message",r,t)}function Td(r){let t=this[De];t._autoPong&&t.pong(r,!this._isServer,co),t.emit("ping",r)}function _d(r){this[De].emit("pong",r)}function po(r){r.resume()}function Ad(r){let t=this[De];t.readyState!==be.CLOSED&&(t.readyState===be.OPEN&&(t._readyState=be.CLOSING,mo(t)),this._socket.end(),t._errorEmitted||(t._errorEmitted=!0,t.emit("error",r)))}function mo(r){r._closeTimer=setTimeout(r._socket.destroy.bind(r._socket),r._closeTimeout)}function fo(){let r=this[De];if(this.removeListener("close",fo),this.removeListener("data",Mn),this.removeListener("end",go),r._readyState=be.CLOSING,!this._readableState.endEmitted&&!r._closeFrameReceived&&!r._receiver._writableState.errorEmitted&&this._readableState.length!==0){let t=this.read(this._readableState.length);r._receiver.write(t)}r._receiver.end(),this[De]=void 0,clearTimeout(r._closeTimer),r._receiver._writableState.finished||r._receiver._writableState.errorEmitted?r.emitClose():(r._receiver.on("error",oo),r._receiver.on("finish",oo))}function Mn(r){this[De]._receiver.write(r)||this.pause()}function go(){let r=this[De];r._readyState=be.CLOSING,r._receiver.end(),this.end()}function yo(){let r=this[De];this.removeListener("error",yo),this.on("error",co),r&&(r._readyState=be.CLOSING,this.destroy())}});var xo=Ve((Np,ko)=>{"use strict";var Op=Ln(),{Duplex:Ed}=require("stream");function wo(r){r.emit("close")}function Pd(){!this.destroyed&&this._writableState.finished&&this.destroy()}function bo(r){this.removeListener("error",bo),this.destroy(),this.listenerCount("error")===0&&this.emit("error",r)}function Rd(r,t){let e=!0,s=new Ed({...t,autoDestroy:!1,emitClose:!1,objectMode:!1,writableObjectMode:!1});return r.on("message",function(a,i){let o=!i&&s._readableState.objectMode?a.toString():a;s.push(o)||r.pause()}),r.once("error",function(a){s.destroyed||(e=!1,s.destroy(a))}),r.once("close",function(){s.destroyed||s.push(null)}),s._destroy=function(n,a){if(r.readyState===r.CLOSED){a(n),process.nextTick(wo,s);return}let i=!1;r.once("error",function(c){i=!0,a(c)}),r.once("close",function(){i||a(n),process.nextTick(wo,s)}),e&&r.terminate()},s._final=function(n){if(r.readyState===r.CONNECTING){r.once("open",function(){s._final(n)});return}r._socket!==null&&(r._socket._writableState.finished?(n(),s._readableState.endEmitted&&s.destroy()):(r._socket.once("finish",function(){n()}),r.close()))},s._read=function(){r.isPaused&&r.resume()},s._write=function(n,a,i){if(r.readyState===r.CONNECTING){r.once("open",function(){s._write(n,a,i)});return}r.send(n,i)},s.on("end",Pd),s.on("error",bo),s}ko.exports=Rd});var Ha=Ve((Bp,So)=>{"use strict";var{tokenChars:Dd}=Zt();function Id(r){let t=new Set,e=-1,s=-1,n=0;for(n;n<r.length;n++){let i=r.charCodeAt(n);if(s===-1&&Dd[i]===1)e===-1&&(e=n);else if(n!==0&&(i===32||i===9))s===-1&&e!==-1&&(s=n);else if(i===44){if(e===-1)throw new SyntaxError(`Unexpected character at index ${n}`);s===-1&&(s=n);let o=r.slice(e,s);if(t.has(o))throw new SyntaxError(`The "${o}" subprotocol is duplicated`);t.add(o),e=s=-1}else throw new SyntaxError(`Unexpected character at index ${n}`)}if(e===-1||s!==-1)throw new SyntaxError("Unexpected end of input");let a=r.slice(e,n);if(t.has(a))throw new SyntaxError(`The "${a}" subprotocol is duplicated`);return t.add(a),t}So.exports={parse:Id}});var Ro=Ve(($p,Po)=>{"use strict";var Md=require("events"),Fn=require("http"),{Duplex:Up}=require("stream"),{createHash:Ld}=require("crypto"),Co=Rn(),Ot=Qt(),Fd=Ha(),Od=Ln(),{CLOSE_TIMEOUT:Nd,GUID:Bd,kWebSocket:Ud}=gt(),$d=/^[+/0-9A-Za-z]{22}==$/,To=0,_o=1,Eo=2,qa=class extends Md{constructor(t,e){if(super(),t={allowSynchronousEvents:!0,autoPong:!0,maxBufferedChunks:1024*1024,maxFragments:128*1024,maxPayload:100*1024*1024,skipUTF8Validation:!1,perMessageDeflate:!1,handleProtocols:null,clientTracking:!0,closeTimeout:Nd,verifyClient:null,noServer:!1,backlog:null,server:null,host:null,path:null,port:null,WebSocket:Od,...t},t.port==null&&!t.server&&!t.noServer||t.port!=null&&(t.server||t.noServer)||t.server&&t.noServer)throw new TypeError('One and only one of the "port", "server", or "noServer" options must be specified');if(t.port!=null?(this._server=Fn.createServer((s,n)=>{let a=Fn.STATUS_CODES[426];n.writeHead(426,{"Content-Length":a.length,"Content-Type":"text/plain"}),n.end(a)}),this._server.listen(t.port,t.host,t.backlog,e)):t.server&&(this._server=t.server),this._server){let s=this.emit.bind(this,"connection");this._removeListeners=jd(this._server,{listening:this.emit.bind(this,"listening"),error:this.emit.bind(this,"error"),upgrade:(n,a,i)=>{this.handleUpgrade(n,a,i,s)}})}t.perMessageDeflate===!0&&(t.perMessageDeflate={}),t.clientTracking&&(this.clients=new Set,this._shouldEmitClose=!1),this.options=t,this._state=To}address(){if(this.options.noServer)throw new Error('The server is operating in "noServer" mode');return this._server?this._server.address():null}close(t){if(this._state===Eo){t&&this.once("close",()=>{t(new Error("The server is not running"))}),process.nextTick(Ds,this);return}if(t&&this.once("close",t),this._state!==_o)if(this._state=_o,this.options.noServer||this.options.server)this._server&&(this._removeListeners(),this._removeListeners=this._server=null),this.clients?this.clients.size?this._shouldEmitClose=!0:process.nextTick(Ds,this):process.nextTick(Ds,this);else{let e=this._server;this._removeListeners(),this._removeListeners=this._server=null,e.close(()=>{Ds(this)})}}shouldHandle(t){if(this.options.path){let e=t.url.indexOf("?");if((e!==-1?t.url.slice(0,e):t.url)!==this.options.path)return!1}return!0}handleUpgrade(t,e,s,n){e.on("error",Ao);let a=t.headers["sec-websocket-key"],i=t.headers.upgrade,o=+t.headers["sec-websocket-version"];if(t.method!=="GET"){Nt(this,t,e,405,"Invalid HTTP method");return}if(i===void 0||i.toLowerCase()!=="websocket"){Nt(this,t,e,400,"Invalid Upgrade header");return}if(a===void 0||!$d.test(a)){Nt(this,t,e,400,"Missing or invalid Sec-WebSocket-Key header");return}if(o!==13&&o!==8){Nt(this,t,e,400,"Missing or invalid Sec-WebSocket-Version header",{"Sec-WebSocket-Version":"13, 8"});return}if(!this.shouldHandle(t)){Is(e,400);return}let c=t.headers["sec-websocket-protocol"],l=new Set;if(c!==void 0)try{l=Fd.parse(c)}catch{Nt(this,t,e,400,"Invalid Sec-WebSocket-Protocol header");return}let h=t.headers["sec-websocket-extensions"],d={};if(this.options.perMessageDeflate&&h!==void 0){let u=new Ot({...this.options.perMessageDeflate,isServer:!0,maxPayload:this.options.maxPayload});try{let p=Co.parse(h);p[Ot.extensionName]&&(u.accept(p[Ot.extensionName]),d[Ot.extensionName]=u)}catch{Nt(this,t,e,400,"Invalid or unacceptable Sec-WebSocket-Extensions header");return}}if(this.options.verifyClient){let u={origin:t.headers[`${o===8?"sec-websocket-origin":"origin"}`],secure:!!(t.socket.authorized||t.socket.encrypted),req:t};if(this.options.verifyClient.length===2){this.options.verifyClient(u,(p,m,f,g)=>{if(!p)return Is(e,m||401,f,g);this.completeUpgrade(d,a,l,t,e,s,n)});return}if(!this.options.verifyClient(u))return Is(e,401)}this.completeUpgrade(d,a,l,t,e,s,n)}completeUpgrade(t,e,s,n,a,i,o){if(!a.readable||!a.writable)return a.destroy();if(a[Ud])throw new Error("server.handleUpgrade() was called more than once with the same socket, possibly due to a misconfiguration");if(this._state>To)return Is(a,503);let l=["HTTP/1.1 101 Switching Protocols","Upgrade: websocket","Connection: Upgrade",`Sec-WebSocket-Accept: ${Ld("sha1").update(e+Bd).digest("base64")}`],h=new this.options.WebSocket(null,void 0,this.options);if(s.size){let d=this.options.handleProtocols?this.options.handleProtocols(s,n):s.values().next().value;d&&(l.push(`Sec-WebSocket-Protocol: ${d}`),h._protocol=d)}if(t[Ot.extensionName]){let d=t[Ot.extensionName].params,u=Co.format({[Ot.extensionName]:[d]});l.push(`Sec-WebSocket-Extensions: ${u}`),h._extensions=t}this.emit("headers",l,n),a.write(l.concat(`\r
2
2
  `).join(`\r
3
- `)),a.removeListener("error",_o),h.setSocket(a,r,{allowSynchronousEvents:this.options.allowSynchronousEvents,maxBufferedChunks:this.options.maxBufferedChunks,maxFragments:this.options.maxFragments,maxPayload:this.options.maxPayload,skipUTF8Validation:this.options.skipUTF8Validation}),this.clients&&(this.clients.add(h),h.on("close",()=>{this.clients.delete(h),this._shouldEmitClose&&!this.clients.size&&process.nextTick(Ds,this)})),o(h,n)}};Eo.exports=qa;function $d(i,t){for(let e of Object.keys(t))i.on(e,t[e]);return function(){for(let s of Object.keys(t))i.removeListener(s,t[s])}}function Ds(i){i._state=Ao,i.emit("close")}function _o(){this.destroy()}function Is(i,t,e,s){e=e||Fn.STATUS_CODES[t],s={Connection:"close","Content-Type":"text/html","Content-Length":Buffer.byteLength(e),...s},i.once("finish",i.destroy),i.end(`HTTP/1.1 ${t} ${Fn.STATUS_CODES[t]}\r
3
+ `)),a.removeListener("error",Ao),h.setSocket(a,i,{allowSynchronousEvents:this.options.allowSynchronousEvents,maxBufferedChunks:this.options.maxBufferedChunks,maxFragments:this.options.maxFragments,maxPayload:this.options.maxPayload,skipUTF8Validation:this.options.skipUTF8Validation}),this.clients&&(this.clients.add(h),h.on("close",()=>{this.clients.delete(h),this._shouldEmitClose&&!this.clients.size&&process.nextTick(Ds,this)})),o(h,n)}};Po.exports=qa;function jd(r,t){for(let e of Object.keys(t))r.on(e,t[e]);return function(){for(let s of Object.keys(t))r.removeListener(s,t[s])}}function Ds(r){r._state=Eo,r.emit("close")}function Ao(){this.destroy()}function Is(r,t,e,s){e=e||Fn.STATUS_CODES[t],s={Connection:"close","Content-Type":"text/html","Content-Length":Buffer.byteLength(e),...s},r.once("finish",r.destroy),r.end(`HTTP/1.1 ${t} ${Fn.STATUS_CODES[t]}\r
4
4
  `+Object.keys(s).map(n=>`${n}: ${s[n]}`).join(`\r
5
5
  `)+`\r
6
6
  \r
7
- `+e)}function Ot(i,t,e,s,n,a){if(i.listenerCount("wsClientError")){let r=new Error(n);Error.captureStackTrace(r,Ot),i.emit("wsClientError",r,e,t)}else Is(e,s,n,a)}});var _h={};cl(_h,{default:()=>zn});module.exports=dl(_h);var Os=require("fs"),Xa=require("os"),Qa=require("path"),Se=require("obsidian");var $t="agent-fleet-agents";var _t="agent-fleet-dashboard",it="agent-fleet-chat",rt={fleetFolder:"_fleet",claudeCliPath:"claude",codexCliPath:"codex",defaultModel:"default",awsRegion:"us-east-1",maxConcurrentRuns:2,runLogRetentionDays:30,catchUpMissedTasks:!0,notificationLevel:"all",showStatusBar:!0,mcpApiKeys:{},mcpTokens:{},channelCredentials:{},maxConcurrentChannelSessions:5,channelIdleTimeoutMinutes:15,channelRateLimitPerConversation:20,channelRateLimitWindowMinutes:5,chatWatchdogMinutes:10,defaultFileHashes:{}},ii=["agents","skills","tasks","runs","memory","channels","mcp","usage"],js=1500,ri="0 3 * * *",oi=3;var Ii=require("path"),x=require("obsidian");var Jn=[{path:"agents/fleet-orchestrator/CONTEXT.md",content:`---
7
+ `+e)}function Nt(r,t,e,s,n,a){if(r.listenerCount("wsClientError")){let i=new Error(n);Error.captureStackTrace(i,Nt),r.emit("wsClientError",i,e,t)}else Is(e,s,n,a)}});var Ah={};dl(Ah,{default:()=>zn});module.exports=hl(Ah);var Os=require("fs"),Xa=require("os"),Qa=require("path"),Se=require("obsidian");var jt="agent-fleet-agents";var _t="agent-fleet-dashboard",rt="agent-fleet-chat",it={fleetFolder:"_fleet",claudeCliPath:"claude",codexCliPath:"codex",defaultModel:"default",awsRegion:"us-east-1",maxConcurrentRuns:2,runLogRetentionDays:30,catchUpMissedTasks:!0,notificationLevel:"all",showStatusBar:!0,mcpApiKeys:{},mcpTokens:{},channelCredentials:{},maxConcurrentChannelSessions:5,channelIdleTimeoutMinutes:15,channelRateLimitPerConversation:20,channelRateLimitWindowMinutes:5,chatWatchdogMinutes:10,defaultFileHashes:{}},rr=["agents","skills","tasks","runs","memory","channels","mcp","usage"],js=1500,ir="0 3 * * *",or=3;var Mr=require("path"),x=require("obsidian");var Jn=[{path:"agents/fleet-orchestrator/CONTEXT.md",content:`---
8
8
  {}
9
9
  ---
10
10
 
@@ -61,9 +61,9 @@ You have deep knowledge of (delegated to the \`agent-fleet-system\` skill):
61
61
  - How to create, modify, and configure agents, tasks, skills, and channels
62
62
  - The scheduling system (cron expressions, task types, heartbeat schedules)
63
63
  - Heartbeat configuration \u2014 autonomous periodic agent runs via HEARTBEAT.md
64
- - Channels \u2014 connecting agents to external chat platforms (Slack, Telegram)
65
- - Multi-agent routing via @agent-name prefix, /agents command, and Telegram inline keyboard
66
- - MCP server management \u2014 assigning servers to agents via mcp_servers field
64
+ - Channels \u2014 connecting agents to external chat platforms (Slack, Telegram, Discord)
65
+ - Multi-agent routing via @agent-name prefix, /agents command, and inline keyboard / button pickers
66
+ - MCP server management \u2014 a fleet-owned registry (\`_fleet/mcp/<name>.md\`); register once and grant per agent via the mcp_servers field; works on both Claude Code and Codex backends
67
67
  - Permission modes and security rules
68
68
  - **Wiki Keeper** \u2014 scoped self-maintaining wikis with inbox + watched ingestion modes, the three bundled skills (wiki-ingest / wiki-query / wiki-lint), and per-scope instances
69
69
  - **Consumer agents** \u2014 the \`wiki_references\` config block lets any agent read + contribute to wikis it doesn't own
@@ -85,13 +85,13 @@ When asked to set up a heartbeat:
85
85
 
86
86
  When asked to set up a channel:
87
87
  1. Create a channel file in _fleet/channels/
88
- 2. Explain required external setup (Slack app with Socket Mode, or Telegram bot via BotFather)
88
+ 2. Explain required external setup (Slack app with Socket Mode, Telegram bot via BotFather, or Discord bot via the Developer Portal with the Message Content intent)
89
89
  3. Set up the allowed agents for multi-agent routing if needed
90
90
 
91
91
  When asked about MCP servers:
92
- 1. Explain that MCP servers are managed from the dashboard (add/remove/authenticate)
93
- 2. Show how to assign servers to agents via the mcp_servers field in agent.md
94
- 3. Explain OAuth authentication flow for HTTP/SSE servers
92
+ 1. Explain that MCP servers are managed from the dashboard (add/remove/authenticate) and live in a fleet-owned registry at \`_fleet/mcp/<name>.md\` \u2014 register once, available to any agent on either adapter
93
+ 2. Show how to assign servers to agents via the mcp_servers field in agent.md (empty list = all enabled servers)
94
+ 3. Explain authentication for HTTP/SSE servers (OAuth 2.1 PKCE or static bearer token, stored in the OS keychain and projected per run)
95
95
 
96
96
  When asked to set up a **Wiki Keeper**:
97
97
  1. Explain that Wiki Keepers are created through Settings \u2192 Agent Fleet \u2192 Wiki Keepers \u2192 + Add (NOT by hand-editing files) because the UI creates the agent folder, the sibling lint task, and seeds the scope's inbox/topics/index/log together.
@@ -444,124 +444,7 @@ auto_compact_threshold: 70 # was 85 by default
444
444
  \`\`\`
445
445
 
446
446
  Explains: "Auto-compact now kicks in at 70% context instead of 85%. Next time the agent's chat reaches that threshold, the session will automatically send \`/compact\` before the next user message, and the user will see a 'Conversation compacted (N \u2192 M tokens)' notification bubble in the chat. Set to 0 to disable the auto-trigger; users can still type \`/compact\` manually at any time."
447
- `},{path:"skills/agent-fleet-system/references.md",content:`# References
448
-
449
- ## Permission Modes
450
-
451
- | Mode | Unblocked commands | Blocked (deny list) | Best for |
452
- |---|---|---|---|
453
- | bypassPermissions | Auto-runs everything | Hard-blocked | Trusted agents with a blacklist |
454
- | dontAsk | Only allow-listed | Hard-blocked | Locked-down agents with a whitelist |
455
- | acceptEdits | File edits auto-approved | Hard-blocked | Agents editing files only |
456
- | plan | Read-only | Hard-blocked | Research/analysis |
457
- | default | Prompts for permission | Hard-blocked | Not useful for headless |
458
-
459
- These are the Claude Code permission modes. **Codex agents** use sandbox levels instead \u2014 \`workspace-write\` / \`read-only\` (and \`bypassPermissions\` maps to full access); the modes above are mapped to the nearest Codex equivalent. \`Bash(...)\` allow/deny rules are translated to Codex execpolicy where possible (command-prefix patterns only). See the "Agent Configuration" permissions notes in \`tools.md\` for the full Codex behavior. The "Claude Code CLI Flags" section below applies to \`claude-code\` agents only.
460
-
461
- ## Cron Expression Format
462
-
463
- Five fields: \`minute hour day-of-month month day-of-week\`
464
-
465
- | Field | Values | Special |
466
- |---|---|---|
467
- | Minute | 0-59 | \`*/N\` = every N minutes |
468
- | Hour | 0-23 | \`*/N\` = every N hours |
469
- | Day of month | 1-31 | \`*\` = every day |
470
- | Month | 1-12 | \`*\` = every month |
471
- | Day of week | 0-7 (0,7=Sun) | \`1-5\` = weekdays |
472
-
473
- Examples:
474
- - \`*/15 * * * *\` \u2014 every 15 minutes
475
- - \`0 9 * * *\` \u2014 daily at 9 AM
476
- - \`0 9 * * 1-5\` \u2014 weekdays at 9 AM
477
- - \`30 18 * * 5\` \u2014 Fridays at 6:30 PM
478
- - \`0 0 1 * *\` \u2014 first of every month at midnight
479
-
480
- ## Claude Code CLI Flags
481
-
482
- The plugin spawns Claude Code with:
483
- \`\`\`
484
- claude -p "<prompt>" --output-format stream-json --verbose [--model <model>]
485
- \`\`\`
486
-
487
- On macOS/Linux, commands run through a login shell (\`/bin/zsh -l -c\` or \`/bin/bash -l -c\`) so shell profile environment variables are available. On Windows, commands spawn directly \u2014 Windows inherits environment variables from the system without a shell wrapper.
488
-
489
- ## Environment Variables
490
-
491
- API tokens and secrets should be set in your shell profile:
492
- - **macOS:** \`~/.zshenv\` or \`~/.zprofile\`
493
- - **Linux:** \`~/.bashrc\` or \`~/.profile\`
494
- - **Windows:** System Environment Variables (Settings \u2192 System \u2192 Advanced \u2192 Environment Variables)
495
-
496
- These are inherited by all agent processes. Never store tokens in vault files.
497
-
498
- ## Channel Types
499
-
500
- | Type | Transport | Status |
501
- |---|---|---|
502
- | slack | Socket Mode WebSocket + Assistants API | Supported |
503
- | telegram | Long-poll via HTTPS (getUpdates) | Supported |
504
- | discord | Gateway WebSocket | Coming soon |
505
-
506
- **Slack requirements:** Slack app with Socket Mode enabled, bot token (xoxb-) + app-level token (xapp-), scopes: chat:write, im:history, im:read, im:write, app_mentions:read, assistant:write, commands.
507
-
508
- **Telegram requirements:** Bot created via @BotFather, bot token. Optional: disable privacy mode for group access, enable threaded mode for forum topics.
509
-
510
- **Channel constraints:**
511
- - Agents with \`approval_required\` cannot be bound to channels (would deadlock)
512
- - Credentials are stored in the OS keychain via Obsidian's SecretStorage API (macOS Keychain, Windows Credential Manager, Linux Secret Service)
513
- - \`allowed_users\` is checked against the platform's verified sender field
514
-
515
- ## Heartbeat vs Tasks
516
-
517
- | | Heartbeat | Task |
518
- |---|---|---|
519
- | Defined in | HEARTBEAT.md in agent folder | _fleet/tasks/<name>.md |
520
- | Prompt source | Heartbeat body | Task body |
521
- | "Run Now" button | Uses heartbeat instruction | Uses task prompt |
522
- | Delivery | Run log + optional channel post (\`channel\` in HEARTBEAT.md) | Run log + optional channel post (\`channel\` in task frontmatter) |
523
- | Scope | One per agent | Many per agent |
524
- | Best for | Autonomous periodic monitoring | Specific scheduled work items |
525
-
526
- ## File Naming Conventions
527
-
528
- - Agent folders: lowercase, kebab-case (\`my-agent\`)
529
- - Skill folders: lowercase, kebab-case (\`git-operations\`)
530
- - Task files: lowercase, kebab-case (\`check-deploy.md\`)
531
- - Channel files: lowercase, kebab-case (\`my-slack.md\`)
532
- - Run logs: auto-generated (\`HHMMSS-agent-task.md\`)
533
-
534
- ## Model Resolution
535
-
536
- When a run happens, the model passed to \`claude --model\` is resolved in this order:
537
-
538
- 1. **\`task.model\`** \u2014 per-task override (if set and non-empty)
539
- 2. **\`agent.model\`** \u2014 per-agent setting (canonical home: \`config.md\` for folder agents)
540
- 3. **\`settings.defaultModel\`** \u2014 plugin-wide default
541
- 4. If all three are empty or one of the sentinels (\`""\`, \`"default"\`, \`"subscription"\`), \`--model\` is omitted \u2192 CLI picks its subscription default.
542
-
543
- Use **aliases** (\`opus\`, \`sonnet\`, \`haiku\`, \`opusplan\`) for backend-agnostic, future-proof selection. They're resolved inside Claude Code itself and work identically on direct API, Bedrock, Vertex, Foundry, and Mantle. Use **Custom** (free text) for pinned concrete IDs when reproducibility matters.
544
-
545
- Run log frontmatter records both what was requested (\`model: opus\`) and what the CLI concretely resolved to (\`resolved_concrete_model: claude-opus-4-7\`) for audit traceability.
546
-
547
- ## Shared Subscription Rate Limits
548
-
549
- All agents authenticate through the same Claude Pro/Max subscription. The rate-limit window (typically 5-hour rolling) is **shared across every agent**. A single context-full agent burning quota on retries will cause other agents to fail requests until the window resets.
550
-
551
- Mitigations:
552
- - Set \`auto_compact_threshold\` on chat-heavy agents so long sessions auto-summarize before they exhaust capacity.
553
- - Use per-task \`model: haiku\` overrides for cheap/routine work to reduce quota pressure.
554
- - The chat stats strip shows the current quota window type and reset time \u2014 hover the \`CONTEXT\` pill for details.
555
-
556
- ## Error Handling in Chat
557
-
558
- When the CLI returns an error result (context overflow, rate limit, auth issue), the session:
559
- - Emits an \`error\` stream event with a human-readable reason drawn from \`api_error_status\` / \`subtype\` / \`result\`
560
- - Renders a red error bubble in the chat: \`Error: <reason>\`
561
- - Flips \`isStreaming\` back to false so the stop button clears and the user can send another message
562
-
563
- A configurable watchdog (default 10 minutes; tunable via plugin Settings \u2192 "Chat watchdog timeout") additionally protects against CLI subprocess hangs: if no stream events arrive while streaming, the turn is auto-rejected, the process is killed, and a timeout error is surfaced in the chat. This guarantees the chat never gets permanently stuck.
564
- `},{path:"skills/agent-fleet-system/skill.md",content:`---
447
+ `},{path:"skills/agent-fleet-system/references.md",content:'# References\n\n## Permission Modes\n\n| Mode | Unblocked commands | Blocked (deny list) | Best for |\n|---|---|---|---|\n| bypassPermissions | Auto-runs everything | Hard-blocked | Trusted agents with a blacklist |\n| dontAsk | Only allow-listed | Hard-blocked | Locked-down agents with a whitelist |\n| acceptEdits | File edits auto-approved | Hard-blocked | Agents editing files only |\n| plan | Read-only | Hard-blocked | Research/analysis |\n| default | Prompts for permission | Hard-blocked | Not useful for headless |\n\nThese are the Claude Code permission modes. **Codex agents** use sandbox levels instead \u2014 `workspace-write` / `read-only` (and `bypassPermissions` maps to full access); the modes above are mapped to the nearest Codex equivalent. `Bash(...)` allow/deny rules are translated to Codex execpolicy where possible (command-prefix patterns only). See the "Agent Configuration" permissions notes in `tools.md` for the full Codex behavior. The "Claude Code CLI Flags" section below applies to `claude-code` agents only.\n\n## Cron Expression Format\n\nFive fields: `minute hour day-of-month month day-of-week`\n\n| Field | Values | Special |\n|---|---|---|\n| Minute | 0-59 | `*/N` = every N minutes |\n| Hour | 0-23 | `*/N` = every N hours |\n| Day of month | 1-31 | `*` = every day |\n| Month | 1-12 | `*` = every month |\n| Day of week | 0-7 (0,7=Sun) | `1-5` = weekdays |\n\nExamples:\n- `*/15 * * * *` \u2014 every 15 minutes\n- `0 9 * * *` \u2014 daily at 9 AM\n- `0 9 * * 1-5` \u2014 weekdays at 9 AM\n- `30 18 * * 5` \u2014 Fridays at 6:30 PM\n- `0 0 1 * *` \u2014 first of every month at midnight\n\n## Claude Code CLI Flags\n\nThe plugin spawns Claude Code with:\n```\nclaude -p "<prompt>" --output-format stream-json --verbose [--model <model>]\n```\n\nOn macOS/Linux, commands run through a login shell (`/bin/zsh -l -c` or `/bin/bash -l -c`) so shell profile environment variables are available. On Windows, commands spawn directly \u2014 Windows inherits environment variables from the system without a shell wrapper.\n\n## Environment Variables\n\nAPI tokens and secrets should be set in your shell profile:\n- **macOS:** `~/.zshenv` or `~/.zprofile`\n- **Linux:** `~/.bashrc` or `~/.profile`\n- **Windows:** System Environment Variables (Settings \u2192 System \u2192 Advanced \u2192 Environment Variables)\n\nThese are inherited by all agent processes. Never store tokens in vault files.\n\n## Channel Types\n\n| Type | Transport | Status |\n|---|---|---|\n| slack | Socket Mode WebSocket + Assistants API | Supported |\n| telegram | Long-poll via HTTPS (getUpdates) | Supported |\n| discord | Gateway WebSocket + REST | Supported (since 0.14.0) |\n\n**Slack requirements:** Slack app with Socket Mode enabled, bot token (xoxb-) + app-level token (xapp-), scopes: chat:write, im:history, im:read, im:write, app_mentions:read, assistant:write, commands.\n\n**Telegram requirements:** Bot created via @BotFather, bot token. Optional: disable privacy mode for group access, enable threaded mode for forum topics.\n\n**Discord requirements:** Bot created at the Discord Developer Portal with the **Message Content** privileged intent enabled (without it the bot receives empty message text), bot token. Invite with scopes `bot` + `applications.commands` and permissions Send Messages + Read Message History \u2014 or just DM the bot. Supports `@agent-name` routing, the `/agents` slash picker, and image attachments; the allowlist is enforced on the authenticated sender\'s numeric Discord user id.\n\n**Channel constraints:**\n- Agents with `approval_required` cannot be bound to channels (would deadlock)\n- Credentials are stored in the OS keychain via Obsidian\'s SecretStorage API (macOS Keychain, Windows Credential Manager, Linux Secret Service)\n- `allowed_users` is checked against the platform\'s verified sender field\n\n## Heartbeat vs Tasks\n\n| | Heartbeat | Task |\n|---|---|---|\n| Defined in | HEARTBEAT.md in agent folder | _fleet/tasks/<name>.md |\n| Prompt source | Heartbeat body | Task body |\n| "Run Now" button | Uses heartbeat instruction | Uses task prompt |\n| Delivery | Run log + optional channel post (`channel` in HEARTBEAT.md) | Run log + optional channel post (`channel` in task frontmatter) |\n| Scope | One per agent | Many per agent |\n| Best for | Autonomous periodic monitoring | Specific scheduled work items |\n\n## File Naming Conventions\n\n- Agent folders: lowercase, kebab-case (`my-agent`)\n- Skill folders: lowercase, kebab-case (`git-operations`)\n- Task files: lowercase, kebab-case (`check-deploy.md`)\n- Channel files: lowercase, kebab-case (`my-slack.md`)\n- Run logs: auto-generated (`HHMMSS-agent-task.md`)\n\n## Model Resolution\n\nWhen a run happens, the model passed to `claude --model` is resolved in this order:\n\n1. **`task.model`** \u2014 per-task override (if set and non-empty)\n2. **`agent.model`** \u2014 per-agent setting (canonical home: `config.md` for folder agents)\n3. **`settings.defaultModel`** \u2014 plugin-wide default\n4. If all three are empty or one of the sentinels (`""`, `"default"`, `"subscription"`), `--model` is omitted \u2192 CLI picks its subscription default.\n\nUse **aliases** (`opus`, `sonnet`, `haiku`, `opusplan`) for backend-agnostic, future-proof selection. They\'re resolved inside Claude Code itself and work identically on direct API, Bedrock, Vertex, Foundry, and Mantle. Use **Custom** (free text) for pinned concrete IDs when reproducibility matters.\n\nRun log frontmatter records both what was requested (`model: opus`) and what the CLI concretely resolved to (`resolved_concrete_model: claude-opus-4-7`) for audit traceability.\n\n## Shared Subscription Rate Limits\n\nAll agents authenticate through the same Claude Pro/Max subscription. The rate-limit window (typically 5-hour rolling) is **shared across every agent**. A single context-full agent burning quota on retries will cause other agents to fail requests until the window resets.\n\nMitigations:\n- Set `auto_compact_threshold` on chat-heavy agents so long sessions auto-summarize before they exhaust capacity.\n- Use per-task `model: haiku` overrides for cheap/routine work to reduce quota pressure.\n- The chat stats strip shows the current quota window type and reset time \u2014 hover the `CONTEXT` pill for details.\n\n## Error Handling in Chat\n\nWhen the CLI returns an error result (context overflow, rate limit, auth issue), the session:\n- Emits an `error` stream event with a human-readable reason drawn from `api_error_status` / `subtype` / `result`\n- Renders a red error bubble in the chat: `Error: <reason>`\n- Flips `isStreaming` back to false so the stop button clears and the user can send another message\n\nA configurable watchdog (default 10 minutes; tunable via plugin Settings \u2192 "Chat watchdog timeout") additionally protects against CLI subprocess hangs: if no stream events arrive while streaming, the turn is auto-rejected, the process is killed, and a timeout error is surfaced in the chat. This guarantees the chat never gets permanently stuck.\n'},{path:"skills/agent-fleet-system/skill.md",content:`---
565
448
  name: agent-fleet-system
566
449
  description: Complete knowledge of the Agent Fleet plugin \u2014 file structures, schemas, configuration, and management operations
567
450
  tags:
@@ -604,12 +487,14 @@ _fleet/
604
487
  \u251C\u2500\u2500 tasks/ Task definitions (single files)
605
488
  \u2502 \u2514\u2500\u2500 <name>.md Task with schedule and prompt
606
489
  \u251C\u2500\u2500 channels/ External chat channel bindings
607
- \u2502 \u2514\u2500\u2500 <name>.md Channel config (Slack, etc.)
490
+ \u2502 \u2514\u2500\u2500 <name>.md Channel config (Slack, Telegram, Discord)
491
+ \u251C\u2500\u2500 mcp/ MCP server registry (one file per server, since 0.13.0)
492
+ \u2502 \u2514\u2500\u2500 <name>.md Server definition (frontmatter; secrets in keychain)
608
493
  \u251C\u2500\u2500 runs/ Execution logs (auto-generated)
609
494
  \u2502 \u2514\u2500\u2500 YYYY-MM-DD/ Daily folders
610
495
  \u2502 \u2514\u2500\u2500 HHMMSS-<agent>-<task>.md
611
- \u2514\u2500\u2500 memory/ Agent memory files
612
- \u2514\u2500\u2500 <agent-name>.md
496
+ \u2514\u2500\u2500 memory/ Agent memory (two-tier store, since 0.12.0)
497
+ \u2514\u2500\u2500 <agent-name>/ working.md + raw/<date>.md + candidates.json + pending/
613
498
  \`\`\`
614
499
 
615
500
  ## Creating an Agent
@@ -789,10 +674,33 @@ channel_context: |
789
674
 
790
675
  Telegram uses long-poll HTTPS (no WebSocket, no SDK). Features: typing indicators, inline keyboard agent picker via \`/agents\`, slash command autocomplete, group chat and forum topic support.
791
676
 
677
+ ### Discord Channel
678
+ \`\`\`yaml
679
+ ---
680
+ name: my-discord
681
+ type: discord
682
+ default_agent: fleet-orchestrator
683
+ allowed_agents:
684
+ - fleet-orchestrator
685
+ - site-monitor
686
+ enabled: true
687
+ credential_ref: my-discord-creds
688
+ allowed_users:
689
+ - "496534272610664448"
690
+ per_user_sessions: true
691
+ channel_context: |
692
+ You are being contacted via Discord. Keep replies concise. Discord uses
693
+ standard Markdown (NOT Slack mrkdwn): **bold**, *italic*, \`code\`, fenced
694
+ blocks with a language tag. Discord does NOT render Markdown tables.
695
+ ---
696
+ \`\`\`
697
+
698
+ Discord uses the Gateway (outbound WebSocket) + REST. Features: \`@agent-name\` routing, the \`/agents\` slash picker, image attachments, allowlist on the authenticated sender, and reconnect/resume. Requires the **Message Content** privileged intent on the bot. See \`DISCORD_SETUP.md\`.
699
+
792
700
  **Important notes:**
793
701
  - \`credential_ref\` must match a credential name in Settings \u2192 Agent Fleet \u2192 Channel Credentials
794
702
  - Credentials are stored securely in the OS keychain via Obsidian's SecretStorage API
795
- - \`allowed_users\`: Slack user IDs (start with U) or Telegram user IDs (numeric)
703
+ - \`allowed_users\`: Slack user IDs (start with U), Telegram user IDs (numeric), or Discord user IDs (numeric)
796
704
  - Agents with \`approval_required\` set cannot be bound to a channel
797
705
  - Multi-agent routing: type \`@agent-name: message\` to switch agents, or use \`/agents\` for interactive picker
798
706
  - Obsidian must be running for channels to work \u2014 when closed, bots go offline
@@ -902,22 +810,21 @@ mcp_servers:
902
810
  ---
903
811
  \`\`\`
904
812
 
905
- At runtime, the plugin writes a temporary \`settings.local.json\` in the agent's working directory that maps these server names to Claude Code's \`mcp__<name>\` allow entries. The file is restored after the run.
813
+ **MCP v2 \u2014 one registry, projected per run (since 0.13.0).** MCP servers live in a fleet-owned registry: one markdown file per server at \`_fleet/mcp/<name>.md\` (frontmatter; secrets never in the vault). Register a server **once** and it's available to **any** agent on **either** adapter \u2014 Claude Code or Codex. At spawn time, only the enabled servers an agent is granted are **projected** into whichever adapter it uses: Claude Code gets a merged \`--mcp-config\`, Codex gets \`-c mcp_servers.*\` overrides. Your native \`~/.claude.json\` and \`~/.codex/config.toml\` are **read-only** \u2014 Agent Fleet never mutates them. On first load, your existing Claude and Codex servers are imported into the registry (deduplicated, marked \`imported\`), and any bearer tokens found are moved into the keychain.
814
+
815
+ Per-agent grants: leave the agent's \`mcp_servers\` list empty to grant every enabled server, or list specific ones. The selection applies identically on both adapters.
906
816
 
907
817
  ### Server Types
908
818
 
909
- - **stdio** \u2014 local process spawned by Claude CLI (e.g., \`npx @some/mcp-server\`)
819
+ - **stdio** \u2014 local process spawned by the CLI (e.g., \`npx @some/mcp-server\`)
910
820
  - **HTTP / SSE** \u2014 remote server accessed via URL, often with OAuth authentication
911
821
 
912
- ### OAuth Authentication
822
+ ### Authentication
823
+
824
+ HTTP/SSE servers authenticate **once** and the token is stored in the OS keychain, then projected per run (to Claude as an \`Authorization\` header, to Codex via \`bearer_token_env_var\` \u2014 passed through the spawn environment, never written to argv or a config file). Two options:
913
825
 
914
- HTTP/SSE servers that require OAuth can be authenticated from the dashboard:
915
- 1. Click "Authenticate" on the server card
916
- 2. Plugin discovers OAuth endpoints automatically (RFC 8414 / RFC 9728)
917
- 3. Registers via Dynamic Client Registration
918
- 4. Opens browser for PKCE authorization flow
919
- 5. Tokens stored in OS keychain and injected into Claude CLI config
920
- 6. Background token refresh keeps agents authenticated
826
+ - **Static bearer token** \u2014 paste it on the server card.
827
+ - **OAuth 2.1 PKCE** \u2014 from the dashboard: click "Authenticate"; the plugin discovers OAuth endpoints automatically (RFC 8414 / RFC 9728), registers via Dynamic Client Registration, opens the browser for the PKCE flow, stores tokens in the keychain, and refreshes them in the background to keep agents authenticated.
921
828
 
922
829
  ## Modifying Agents, Tasks, Skills, or Channels
923
830
 
@@ -11722,45 +11629,49 @@ python scripts/office/validate.py <path> [--original <original_file>] [--auto-re
11722
11629
 
11723
11630
  - \`paraId\`/\`durableId\` values that exceed OOXML limits
11724
11631
  - Missing \`xml:space="preserve"\` on \`w:t\` elements with whitespace
11725
- `}];var Ws=require("obsidian");function J(i){let t=i.match(/^---\n([\s\S]*?)\n---\n?([\s\S]*)$/);if(!t)return{frontmatter:{},body:i.trim()};let e=t[1]??"",s=t[2]??"",n;try{n=(0,Ws.parseYaml)(e)??{}}catch(a){console.warn("Agent Fleet: malformed YAML frontmatter, treating as empty",a),n={}}return{frontmatter:n,body:s.trim()}}function H(i,t){let e=(0,Ws.stringifyYaml)(i).trim(),s=t.trim();return`---
11632
+ `}];var Ws=require("obsidian");function J(r){let t=r.match(/^---\n([\s\S]*?)\n---\n?([\s\S]*)$/);if(!t)return{frontmatter:{},body:r.trim()};let e=t[1]??"",s=t[2]??"",n;try{n=(0,Ws.parseYaml)(e)??{}}catch(a){console.warn("Agent Fleet: malformed YAML frontmatter, treating as empty",a),n={}}return{frontmatter:n,body:s.trim()}}function H(r,t){let e=(0,Ws.stringifyYaml)(r).trim(),s=t.trim();return`---
11726
11633
  ${e}
11727
11634
  ---
11728
11635
 
11729
11636
  ${s}
11730
- `}function oe(i){return i.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/(^-|-$)/g,"")}function jt(i,t){return i.length<=t?i:`${i.slice(0,t-1)}\u2026`}var Hs=2,di=["Preferences","Procedures","Observations","Recent"],hi="Recent",Qn=/\[REMEMBER(?::([a-zA-Z]+))?\]([\s\S]*?)\[\/REMEMBER\]/g;function hl(i){switch((i??"").toLowerCase()){case"pin":case"preference":case"preferences":return{pinned:!0,section:"Preferences"};case"procedure":case"procedures":return{pinned:!1,section:"Procedures"};case"observation":case"observations":return{pinned:!1,section:"Observations"};default:return{pinned:!1}}}function qs(i){let t=[];for(let e of i.matchAll(Qn)){let s=(e[2]??"").trim();if(!s)continue;let n=hl(e[1]);t.push({text:s,pinned:n.pinned,section:n.section})}return t}function hs(i){return i.replace(Qn,"").replace(/[ \t]+\n/g,`
11637
+ `}function oe(r){return r.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/(^-|-$)/g,"")}function Wt(r,t){return r.length<=t?r:`${r.slice(0,t-1)}\u2026`}var Hs=2,dr=["Preferences","Procedures","Observations","Recent"],hr="Recent",Qn=/\[REMEMBER(?::([a-zA-Z]+))?\]([\s\S]*?)\[\/REMEMBER\]/g;function ul(r){switch((r??"").toLowerCase()){case"pin":case"preference":case"preferences":return{pinned:!0,section:"Preferences"};case"procedure":case"procedures":return{pinned:!1,section:"Procedures"};case"observation":case"observations":return{pinned:!1,section:"Observations"};default:return{pinned:!1}}}function qs(r){let t=[];for(let e of r.matchAll(Qn)){let s=(e[2]??"").trim();if(!s)continue;let n=ul(e[1]);t.push({text:s,pinned:n.pinned,section:n.section})}return t}function At(r){return r.replace(Qn,"").replace(/[ \t]+\n/g,`
11731
11638
  `).replace(/\n{3,}/g,`
11732
11639
 
11733
- `).trim()}function ui(i){let t=i.replace(Qn,""),e=t.match(/\[REMEMBER(?::[a-zA-Z]+)?\][\s\S]*$/i);if(e&&e.index!==void 0)return t.slice(0,e.index);let s=t.match(/\[(?:R(?:E(?:M(?:E(?:M(?:B(?:E(?:R(?::[a-zA-Z]*)?)?)?)?)?)?)?)?)?$/i);return s&&s.index!==void 0?t.slice(0,s.index):t}var ul={Preferences:"Preferences",Procedures:"Procedures",Observations:"Observations",Recent:"Recent (uncurated)"};function pl(i){let t=i.trim().toLowerCase();return t.startsWith("preference")?"Preferences":t.startsWith("procedure")?"Procedures":t.startsWith("observation")?"Observations":t.startsWith("recent")?"Recent":"Observations"}function us(i){return Math.ceil(i.length/4)}var li=500;function pi(i){let t=i.replace(/\s+/g," ").trim();return t.length<=li?t:`${t.slice(0,li-1).trimEnd()}\u2026`}function Xn(i){return typeof i=="string"?i:void 0}function ml(i,t){return typeof i=="number"&&Number.isFinite(i)?i:t}function mi(i,t){return{filePath:i,agent:t,schema:Hs,tokenEstimate:0,sections:[]}}var fl=/\s*<!--\s*([\s\S]*?)\s*-->\s*$/,ci=/^\[pin\]\s+/i;function gl(i){let t=i.match(/^[-*]\s+(.*)$/);if(!t)return null;let e=(t[1]??"").trim();if(!e)return null;let s,n,a=e.match(fl);if(a){e=e.slice(0,a.index).trim();let o=(a[1]??"").trim(),c=o.match(/^src:(\S+)(?:\s+(.+))?$/);c?(s=c[1],n=c[2]?.trim()||void 0):o&&(n=o)}let r=!1;return ci.test(e)&&(r=!0,e=e.replace(ci,"").trim()),e?{text:e,source:s,date:n,pinned:r}:null}function fi(i){let t=i.pinned?"[pin] ":"",e="";if(i.source||i.date){let s=[];i.source&&s.push(`src:${i.source}`),i.date&&s.push(i.date),e=` <!-- ${s.join(" ")} -->`}return`- ${t}${i.text}${e}`}function Ke(i){return yl(i).map(e=>{let s=e.entries.map(fi).join(`
11734
- `);return`## ${ul[e.name]}
11640
+ `).trim()}function ur(r){let t=r.replace(Qn,""),e=t.match(/\[REMEMBER(?::[a-zA-Z]+)?\][\s\S]*$/i);if(e&&e.index!==void 0)return t.slice(0,e.index);let s=t.match(/\[(?:R(?:E(?:M(?:E(?:M(?:B(?:E(?:R(?::[a-zA-Z]*)?)?)?)?)?)?)?)?)?$/i);return s&&s.index!==void 0?t.slice(0,s.index):t}var pl={Preferences:"Preferences",Procedures:"Procedures",Observations:"Observations",Recent:"Recent (uncurated)"};function ml(r){let t=r.trim().toLowerCase();return t.startsWith("preference")?"Preferences":t.startsWith("procedure")?"Procedures":t.startsWith("observation")?"Observations":t.startsWith("recent")?"Recent":"Observations"}function us(r){return Math.ceil(r.length/4)}var lr=500;function pr(r){let t=r.replace(/\s+/g," ").trim();return t.length<=lr?t:`${t.slice(0,lr-1).trimEnd()}\u2026`}function Xn(r){return typeof r=="string"?r:void 0}function fl(r,t){return typeof r=="number"&&Number.isFinite(r)?r:t}function mr(r,t){return{filePath:r,agent:t,schema:Hs,tokenEstimate:0,sections:[]}}var gl=/\s*<!--\s*([\s\S]*?)\s*-->\s*$/,cr=/^\[pin\]\s+/i;function yl(r){let t=r.match(/^[-*]\s+(.*)$/);if(!t)return null;let e=(t[1]??"").trim();if(!e)return null;let s,n,a=e.match(gl);if(a){e=e.slice(0,a.index).trim();let o=(a[1]??"").trim(),c=o.match(/^src:(\S+)(?:\s+(.+))?$/);c?(s=c[1],n=c[2]?.trim()||void 0):o&&(n=o)}let i=!1;return cr.test(e)&&(i=!0,e=e.replace(cr,"").trim()),e?{text:e,source:s,date:n,pinned:i}:null}function fr(r){let t=r.pinned?"[pin] ":"",e="";if(r.source||r.date){let s=[];r.source&&s.push(`src:${r.source}`),r.date&&s.push(r.date),e=` <!-- ${s.join(" ")} -->`}return`- ${t}${r.text}${e}`}function Ke(r){return vl(r).map(e=>{let s=e.entries.map(fr).join(`
11641
+ `);return`## ${pl[e.name]}
11735
11642
  ${s}`}).join(`
11736
11643
 
11737
- `)}function yl(i){let t=[];for(let e of di){let s=i.find(n=>n.name===e);s&&s.entries.length>0&&t.push(s)}return t}function gi(i,t,e){let{frontmatter:s,body:n}=J(i),a=zs(n);return{filePath:t,agent:Xn(s.agent)??e,schema:ml(s.schema,Hs),lastUpdated:Xn(s.last_updated),lastReflection:Xn(s.last_reflection),tokenEstimate:us(Ke(a)),sections:a}}function zs(i){let t=new Map,e=(a,r)=>{let o=t.get(a)??[];o.push(r),t.set(a,o)},s="Observations";for(let a of i.split(`
11738
- `)){let r=a.match(/^#{1,6}\s+(.+?)\s*$/);if(r){s=pl(r[1]??"");continue}let o=gl(a);o&&e(s,o)}let n=[];for(let a of di){let r=t.get(a);r&&r.length&&n.push({name:a,entries:r})}return n}function yi(i){let t=Ke(i.sections),e={agent:i.agent,schema:i.schema||Hs,last_updated:i.lastUpdated??"",token_estimate:us(t)};return i.lastReflection&&(e.last_reflection=i.lastReflection),H(e,t||"## Observations")}function vi(i,t,e,s){if(t.length===0)return i;let n=i.sections.map(o=>({name:o.name,entries:[...o.entries]})),a=n.find(o=>o.name===e);return a||(a={name:e,entries:[]},n.push(a)),a.entries.push(...t),{...i,sections:n,lastUpdated:s??i.lastUpdated,tokenEstimate:us(Ke(n))}}var vl=["Recent","Observations","Procedures"];function wi(i,t){if(i.tokenEstimate<=t)return{wm:i,spilled:[]};let e=i.sections.map(o=>({name:o.name,entries:[...o.entries]})),s=[],n=Ke(e).length,a=()=>Math.ceil(n/4)>t;for(let o of vl){if(!a())break;let c=e.find(l=>l.name===o);if(c)for(;c.entries.length>0&&a();){let l=c.entries.findIndex(d=>!d.pinned);if(l===-1)break;let h=c.entries.splice(l,1)[0];h&&(s.push(h),n-=fi(h).length+1)}}let r=e.filter(o=>o.entries.length>0);return{wm:{...i,sections:r,tokenEstimate:us(Ke(r))},spilled:s}}function Zn(i,t,e,s){let n=zs(i);return{filePath:t,agent:e,schema:Hs,lastUpdated:s,tokenEstimate:us(Ke(n)),sections:n}}function bi(i,t){let e=(i?.sections??[]).flatMap(r=>r.entries).filter(r=>r.pinned);if(e.length===0||t.some(r=>r.entries.some(o=>o.pinned)))return t;let n=t.map(r=>({name:r.name,entries:[...r.entries]})),a=n.find(r=>r.name==="Preferences");return a||(a={name:"Preferences",entries:[]},n.unshift(a)),a.entries.unshift(...e),n}var wl="When you learn a durable fact about the user, their preferences, or how to do your work better, save it to memory. Prefer the `remember` tool \u2014 call remember(fact, pin?, section?); it is the reliable way to record a memory. If that tool is not available, fall back to writing [REMEMBER] <one concise fact> [/REMEMBER] in your reply (use [REMEMBER:pin] for standing preferences and hard constraints). Record only durable, reusable facts \u2014 not transient task details. These notes persist into your future runs.";function Gs(i,t){if(!i.memory)return"";let s=(t?Ke(t.sections).trim():"")||"Nothing yet \u2014 this is a fresh agent.";return`## Memory
11739
- ${wl}
11644
+ `)}function vl(r){let t=[];for(let e of dr){let s=r.find(n=>n.name===e);s&&s.entries.length>0&&t.push(s)}return t}function gr(r,t,e){let{frontmatter:s,body:n}=J(r),a=zs(n);return{filePath:t,agent:Xn(s.agent)??e,schema:fl(s.schema,Hs),lastUpdated:Xn(s.last_updated),lastReflection:Xn(s.last_reflection),tokenEstimate:us(Ke(a)),sections:a}}function zs(r){let t=new Map,e=(a,i)=>{let o=t.get(a)??[];o.push(i),t.set(a,o)},s="Observations";for(let a of r.split(`
11645
+ `)){let i=a.match(/^#{1,6}\s+(.+?)\s*$/);if(i){s=ml(i[1]??"");continue}let o=yl(a);o&&e(s,o)}let n=[];for(let a of dr){let i=t.get(a);i&&i.length&&n.push({name:a,entries:i})}return n}function yr(r){let t=Ke(r.sections),e={agent:r.agent,schema:r.schema||Hs,last_updated:r.lastUpdated??"",token_estimate:us(t)};return r.lastReflection&&(e.last_reflection=r.lastReflection),H(e,t||"## Observations")}function vr(r,t,e,s){if(t.length===0)return r;let n=r.sections.map(o=>({name:o.name,entries:[...o.entries]})),a=n.find(o=>o.name===e);return a||(a={name:e,entries:[]},n.push(a)),a.entries.push(...t),{...r,sections:n,lastUpdated:s??r.lastUpdated,tokenEstimate:us(Ke(n))}}var wl=["Recent","Observations","Procedures"];function wr(r,t){if(r.tokenEstimate<=t)return{wm:r,spilled:[]};let e=r.sections.map(o=>({name:o.name,entries:[...o.entries]})),s=[],n=Ke(e).length,a=()=>Math.ceil(n/4)>t;for(let o of wl){if(!a())break;let c=e.find(l=>l.name===o);if(c)for(;c.entries.length>0&&a();){let l=c.entries.findIndex(d=>!d.pinned);if(l===-1)break;let h=c.entries.splice(l,1)[0];h&&(s.push(h),n-=fr(h).length+1)}}let i=e.filter(o=>o.entries.length>0);return{wm:{...r,sections:i,tokenEstimate:us(Ke(i))},spilled:s}}function Zn(r,t,e,s){let n=zs(r);return{filePath:t,agent:e,schema:Hs,lastUpdated:s,tokenEstimate:us(Ke(n)),sections:n}}function br(r,t){let e=(r?.sections??[]).flatMap(i=>i.entries).filter(i=>i.pinned);if(e.length===0||t.some(i=>i.entries.some(o=>o.pinned)))return t;let n=t.map(i=>({name:i.name,entries:[...i.entries]})),a=n.find(i=>i.name==="Preferences");return a||(a={name:"Preferences",entries:[]},n.unshift(a)),a.entries.unshift(...e),n}var bl="When you learn a durable fact about the user, their preferences, or how to do your work better, save it to memory. Prefer the `remember` tool \u2014 call remember(fact, pin?, section?); it is the reliable way to record a memory. If that tool is not available, fall back to writing [REMEMBER] <one concise fact> [/REMEMBER] in your reply (use [REMEMBER:pin] for standing preferences and hard constraints). Record only durable, reusable facts \u2014 not transient task details. These notes persist into your future runs.";function Gs(r,t){if(!r.memory)return"";let s=(t?Ke(t.sections).trim():"")||"Nothing yet \u2014 this is a fresh agent.";return`## Memory
11646
+ ${bl}
11740
11647
 
11741
11648
  ### What you've learned so far
11742
- ${s}`}var mt=require("child_process"),ki=require("fs"),Ys=require("os"),At=require("path");function ea(){return(0,Ys.homedir)()}function xi(){if(process.platform==="darwin")return"/bin/zsh";for(let i of["/bin/bash","/bin/zsh","/bin/sh"])if((0,ki.existsSync)(i))return i;return"/bin/sh"}function bl(i){return`'${i.replace(/'/g,"'\\''")}'`}function dt(i,t,e){let s={cwd:e?.cwd,env:e?.env};if(process.platform==="win32")return(0,mt.spawn)(i,t,s);let n=xi(),a=[i,...t].map(bl).join(" ");return(0,mt.spawn)(n,["-l","-c",a],s)}function Si(i,t){let e={cwd:t?.cwd,env:t?.env,stdio:["pipe","pipe","pipe"]};if(process.platform==="win32")return(0,mt.spawn)(i,[],{...e,shell:!0});let s=xi();return(0,mt.spawn)(s,["-l","-c",i],e)}function Ci(i){try{require("electron").shell.openExternal(i)}catch{switch(process.platform){case"darwin":(0,mt.spawn)("open",[i],{stdio:"ignore"});break;case"win32":(0,mt.spawn)("cmd.exe",["/c","start","",i.replace(/&/g,"^&")],{stdio:"ignore"});break;default:(0,mt.spawn)("xdg-open",[i],{stdio:"ignore"});break}}}function ye(i){return i.split(/\r?\n/)}function Ti(i){let t=(0,Ys.homedir)();if(process.platform==="win32")return[i,(0,At.join)(process.env.APPDATA??"","Claude","claude.exe"),(0,At.join)(process.env.LOCALAPPDATA??"","Claude","claude.exe"),(0,At.join)(t,".local","bin","claude.exe"),"claude.exe","claude"].filter(s=>!!s&&Vs(s));let e=[i,(0,At.join)(t,".local","bin","claude")];return process.platform==="darwin"&&e.push("/opt/homebrew/bin/claude"),e.push("/usr/local/bin/claude","/usr/bin/claude","claude"),e.filter(s=>!!s&&Vs(s))}function ta(i){let t=(0,Ys.homedir)();if(process.platform==="win32")return[i,(0,At.join)(t,".local","bin","codex.exe"),"codex.exe","codex"].filter(s=>!!s&&Vs(s));let e=[i,(0,At.join)(t,".local","bin","codex")];return process.platform==="darwin"&&e.push("/opt/homebrew/bin/codex"),e.push("/usr/local/bin/codex","/usr/bin/codex","codex"),e.filter(s=>!!s&&Vs(s))}function Vs(i){return!i||/[\n\r\0]/.test(i)?!1:i.startsWith("/")?/^[\w/.@+-]+$/.test(i):i.startsWith("~")?/^~[\w/.@+-]*$/.test(i):/^[a-zA-Z]:[\\/]/.test(i)?/^[a-zA-Z]:[\\/][\w\\/. @+-]+$/.test(i):i.startsWith("\\\\")?/^\\\\[\w\\/. @+-]+$/.test(i):!i.includes("/")&&!i.includes("\\")?/^[\w.@+-]+$/.test(i):!1}function sa(i){return!!(i.includes("/")||i.includes("\\"))}function Wt(i){return typeof i=="object"&&i!==null}function A(i){return typeof i=="string"?i:void 0}function Be(i,t){return typeof i=="boolean"?i:t}function Re(i,t){return typeof i=="number"&&Number.isFinite(i)?i:t}function ue(i){return Array.isArray(i)?i.filter(t=>typeof t=="string"):[]}function _i(i){if(!Wt(i))return;let t={};for(let[e,s]of Object.entries(i))typeof s=="string"&&(t[e]=s);return Object.keys(t).length>0?t:void 0}var Ai=!1;function Ei(i,t={}){let e=i.memory_token_budget??t.memory_token_budget;return e!==void 0?Re(e,js):((i.memory_max_entries??t.memory_max_entries)!==void 0&&!Ai&&(Ai=!0,console.info(`Agent Fleet: \`memory_max_entries\` is deprecated and no longer enforced; memory is now bounded by \`memory_token_budget\` (default ${js}). Set that field to tune memory size.`)),js)}function Pi(i,t={}){let e=s=>i[s]??t[s];return{enabled:Be(e("reflection_enabled"),!1),schedule:A(e("reflection_schedule"))??ri,recurrenceThreshold:Re(e("reflection_recurrence_threshold"),oi),proposeSkills:Be(e("reflection_propose_skills"),!1),model:A(e("reflection_model"))}}function Ri(i){return(0,x.normalizePath)(i.replace(/\.md$/,".permissions.json"))}function Di(i){let t=0;for(let e=0;e<i.length;e++){let s=i.charCodeAt(e);t=(t<<5)-t+s|0}return t.toString(36)}var ps=class{constructor(t,e){this.app=t;this.settings=e;this.vault=t.vault}agents=new Map;skills=new Map;tasks=new Map;channels=new Map;mcpServers=new Map;validationIssues=new Map;channelCredentialGetter;warnedFolderAgentModelConflict=new Set;warnedLegacyPerms=new Set;migratingMemory=new Set;vault;setChannelCredentialGetter(t){this.channelCredentialGetter=t}getVaultBasePath(){let t=this.vault.adapter;return t instanceof x.FileSystemAdapter?t.getBasePath():void 0}getFleetRoot(){return(0,x.normalizePath)(this.settings.fleetFolder)}getSubfolder(t){return(0,x.normalizePath)(`${this.getFleetRoot()}/${t}`)}async ensureFleetStructure(){let t=this.getFleetRoot(),e=!this.vault.getAbstractFileByPath(t);await this.ensureFolder(t);for(let s of ii)await this.ensureFolder(this.getSubfolder(s));return e}async ensureSamples(){let t=this.getFleetRoot();for(let e of Jn){let s=(0,x.normalizePath)(`${t}/${e.path}`),n=s.substring(0,s.lastIndexOf("/"));await this.ensureFolder(n),await this.createFileIfMissing(s,e.content)}}async updateDefaults(t){let e=this.getFleetRoot(),s={...t};for(let n of Jn){let a=(0,x.normalizePath)(`${e}/${n.path}`),r=Di(n.content),o=t[n.path];if(o===r)continue;let c=this.vault.getAbstractFileByPath(a);if(!(c instanceof x.TFile)){let d=a.substring(0,a.lastIndexOf("/"));await this.ensureFolder(d),await this.createFileIfMissing(a,n.content),s[n.path]=r;continue}let l=await this.vault.cachedRead(c),h=Di(l);(!o||h===o)&&(await this.vault.modify(c,n.content),s[n.path]=r)}return s}async loadAll(){this.agents.clear(),this.skills.clear(),this.tasks.clear(),this.channels.clear(),this.mcpServers.clear(),this.validationIssues.clear(),await this.loadFolderAgents(),await this.loadFolderSkills();let t=this.vault.getMarkdownFiles().filter(e=>e.path.startsWith(`${this.getFleetRoot()}/`));for(let e of t)await this.loadFile(e);return this.validateReferences(),this.getSnapshot()}async loadFile(t){let e=typeof t=="string"?this.vault.getAbstractFileByPath(t):t;if(!(e instanceof x.TFile)||e.extension!=="md")return;if(this.isInsideAgentFolder(e.path)){await this.reloadFolderAgentContaining(e.path);return}if(this.isInsideSkillFolder(e.path)){await this.reloadFolderSkillContaining(e.path);return}this.clearStoredFile(e.path);let s=`${this.getSubfolder("channels")}/`;if(e.path.startsWith(s)){if(!e.path.slice(s.length).includes("/")){let c=await this.vault.cachedRead(e),l=this.parseChannelFile(e.path,c);l&&this.channels.set(e.path,l)}return}let n=`${this.getSubfolder("mcp")}/`;if(e.path.startsWith(n)){if(!e.path.slice(n.length).includes("/")){let c=await this.vault.cachedRead(e),l=this.parseMcpServerFile(e.path,c);l&&this.mcpServers.set(e.path,l)}return}let a=await this.vault.cachedRead(e),r=this.parseFile(e.path,a);if(r)if("taskId"in r)this.tasks.set(e.path,r);else if("model"in r){if(!r.isFolder){let o=Ri(e.path),c=this.vault.getAbstractFileByPath(o);if(c instanceof x.TFile)try{let l=await this.vault.cachedRead(c),h=JSON.parse(l);r.permissionRules={allow:ue(h.allow),deny:ue(h.deny)}}catch{}}this.agents.set(e.path,r)}else this.skills.set(e.path,r)}async reloadFolderAgentContaining(t){let e=`${this.getSubfolder("agents")}/`,n=t.slice(e.length).split("/")[0];if(!n)return;let a=(0,x.normalizePath)(`${e}${n}`),r=(0,x.normalizePath)(`${a}/agent.md`);if(this.agents.delete(r),!(this.vault.getAbstractFileByPath(a)instanceof x.TFolder))return;let c=this.vault.getAbstractFileByPath(r);if(!(c instanceof x.TFile))return;let l=await this.loadFolderAgent(a,c);l&&this.agents.set(r,l)}isInsideAgentFolder(t){let e=`${this.getSubfolder("agents")}/`;return t.startsWith(e)?t.slice(e.length).includes("/"):!1}isInsideSkillFolder(t){let e=`${this.getSubfolder("skills")}/`;return t.startsWith(e)?t.slice(e.length).includes("/"):!1}async reloadFolderSkillContaining(t){let e=`${this.getSubfolder("skills")}/`,n=t.slice(e.length).split("/")[0];if(!n)return;let a=(0,x.normalizePath)(`${e}${n}`),r=(0,x.normalizePath)(`${a}/skill.md`);if(this.skills.delete(r),!(this.vault.getAbstractFileByPath(a)instanceof x.TFolder))return;let c=this.vault.getAbstractFileByPath(r);if(!(c instanceof x.TFile))return;let l=await this.loadFolderSkill(a,c);l&&this.skills.set(r,l)}async loadFolderSkills(){let t=this.vault.getAbstractFileByPath(this.getSubfolder("skills"));if(t instanceof x.TFolder)for(let e of t.children){if(!(e instanceof x.TFolder))continue;let s=(0,x.normalizePath)(`${e.path}/skill.md`),n=this.vault.getAbstractFileByPath(s);if(!(n instanceof x.TFile))continue;let a=await this.loadFolderSkill(e.path,n);a&&this.skills.set(s,a)}}async loadFolderSkill(t,e){let s=await this.vault.cachedRead(e),{frontmatter:n,body:a}=J(s),r=A(n.name);if(!r)return this.setIssue(e.path,"Folder skill skill.md requires string field `name`."),null;let o=async c=>{let l=(0,x.normalizePath)(`${t}/${c}`),h=this.vault.getAbstractFileByPath(l);if(!(h instanceof x.TFile))return"";let d=await this.vault.cachedRead(h);return J(d).body};return{filePath:e.path,name:r,description:A(n.description),tags:ue(n.tags),body:a,toolsBody:await o("tools.md"),referencesBody:await o("references.md"),examplesBody:await o("examples.md"),isFolder:!0}}async loadFolderAgents(){let t=this.vault.getAbstractFileByPath(this.getSubfolder("agents"));if(t instanceof x.TFolder)for(let e of t.children){if(!(e instanceof x.TFolder))continue;let s=(0,x.normalizePath)(`${e.path}/agent.md`),n=this.vault.getAbstractFileByPath(s);if(!(n instanceof x.TFile))continue;let a=await this.loadFolderAgent(e.path,n);a&&this.agents.set(s,a)}}async loadFolderAgent(t,e){let s=await this.vault.cachedRead(e),{frontmatter:n,body:a}=J(s),r=A(n.name);if(!r)return this.setIssue(e.path,"Folder agent agent.md requires string field `name`."),null;let o={},c=(0,x.normalizePath)(`${t}/config.md`),l=this.vault.getAbstractFileByPath(c);if(l instanceof x.TFile){let M=await this.vault.cachedRead(l);o=J(M).frontmatter}let h={allow:[],deny:[]},d=(0,x.normalizePath)(`${t}/permissions.json`),u=this.vault.getAbstractFileByPath(d);if(u instanceof x.TFile)try{let M=await this.vault.cachedRead(u),U=JSON.parse(M);h={allow:ue(U.allow),deny:ue(U.deny)}}catch{}if(h.allow.length===0&&h.deny.length===0){let M=ue(o.allowed_tools),U=ue(o.blocked_tools);(M.length>0||U.length>0)&&(h={allow:M,deny:U},this.warnedLegacyPerms.has(r)||(this.warnedLegacyPerms.add(r),console.warn(`Agent Fleet: "${r}" still uses legacy allowed_tools/blocked_tools in config.md. Permission rules now live in permissions.json. Open this agent in Edit and Save to migrate.`)))}let p="",m=(0,x.normalizePath)(`${t}/SKILLS.md`),f=this.vault.getAbstractFileByPath(m);if(f instanceof x.TFile){let M=await this.vault.cachedRead(f);p=J(M).body}let g="",y=(0,x.normalizePath)(`${t}/CONTEXT.md`),v=this.vault.getAbstractFileByPath(y);if(v instanceof x.TFile){let M=await this.vault.cachedRead(v);g=J(M).body}let k=!1,w="",S="",T=!0,_="",D="",O=(0,x.normalizePath)(`${t}/HEARTBEAT.md`),C=this.vault.getAbstractFileByPath(O);if(C instanceof x.TFile){let M=await this.vault.cachedRead(C),U=J(M);k=Be(U.frontmatter.enabled,!1),w=A(U.frontmatter.schedule)??"",T=Be(U.frontmatter.notify,!0),_=A(U.frontmatter.channel)??"",D=A(U.frontmatter.channel_target)??"",S=U.body}let E=A(n.model),P=A(o.model);E&&P&&E!==P&&(this.warnedFolderAgentModelConflict.has(r)||(this.warnedFolderAgentModelConflict.add(r),console.warn(`Agent Fleet: "${r}" has conflicting model fields \u2014 agent.md says "${E}", config.md says "${P}". config.md wins. Remove agent.md's model field or sync the values to silence this warning.`)));let N=P??E??this.settings.defaultModel;return{filePath:e.path,name:r,description:A(n.description),model:N,adapter:A(o.adapter)??"claude-code",permissionMode:A(o.permission_mode)??"bypassPermissions",effort:A(o.effort),maxRetries:Re(o.max_retries,1),skills:ue(n.skills),mcpServers:ue(n.mcp_servers),cwd:A(o.cwd)||A(n.cwd),enabled:Be(n.enabled,!0),timeout:Re(o.timeout,Re(n.timeout,300)),approvalRequired:ue(o.approval_required),memory:Be(o.memory,Be(n.memory,!1)),memoryMaxEntries:Re(o.memory_max_entries,100),memoryTokenBudget:Ei(o,n),reflection:Pi(o,n),autoCompactThreshold:Re(o.auto_compact_threshold??n.auto_compact_threshold,85),tags:ue(n.tags),avatar:A(n.avatar)??"",body:a,contextBody:g,skillsBody:p,env:this.parseEnvMap(o.env),permissionRules:h,isFolder:!0,heartbeatEnabled:k,heartbeatSchedule:w,heartbeatBody:S,heartbeatNotify:T,heartbeatChannel:_,heartbeatChannelTarget:D,wikiKeeper:this.parseWikiKeeperConfig(o.wiki_keeper??n.wiki_keeper),wikiReferences:this.parseWikiReferences(o.wiki_references??n.wiki_references)}}parseWikiReferences(t){if(!Array.isArray(t))return;let e=[];for(let s of t)if(typeof s=="string"&&s.trim())e.push({agent:s.trim()});else if(s&&typeof s=="object"){let n=s.agent;typeof n=="string"&&n.trim()&&e.push({agent:n.trim()})}return e.length>0?e:void 0}parseWikiKeeperConfig(t){if(!t||typeof t!="object")return;let e=t;return{scopeRoot:A(e.scope_root)??"",inboxPath:A(e.inbox_path)??"_sources/inbox",archivePath:A(e.archive_path)??"_sources/archive",failedPath:A(e.failed_path)??"_sources/failed",topicsRoot:A(e.topics_root)??"_topics",indexPath:A(e.index_path)??"index.md",logPath:A(e.log_path)??"log.md",watchedFolders:ue(e.watched_folders),excludePatterns:ue(e.exclude_patterns),watchedSince:A(e.watched_since)??"",fileSubstantiveAnswers:Be(e.file_substantive_answers,!0),obsidianUrlScheme:Be(e.obsidian_url_scheme,!0),maxTokensPerIngest:Re(e.max_tokens_per_ingest,6e4),maxTokensPerRefresh:Re(e.max_tokens_per_refresh,3e4),indexSplitThreshold:Re(e.index_split_threshold,100),dedupSimilarityThreshold:Re(e.dedup_similarity_threshold,.82),summaryStaleDays:Re(e.summary_stale_days,30),stateFile:A(e.state_file)??".wiki-keeper-state.json"}}removeFile(t){this.clearStoredFile(t)}getSnapshot(){return{agents:Array.from(this.agents.values()).sort((t,e)=>t.name.localeCompare(e.name)),skills:Array.from(this.skills.values()).sort((t,e)=>t.name.localeCompare(e.name)),tasks:Array.from(this.tasks.values()).sort((t,e)=>t.taskId.localeCompare(e.taskId)),channels:Array.from(this.channels.values()).sort((t,e)=>t.name.localeCompare(e.name)),mcpServers:Array.from(this.mcpServers.values()).sort((t,e)=>t.name.localeCompare(e.name)),validationIssues:Array.from(this.validationIssues.values()).flat()}}getAgentByName(t){return Array.from(this.agents.values()).find(e=>e.name===t)}getSkillByName(t){return Array.from(this.skills.values()).find(e=>e.name===t)}getTaskById(t){return Array.from(this.tasks.values()).find(e=>e.taskId===t)}getTasksForAgent(t){return Array.from(this.tasks.values()).filter(e=>e.agent===t)}getChannelByName(t){return Array.from(this.channels.values()).find(e=>e.name===t)}getChannelsForAgent(t){return Array.from(this.channels.values()).filter(e=>e.defaultAgent===t)}getMcpServers(){return Array.from(this.mcpServers.values()).sort((t,e)=>t.name.localeCompare(e.name))}getMcpServerByName(t){return Array.from(this.mcpServers.values()).find(e=>e.name===t)}getRunsRoot(){return this.getSubfolder("runs")}getMemoryPath(t){return(0,x.normalizePath)(`${this.getSubfolder("memory")}/${oe(t)}.md`)}getMemoryDir(t){return(0,x.normalizePath)(`${this.getSubfolder("memory")}/${oe(t)}`)}getWorkingMemoryPath(t){return(0,x.normalizePath)(`${this.getMemoryDir(t)}/working.md`)}getRawMemoryPath(t,e){let s=e.slice(0,10);return(0,x.normalizePath)(`${this.getMemoryDir(t)}/raw/${s}.md`)}async readWorkingMemory(t){let e=this.getWorkingMemoryPath(t),s=this.vault.getAbstractFileByPath(e);if(s instanceof x.TFile){let r=await this.vault.cachedRead(s);return gi(r,e,t)}let n=this.getMemoryPath(t),a=this.vault.getAbstractFileByPath(n);if(a instanceof x.TFile){let r=await this.vault.cachedRead(a),{body:o}=J(r);return Zn(o,e,t)}return null}async writeWorkingMemory(t,e){let s=this.getWorkingMemoryPath(t);await this.ensureFolder(this.getMemoryDir(t));let n=yi(e),a=this.vault.getAbstractFileByPath(s);a instanceof x.TFile?await this.vault.modify(a,n):await this.createFileIfMissing(s,n);let r=this.getMemoryPath(t);r!==s&&this.vault.getAbstractFileByPath(r)&&await this.trashFile(r)}async migrateLegacyMemory(t){let e=this.getWorkingMemoryPath(t);if(this.vault.getAbstractFileByPath(e))return;let s=this.getMemoryPath(t),n=this.vault.getAbstractFileByPath(s);if(n instanceof x.TFile&&!this.migratingMemory.has(t)){this.migratingMemory.add(t);try{let a=await this.vault.cachedRead(n),{body:r}=J(a),o=new Date().toISOString(),c=r.split(`
11743
- `).map(d=>d.replace(/^\s*[-*]\s+/,"").trim()).filter(d=>d.length>0&&!/^#{1,6}\s/.test(d)),l=[`- ${o} [migrated] Imported ${c.length} legacy memory entr${c.length===1?"y":"ies"} (pre-v2).`,...c.map(d=>`- ${o} [migrated] ${d}`)];await this.appendRawMemory(t,l,o);let h=Zn(r,e,t,o);await this.writeWorkingMemory(t,h)}finally{this.migratingMemory.delete(t)}}}async migrateAllLegacyMemory(){for(let t of this.getSnapshot().agents)if(t.memory)try{await this.migrateLegacyMemory(t.name)}catch(e){console.warn(`Agent Fleet: legacy memory migration failed for "${t.name}"`,e)}}getPendingDir(t){return(0,x.normalizePath)(`${this.getMemoryDir(t)}/pending`)}getPendingDirAbsolutePath(t){let e=this.vault.adapter;return e instanceof x.FileSystemAdapter?(0,Ii.join)(e.getBasePath(),this.getPendingDir(t)):null}async ensureMemoryDir(t){await this.ensureFolder(this.getMemoryDir(t))}async readAndClearPending(t){let e=this.vault.adapter,s=this.getPendingDir(t),n;try{if(!await e.exists(s))return[];n=(await e.list(s)).files}catch{return[]}let a=[];for(let r of n)if(r.endsWith(".json"))try{let o=await e.read(r);await e.remove(r);for(let c of o.split(`
11744
- `))c.trim().length>0&&a.push(c)}catch{}return a}async readRecentRaw(t,e=2){let s=(0,x.normalizePath)(`${this.getMemoryDir(t)}/raw`),n=this.vault.getAbstractFileByPath(s);if(!(n instanceof x.TFolder))return"";let a=n.children.filter(o=>o instanceof x.TFile&&o.extension==="md").sort((o,c)=>c.name.localeCompare(o.name)).slice(0,e),r=[];for(let o of a.reverse())r.push(`### ${o.basename}
11745
- ${await this.vault.cachedRead(o)}`);return r.join(`
11649
+ ${s}`}var mt=require("child_process"),kr=require("fs"),Ys=require("os"),Et=require("path");function ea(){return(0,Ys.homedir)()}function xr(){if(process.platform==="darwin")return"/bin/zsh";for(let r of["/bin/bash","/bin/zsh","/bin/sh"])if((0,kr.existsSync)(r))return r;return"/bin/sh"}function kl(r){return`'${r.replace(/'/g,"'\\''")}'`}function dt(r,t,e){let s={cwd:e?.cwd,env:e?.env};if(process.platform==="win32")return(0,mt.spawn)(r,t,s);let n=xr(),a=[r,...t].map(kl).join(" ");return(0,mt.spawn)(n,["-l","-c",a],s)}function Sr(r,t){let e={cwd:t?.cwd,env:t?.env,stdio:["pipe","pipe","pipe"]};if(process.platform==="win32")return(0,mt.spawn)(r,[],{...e,shell:!0});let s=xr();return(0,mt.spawn)(s,["-l","-c",r],e)}function Cr(r){try{require("electron").shell.openExternal(r)}catch{switch(process.platform){case"darwin":(0,mt.spawn)("open",[r],{stdio:"ignore"});break;case"win32":(0,mt.spawn)("cmd.exe",["/c","start","",r.replace(/&/g,"^&")],{stdio:"ignore"});break;default:(0,mt.spawn)("xdg-open",[r],{stdio:"ignore"});break}}}function ye(r){return r.split(/\r?\n/)}function Tr(r){let t=(0,Ys.homedir)();if(process.platform==="win32")return[r,(0,Et.join)(process.env.APPDATA??"","Claude","claude.exe"),(0,Et.join)(process.env.LOCALAPPDATA??"","Claude","claude.exe"),(0,Et.join)(t,".local","bin","claude.exe"),"claude.exe","claude"].filter(s=>!!s&&Vs(s));let e=[r,(0,Et.join)(t,".local","bin","claude")];return process.platform==="darwin"&&e.push("/opt/homebrew/bin/claude"),e.push("/usr/local/bin/claude","/usr/bin/claude","claude"),e.filter(s=>!!s&&Vs(s))}function ta(r){let t=(0,Ys.homedir)();if(process.platform==="win32")return[r,(0,Et.join)(t,".local","bin","codex.exe"),"codex.exe","codex"].filter(s=>!!s&&Vs(s));let e=[r,(0,Et.join)(t,".local","bin","codex")];return process.platform==="darwin"&&e.push("/opt/homebrew/bin/codex"),e.push("/usr/local/bin/codex","/usr/bin/codex","codex"),e.filter(s=>!!s&&Vs(s))}function Vs(r){return!r||/[\n\r\0]/.test(r)?!1:r.startsWith("/")?/^[\w/.@+-]+$/.test(r):r.startsWith("~")?/^~[\w/.@+-]*$/.test(r):/^[a-zA-Z]:[\\/]/.test(r)?/^[a-zA-Z]:[\\/][\w\\/. @+-]+$/.test(r):r.startsWith("\\\\")?/^\\\\[\w\\/. @+-]+$/.test(r):!r.includes("/")&&!r.includes("\\")?/^[\w.@+-]+$/.test(r):!1}function sa(r){return!!(r.includes("/")||r.includes("\\"))}function _r(r){let t=new Map;for(let s of r){if(typeof s.costUsd!="number")continue;let n=t.get(s.agent);n?n.push(s):t.set(s.agent,[s])}let e=0;for(let s of t.values()){s.sort((a,i)=>a.ts.localeCompare(i.ts));let n=0;for(let a of s){let i=a.costUsd,o=i>=n?i-n:i;n=i,o!==i&&e++,a.costUsd=o}}return e}function Ht(r){return typeof r=="object"&&r!==null}function A(r){return typeof r=="string"?r:void 0}function Be(r,t){return typeof r=="boolean"?r:t}function Re(r,t){return typeof r=="number"&&Number.isFinite(r)?r:t}function ue(r){return Array.isArray(r)?r.filter(t=>typeof t=="string"):[]}function Ar(r){if(!Ht(r))return;let t={};for(let[e,s]of Object.entries(r))typeof s=="string"&&(t[e]=s);return Object.keys(t).length>0?t:void 0}var Er=!1;function Pr(r,t={}){let e=r.memory_token_budget??t.memory_token_budget;return e!==void 0?Re(e,js):((r.memory_max_entries??t.memory_max_entries)!==void 0&&!Er&&(Er=!0,console.info(`Agent Fleet: \`memory_max_entries\` is deprecated and no longer enforced; memory is now bounded by \`memory_token_budget\` (default ${js}). Set that field to tune memory size.`)),js)}function Rr(r,t={}){let e=s=>r[s]??t[s];return{enabled:Be(e("reflection_enabled"),!1),schedule:A(e("reflection_schedule"))??ir,recurrenceThreshold:Re(e("reflection_recurrence_threshold"),or),proposeSkills:Be(e("reflection_propose_skills"),!1),model:A(e("reflection_model"))}}function Dr(r){return(0,x.normalizePath)(r.replace(/\.md$/,".permissions.json"))}function Ir(r){let t=0;for(let e=0;e<r.length;e++){let s=r.charCodeAt(e);t=(t<<5)-t+s|0}return t.toString(36)}var ps=class{constructor(t,e){this.app=t;this.settings=e;this.vault=t.vault}agents=new Map;skills=new Map;tasks=new Map;channels=new Map;mcpServers=new Map;validationIssues=new Map;channelCredentialGetter;warnedFolderAgentModelConflict=new Set;warnedLegacyPerms=new Set;migratingMemory=new Set;vault;setChannelCredentialGetter(t){this.channelCredentialGetter=t}getVaultBasePath(){let t=this.vault.adapter;return t instanceof x.FileSystemAdapter?t.getBasePath():void 0}getFleetRoot(){return(0,x.normalizePath)(this.settings.fleetFolder)}getSubfolder(t){return(0,x.normalizePath)(`${this.getFleetRoot()}/${t}`)}async ensureFleetStructure(){let t=this.getFleetRoot(),e=!this.vault.getAbstractFileByPath(t);await this.ensureFolder(t);for(let s of rr)await this.ensureFolder(this.getSubfolder(s));return e}async ensureSamples(){let t=this.getFleetRoot();for(let e of Jn){let s=(0,x.normalizePath)(`${t}/${e.path}`),n=s.substring(0,s.lastIndexOf("/"));await this.ensureFolder(n),await this.createFileIfMissing(s,e.content)}}async updateDefaults(t){let e=this.getFleetRoot(),s={...t};for(let n of Jn){let a=(0,x.normalizePath)(`${e}/${n.path}`),i=Ir(n.content),o=t[n.path];if(o===i)continue;let c=this.vault.getAbstractFileByPath(a);if(!(c instanceof x.TFile)){let d=a.substring(0,a.lastIndexOf("/"));await this.ensureFolder(d),await this.createFileIfMissing(a,n.content),s[n.path]=i;continue}let l=await this.vault.cachedRead(c),h=Ir(l);(!o||h===o)&&(await this.vault.modify(c,n.content),s[n.path]=i)}return s}async loadAll(){this.agents.clear(),this.skills.clear(),this.tasks.clear(),this.channels.clear(),this.mcpServers.clear(),this.validationIssues.clear(),await this.loadFolderAgents(),await this.loadFolderSkills();let t=this.vault.getMarkdownFiles().filter(e=>e.path.startsWith(`${this.getFleetRoot()}/`));for(let e of t)await this.loadFile(e);return this.validateReferences(),this.getSnapshot()}async loadFile(t){let e=typeof t=="string"?this.vault.getAbstractFileByPath(t):t;if(!(e instanceof x.TFile)||e.extension!=="md")return;if(this.isInsideAgentFolder(e.path)){await this.reloadFolderAgentContaining(e.path);return}if(this.isInsideSkillFolder(e.path)){await this.reloadFolderSkillContaining(e.path);return}this.clearStoredFile(e.path);let s=`${this.getSubfolder("channels")}/`;if(e.path.startsWith(s)){if(!e.path.slice(s.length).includes("/")){let c=await this.vault.cachedRead(e),l=this.parseChannelFile(e.path,c);l&&this.channels.set(e.path,l)}return}let n=`${this.getSubfolder("mcp")}/`;if(e.path.startsWith(n)){if(!e.path.slice(n.length).includes("/")){let c=await this.vault.cachedRead(e),l=this.parseMcpServerFile(e.path,c);l&&this.mcpServers.set(e.path,l)}return}let a=await this.vault.cachedRead(e),i=this.parseFile(e.path,a);if(i)if("taskId"in i)this.tasks.set(e.path,i);else if("model"in i){if(!i.isFolder){let o=Dr(e.path),c=this.vault.getAbstractFileByPath(o);if(c instanceof x.TFile)try{let l=await this.vault.cachedRead(c),h=JSON.parse(l);i.permissionRules={allow:ue(h.allow),deny:ue(h.deny)}}catch{}}this.agents.set(e.path,i)}else this.skills.set(e.path,i)}async reloadFolderAgentContaining(t){let e=`${this.getSubfolder("agents")}/`,n=t.slice(e.length).split("/")[0];if(!n)return;let a=(0,x.normalizePath)(`${e}${n}`),i=(0,x.normalizePath)(`${a}/agent.md`);if(this.agents.delete(i),!(this.vault.getAbstractFileByPath(a)instanceof x.TFolder))return;let c=this.vault.getAbstractFileByPath(i);if(!(c instanceof x.TFile))return;let l=await this.loadFolderAgent(a,c);l&&this.agents.set(i,l)}isInsideAgentFolder(t){let e=`${this.getSubfolder("agents")}/`;return t.startsWith(e)?t.slice(e.length).includes("/"):!1}isInsideSkillFolder(t){let e=`${this.getSubfolder("skills")}/`;return t.startsWith(e)?t.slice(e.length).includes("/"):!1}async reloadFolderSkillContaining(t){let e=`${this.getSubfolder("skills")}/`,n=t.slice(e.length).split("/")[0];if(!n)return;let a=(0,x.normalizePath)(`${e}${n}`),i=(0,x.normalizePath)(`${a}/skill.md`);if(this.skills.delete(i),!(this.vault.getAbstractFileByPath(a)instanceof x.TFolder))return;let c=this.vault.getAbstractFileByPath(i);if(!(c instanceof x.TFile))return;let l=await this.loadFolderSkill(a,c);l&&this.skills.set(i,l)}async loadFolderSkills(){let t=this.vault.getAbstractFileByPath(this.getSubfolder("skills"));if(t instanceof x.TFolder)for(let e of t.children){if(!(e instanceof x.TFolder))continue;let s=(0,x.normalizePath)(`${e.path}/skill.md`),n=this.vault.getAbstractFileByPath(s);if(!(n instanceof x.TFile))continue;let a=await this.loadFolderSkill(e.path,n);a&&this.skills.set(s,a)}}async loadFolderSkill(t,e){let s=await this.vault.cachedRead(e),{frontmatter:n,body:a}=J(s),i=A(n.name);if(!i)return this.setIssue(e.path,"Folder skill skill.md requires string field `name`."),null;let o=async c=>{let l=(0,x.normalizePath)(`${t}/${c}`),h=this.vault.getAbstractFileByPath(l);if(!(h instanceof x.TFile))return"";let d=await this.vault.cachedRead(h);return J(d).body};return{filePath:e.path,name:i,description:A(n.description),tags:ue(n.tags),body:a,toolsBody:await o("tools.md"),referencesBody:await o("references.md"),examplesBody:await o("examples.md"),isFolder:!0}}async loadFolderAgents(){let t=this.vault.getAbstractFileByPath(this.getSubfolder("agents"));if(t instanceof x.TFolder)for(let e of t.children){if(!(e instanceof x.TFolder))continue;let s=(0,x.normalizePath)(`${e.path}/agent.md`),n=this.vault.getAbstractFileByPath(s);if(!(n instanceof x.TFile))continue;let a=await this.loadFolderAgent(e.path,n);a&&this.agents.set(s,a)}}async loadFolderAgent(t,e){let s=await this.vault.cachedRead(e),{frontmatter:n,body:a}=J(s),i=A(n.name);if(!i)return this.setIssue(e.path,"Folder agent agent.md requires string field `name`."),null;let o={},c=(0,x.normalizePath)(`${t}/config.md`),l=this.vault.getAbstractFileByPath(c);if(l instanceof x.TFile){let M=await this.vault.cachedRead(l);o=J(M).frontmatter}let h={allow:[],deny:[]},d=(0,x.normalizePath)(`${t}/permissions.json`),u=this.vault.getAbstractFileByPath(d);if(u instanceof x.TFile)try{let M=await this.vault.cachedRead(u),U=JSON.parse(M);h={allow:ue(U.allow),deny:ue(U.deny)}}catch{}if(h.allow.length===0&&h.deny.length===0){let M=ue(o.allowed_tools),U=ue(o.blocked_tools);(M.length>0||U.length>0)&&(h={allow:M,deny:U},this.warnedLegacyPerms.has(i)||(this.warnedLegacyPerms.add(i),console.warn(`Agent Fleet: "${i}" still uses legacy allowed_tools/blocked_tools in config.md. Permission rules now live in permissions.json. Open this agent in Edit and Save to migrate.`)))}let p="",m=(0,x.normalizePath)(`${t}/SKILLS.md`),f=this.vault.getAbstractFileByPath(m);if(f instanceof x.TFile){let M=await this.vault.cachedRead(f);p=J(M).body}let g="",y=(0,x.normalizePath)(`${t}/CONTEXT.md`),v=this.vault.getAbstractFileByPath(y);if(v instanceof x.TFile){let M=await this.vault.cachedRead(v);g=J(M).body}let k=!1,w="",S="",T=!0,_="",D="",O=(0,x.normalizePath)(`${t}/HEARTBEAT.md`),C=this.vault.getAbstractFileByPath(O);if(C instanceof x.TFile){let M=await this.vault.cachedRead(C),U=J(M);k=Be(U.frontmatter.enabled,!1),w=A(U.frontmatter.schedule)??"",T=Be(U.frontmatter.notify,!0),_=A(U.frontmatter.channel)??"",D=A(U.frontmatter.channel_target)??"",S=U.body}let E=A(n.model),P=A(o.model);E&&P&&E!==P&&(this.warnedFolderAgentModelConflict.has(i)||(this.warnedFolderAgentModelConflict.add(i),console.warn(`Agent Fleet: "${i}" has conflicting model fields \u2014 agent.md says "${E}", config.md says "${P}". config.md wins. Remove agent.md's model field or sync the values to silence this warning.`)));let N=P??E??this.settings.defaultModel;return{filePath:e.path,name:i,description:A(n.description),model:N,adapter:A(o.adapter)??"claude-code",permissionMode:A(o.permission_mode)??"bypassPermissions",effort:A(o.effort),maxRetries:Re(o.max_retries,1),skills:ue(n.skills),mcpServers:ue(n.mcp_servers),cwd:A(o.cwd)||A(n.cwd),enabled:Be(n.enabled,!0),timeout:Re(o.timeout,Re(n.timeout,300)),approvalRequired:ue(o.approval_required),memory:Be(o.memory,Be(n.memory,!1)),memoryMaxEntries:Re(o.memory_max_entries,100),memoryTokenBudget:Pr(o,n),reflection:Rr(o,n),autoCompactThreshold:Re(o.auto_compact_threshold??n.auto_compact_threshold,85),tags:ue(n.tags),avatar:A(n.avatar)??"",body:a,contextBody:g,skillsBody:p,env:this.parseEnvMap(o.env),permissionRules:h,isFolder:!0,heartbeatEnabled:k,heartbeatSchedule:w,heartbeatBody:S,heartbeatNotify:T,heartbeatChannel:_,heartbeatChannelTarget:D,wikiKeeper:this.parseWikiKeeperConfig(o.wiki_keeper??n.wiki_keeper),wikiReferences:this.parseWikiReferences(o.wiki_references??n.wiki_references)}}parseWikiReferences(t){if(!Array.isArray(t))return;let e=[];for(let s of t)if(typeof s=="string"&&s.trim())e.push({agent:s.trim()});else if(s&&typeof s=="object"){let n=s.agent;typeof n=="string"&&n.trim()&&e.push({agent:n.trim()})}return e.length>0?e:void 0}parseWikiKeeperConfig(t){if(!t||typeof t!="object")return;let e=t;return{scopeRoot:A(e.scope_root)??"",inboxPath:A(e.inbox_path)??"_sources/inbox",archivePath:A(e.archive_path)??"_sources/archive",failedPath:A(e.failed_path)??"_sources/failed",topicsRoot:A(e.topics_root)??"_topics",indexPath:A(e.index_path)??"index.md",logPath:A(e.log_path)??"log.md",watchedFolders:ue(e.watched_folders),excludePatterns:ue(e.exclude_patterns),watchedSince:A(e.watched_since)??"",fileSubstantiveAnswers:Be(e.file_substantive_answers,!0),obsidianUrlScheme:Be(e.obsidian_url_scheme,!0),maxTokensPerIngest:Re(e.max_tokens_per_ingest,6e4),maxTokensPerRefresh:Re(e.max_tokens_per_refresh,3e4),indexSplitThreshold:Re(e.index_split_threshold,100),dedupSimilarityThreshold:Re(e.dedup_similarity_threshold,.82),summaryStaleDays:Re(e.summary_stale_days,30),stateFile:A(e.state_file)??".wiki-keeper-state.json"}}removeFile(t){this.clearStoredFile(t)}getSnapshot(){return{agents:Array.from(this.agents.values()).sort((t,e)=>t.name.localeCompare(e.name)),skills:Array.from(this.skills.values()).sort((t,e)=>t.name.localeCompare(e.name)),tasks:Array.from(this.tasks.values()).sort((t,e)=>t.taskId.localeCompare(e.taskId)),channels:Array.from(this.channels.values()).sort((t,e)=>t.name.localeCompare(e.name)),mcpServers:Array.from(this.mcpServers.values()).sort((t,e)=>t.name.localeCompare(e.name)),validationIssues:Array.from(this.validationIssues.values()).flat()}}getAgentByName(t){return Array.from(this.agents.values()).find(e=>e.name===t)}getSkillByName(t){return Array.from(this.skills.values()).find(e=>e.name===t)}getTaskById(t){return Array.from(this.tasks.values()).find(e=>e.taskId===t)}getTasksForAgent(t){return Array.from(this.tasks.values()).filter(e=>e.agent===t)}getChannelByName(t){return Array.from(this.channels.values()).find(e=>e.name===t)}getChannelsForAgent(t){return Array.from(this.channels.values()).filter(e=>e.defaultAgent===t)}getMcpServers(){return Array.from(this.mcpServers.values()).sort((t,e)=>t.name.localeCompare(e.name))}getMcpServerByName(t){return Array.from(this.mcpServers.values()).find(e=>e.name===t)}getRunsRoot(){return this.getSubfolder("runs")}getMemoryPath(t){return(0,x.normalizePath)(`${this.getSubfolder("memory")}/${oe(t)}.md`)}getMemoryDir(t){return(0,x.normalizePath)(`${this.getSubfolder("memory")}/${oe(t)}`)}getWorkingMemoryPath(t){return(0,x.normalizePath)(`${this.getMemoryDir(t)}/working.md`)}getRawMemoryPath(t,e){let s=e.slice(0,10);return(0,x.normalizePath)(`${this.getMemoryDir(t)}/raw/${s}.md`)}async readWorkingMemory(t){let e=this.getWorkingMemoryPath(t),s=this.vault.getAbstractFileByPath(e);if(s instanceof x.TFile){let i=await this.vault.cachedRead(s);return gr(i,e,t)}let n=this.getMemoryPath(t),a=this.vault.getAbstractFileByPath(n);if(a instanceof x.TFile){let i=await this.vault.cachedRead(a),{body:o}=J(i);return Zn(o,e,t)}return null}async writeWorkingMemory(t,e){let s=this.getWorkingMemoryPath(t);await this.ensureFolder(this.getMemoryDir(t));let n=yr(e),a=this.vault.getAbstractFileByPath(s);a instanceof x.TFile?await this.vault.modify(a,n):await this.createFileIfMissing(s,n);let i=this.getMemoryPath(t);i!==s&&this.vault.getAbstractFileByPath(i)&&await this.trashFile(i)}async migrateLegacyMemory(t){let e=this.getWorkingMemoryPath(t);if(this.vault.getAbstractFileByPath(e))return;let s=this.getMemoryPath(t),n=this.vault.getAbstractFileByPath(s);if(n instanceof x.TFile&&!this.migratingMemory.has(t)){this.migratingMemory.add(t);try{let a=await this.vault.cachedRead(n),{body:i}=J(a),o=new Date().toISOString(),c=i.split(`
11650
+ `).map(d=>d.replace(/^\s*[-*]\s+/,"").trim()).filter(d=>d.length>0&&!/^#{1,6}\s/.test(d)),l=[`- ${o} [migrated] Imported ${c.length} legacy memory entr${c.length===1?"y":"ies"} (pre-v2).`,...c.map(d=>`- ${o} [migrated] ${d}`)];await this.appendRawMemory(t,l,o);let h=Zn(i,e,t,o);await this.writeWorkingMemory(t,h)}finally{this.migratingMemory.delete(t)}}}async migrateAllLegacyMemory(){for(let t of this.getSnapshot().agents)if(t.memory)try{await this.migrateLegacyMemory(t.name)}catch(e){console.warn(`Agent Fleet: legacy memory migration failed for "${t.name}"`,e)}}getPendingDir(t){return(0,x.normalizePath)(`${this.getMemoryDir(t)}/pending`)}getPendingDirAbsolutePath(t){let e=this.vault.adapter;return e instanceof x.FileSystemAdapter?(0,Mr.join)(e.getBasePath(),this.getPendingDir(t)):null}async ensureMemoryDir(t){await this.ensureFolder(this.getMemoryDir(t))}async readAndClearPending(t){let e=this.vault.adapter,s=this.getPendingDir(t),n;try{if(!await e.exists(s))return[];n=(await e.list(s)).files}catch{return[]}let a=[];for(let i of n)if(i.endsWith(".json"))try{let o=await e.read(i);await e.remove(i);for(let c of o.split(`
11651
+ `))c.trim().length>0&&a.push(c)}catch{}return a}async readRecentRaw(t,e=2){let s=(0,x.normalizePath)(`${this.getMemoryDir(t)}/raw`),n=this.vault.getAbstractFileByPath(s);if(!(n instanceof x.TFolder))return"";let a=n.children.filter(o=>o instanceof x.TFile&&o.extension==="md").sort((o,c)=>c.name.localeCompare(o.name)).slice(0,e),i=[];for(let o of a.reverse())i.push(`### ${o.basename}
11652
+ ${await this.vault.cachedRead(o)}`);return i.join(`
11746
11653
 
11747
- `)}getCandidatesPath(t){return(0,x.normalizePath)(`${this.getMemoryDir(t)}/candidates.json`)}async readCandidates(t){let e=this.getCandidatesPath(t),s=this.vault.getAbstractFileByPath(e);if(!(s instanceof x.TFile))return[];try{let n=JSON.parse(await this.vault.cachedRead(s));return Array.isArray(n)?n:[]}catch{return[]}}async writeCandidates(t,e){let s=this.getCandidatesPath(t);await this.ensureFolder(this.getMemoryDir(t));let n=JSON.stringify(e,null,2),a=this.vault.getAbstractFileByPath(s);a instanceof x.TFile?await this.vault.modify(a,n):await this.createFileIfMissing(s,n)}getProposalsDir(){return(0,x.normalizePath)(`${this.getFleetRoot()}/proposals`)}async listProposals(){let t=this.vault.getAbstractFileByPath(this.getProposalsDir());if(!(t instanceof x.TFolder))return[];let e=[];for(let s of t.children){if(!(s instanceof x.TFile)||s.extension!=="md")continue;let n=await this.readProposal(s.basename);n&&e.push(n)}return e.sort((s,n)=>n.created.localeCompare(s.created)),e}async readProposal(t){let e=(0,x.normalizePath)(`${this.getProposalsDir()}/${t}.md`),s=this.vault.getAbstractFileByPath(e);if(!(s instanceof x.TFile))return null;let{frontmatter:n,body:a}=J(await this.vault.cachedRead(s)),r=n.type==="skill_modify"?"skill_modify":"skill_create",o=n.status==="accepted"||n.status==="rejected"?n.status:"pending";return{id:t,type:r,agent:A(n.agent)??"",status:o,created:A(n.created)??"",targetSkill:A(n.target_skill),candidate:A(n.candidate),evidence:ue(n.evidence),rationale:A(n.rationale)??"",body:a}}async writeProposal(t){await this.ensureFolder(this.getProposalsDir());let e=(0,x.normalizePath)(`${this.getProposalsDir()}/${t.id}.md`),s={id:t.id,type:t.type,agent:t.agent,status:t.status,created:t.created,target_skill:t.targetSkill||void 0,candidate:t.candidate||void 0,evidence:t.evidence.length?t.evidence:void 0,rationale:t.rationale||void 0},n=H(s,t.body||""),a=this.vault.getAbstractFileByPath(e);a instanceof x.TFile?await this.vault.modify(a,n):await this.createFileIfMissing(e,n)}async setProposalStatus(t,e){let s=await this.readProposal(t);s&&(s.status=e,await this.writeProposal(s))}async applyProposal(t){if(t.type==="skill_create"){let e=t.targetSkill||`learned-${oe(t.rationale).slice(0,24)||"skill"}`,s=await this.getAvailablePath(this.getSubfolder("skills"),oe(e)),a={name:s.split("/").pop()?.replace(/\.md$/,"")||oe(e),description:t.rationale||`Auto-proposed from recurring pattern for ${t.agent}.`,tags:["proposed"]};return await this.vault.create(s,H(a,t.body||"Skill instructions go here.")),s}if(t.targetSkill){let e=this.getSkillByName(t.targetSkill);if(e){let s=this.vault.getAbstractFileByPath(e.filePath);if(s instanceof x.TFile){let n=await this.vault.cachedRead(s),a=`
11654
+ `)}getCandidatesPath(t){return(0,x.normalizePath)(`${this.getMemoryDir(t)}/candidates.json`)}async readCandidates(t){let e=this.getCandidatesPath(t),s=this.vault.getAbstractFileByPath(e);if(!(s instanceof x.TFile))return[];try{let n=JSON.parse(await this.vault.cachedRead(s));return Array.isArray(n)?n:[]}catch{return[]}}async writeCandidates(t,e){let s=this.getCandidatesPath(t);await this.ensureFolder(this.getMemoryDir(t));let n=JSON.stringify(e,null,2),a=this.vault.getAbstractFileByPath(s);a instanceof x.TFile?await this.vault.modify(a,n):await this.createFileIfMissing(s,n)}getProposalsDir(){return(0,x.normalizePath)(`${this.getFleetRoot()}/proposals`)}async listProposals(){let t=this.vault.getAbstractFileByPath(this.getProposalsDir());if(!(t instanceof x.TFolder))return[];let e=[];for(let s of t.children){if(!(s instanceof x.TFile)||s.extension!=="md")continue;let n=await this.readProposal(s.basename);n&&e.push(n)}return e.sort((s,n)=>n.created.localeCompare(s.created)),e}async readProposal(t){let e=(0,x.normalizePath)(`${this.getProposalsDir()}/${t}.md`),s=this.vault.getAbstractFileByPath(e);if(!(s instanceof x.TFile))return null;let{frontmatter:n,body:a}=J(await this.vault.cachedRead(s)),i=n.type==="skill_modify"?"skill_modify":"skill_create",o=n.status==="accepted"||n.status==="rejected"?n.status:"pending";return{id:t,type:i,agent:A(n.agent)??"",status:o,created:A(n.created)??"",targetSkill:A(n.target_skill),candidate:A(n.candidate),evidence:ue(n.evidence),rationale:A(n.rationale)??"",body:a}}async writeProposal(t){await this.ensureFolder(this.getProposalsDir());let e=(0,x.normalizePath)(`${this.getProposalsDir()}/${t.id}.md`),s={id:t.id,type:t.type,agent:t.agent,status:t.status,created:t.created,target_skill:t.targetSkill||void 0,candidate:t.candidate||void 0,evidence:t.evidence.length?t.evidence:void 0,rationale:t.rationale||void 0},n=H(s,t.body||""),a=this.vault.getAbstractFileByPath(e);a instanceof x.TFile?await this.vault.modify(a,n):await this.createFileIfMissing(e,n)}async setProposalStatus(t,e){let s=await this.readProposal(t);s&&(s.status=e,await this.writeProposal(s))}async applyProposal(t){if(t.type==="skill_create"){let e=t.targetSkill||`learned-${oe(t.rationale).slice(0,24)||"skill"}`,s=await this.getAvailablePath(this.getSubfolder("skills"),oe(e)),a={name:s.split("/").pop()?.replace(/\.md$/,"")||oe(e),description:t.rationale||`Auto-proposed from recurring pattern for ${t.agent}.`,tags:["proposed"]};return await this.vault.create(s,H(a,t.body||"Skill instructions go here.")),s}if(t.targetSkill){let e=this.getSkillByName(t.targetSkill);if(e){let s=this.vault.getAbstractFileByPath(e.filePath);if(s instanceof x.TFile){let n=await this.vault.cachedRead(s),a=`
11748
11655
 
11749
11656
  ## Proposed update (${new Date().toISOString().slice(0,10)})
11750
11657
  ${t.body}`;return await this.vault.modify(s,`${n.trimEnd()}${a}
11751
11658
  `),e.filePath}}}return null}async deleteProposal(t){await this.trashFile((0,x.normalizePath)(`${this.getProposalsDir()}/${t}.md`))}async appendRawMemory(t,e,s){if(e.length===0)return;let n=this.getRawMemoryPath(t,s);await this.ensureFolder(n.replace(/\/[^/]+$/,""));let a=e.map(o=>o.trimEnd()).join(`
11752
- `),r=this.vault.getAbstractFileByPath(n);if(r instanceof x.TFile){let o=await this.vault.cachedRead(r);await this.vault.modify(r,`${o.trimEnd()}
11659
+ `),i=this.vault.getAbstractFileByPath(n);if(i instanceof x.TFile){let o=await this.vault.cachedRead(i);await this.vault.modify(i,`${o.trimEnd()}
11753
11660
  ${a}
11754
11661
  `)}else await this.createFileIfMissing(n,`${a}
11755
- `)}getConversationsDir(t){if(t.isFolder){let s=t.filePath.replace(/\/agent\.md$/,"");return(0,x.normalizePath)(`${s}/conversations`)}let e=this.getMemoryPath(t.name).replace(/\/[^/]+$/,"");return(0,x.normalizePath)(`${e}/${t.name}-conversations`)}getConversationPath(t,e){let s=this.getConversationsDir(t),n=oe(e)||"conversation";return(0,x.normalizePath)(`${s}/${n}.json`)}async listConversations(t){let e=[],s=this.getConversationsDir(t),n=this.vault.getAbstractFileByPath(s);if(n instanceof x.TFolder)for(let a of n.children){if(!(a instanceof x.TFile)||a.extension!=="json")continue;let r=a.basename,o=await this.readConversationMeta(a,r);o&&e.push(o)}return e.sort((a,r)=>r.lastActive.localeCompare(a.lastActive)),e}async readConversationMeta(t,e){try{let s=await this.vault.cachedRead(t),n=JSON.parse(s);return{id:e,name:n.name?.trim()||"Untitled",lastActive:n.lastActive??new Date(t.stat.mtime).toISOString(),messageCount:Array.isArray(n.messages)?n.messages.length:0}}catch{return null}}async createConversation(t,e,s){let n=this.getConversationPath(t,e);if(this.vault.getAbstractFileByPath(n)instanceof x.TFile)return;let r=n.replace(/\/[^/]+$/,"");await this.ensureFolder(r);let o=new Date().toISOString(),c={sessionId:null,messages:[],lastActive:o,createdAt:o,name:s.trim()};await this.vault.create(n,JSON.stringify(c,null,2))}async renameConversation(t,e,s){let n=this.getConversationPath(t,e),a=this.vault.getAbstractFileByPath(n);if(!(a instanceof x.TFile))return;let r=await this.vault.cachedRead(a),o;try{o=JSON.parse(r)}catch{return}o.name=s.trim(),await this.vault.modify(a,JSON.stringify(o,null,2))}async deleteConversation(t,e){let s=this.getConversationPath(t,e),n=this.vault.getAbstractFileByPath(s);n instanceof x.TFile&&await this.app.fileManager.trashFile(n);let a=s.replace(/\.json$/,".threads"),r=this.vault.getAbstractFileByPath(a);r instanceof x.TFolder&&await this.app.fileManager.trashFile(r);let o=this.getConversationsDir(t),c=this.vault.getAbstractFileByPath(o);c instanceof x.TFolder&&c.children.length===0&&await this.app.fileManager.trashFile(c)}async getMemory(t){let e=await this.readWorkingMemory(t);return e?{filePath:e.filePath,agent:e.agent,lastUpdated:e.lastUpdated,body:Ke(e.sections)}:null}async appendMemory(t,e){if(e.length===0)return;let s=this.getMemoryPath(t),n=this.vault.getAbstractFileByPath(s),a=new Date().toISOString(),r=e.map(o=>`- ${o.trim()}`).join(`
11662
+ `)}getConversationsDir(t){if(t.isFolder){let s=t.filePath.replace(/\/agent\.md$/,"");return(0,x.normalizePath)(`${s}/conversations`)}let e=this.getMemoryPath(t.name).replace(/\/[^/]+$/,"");return(0,x.normalizePath)(`${e}/${t.name}-conversations`)}getConversationPath(t,e){let s=this.getConversationsDir(t),n=oe(e)||"conversation";return(0,x.normalizePath)(`${s}/${n}.json`)}async listConversations(t){let e=[],s=this.getConversationsDir(t),n=this.vault.getAbstractFileByPath(s);if(n instanceof x.TFolder)for(let a of n.children){if(!(a instanceof x.TFile)||a.extension!=="json")continue;let i=a.basename,o=await this.readConversationMeta(a,i);o&&e.push(o)}return e.sort((a,i)=>i.lastActive.localeCompare(a.lastActive)),e}async readConversationMeta(t,e){try{let s=await this.vault.cachedRead(t),n=JSON.parse(s);return{id:e,name:n.name?.trim()||"Untitled",lastActive:n.lastActive??new Date(t.stat.mtime).toISOString(),messageCount:Array.isArray(n.messages)?n.messages.length:0}}catch{return null}}async createConversation(t,e,s){let n=this.getConversationPath(t,e);if(this.vault.getAbstractFileByPath(n)instanceof x.TFile)return;let i=n.replace(/\/[^/]+$/,"");await this.ensureFolder(i);let o=new Date().toISOString(),c={sessionId:null,messages:[],lastActive:o,createdAt:o,name:s.trim()};await this.vault.create(n,JSON.stringify(c,null,2))}async renameConversation(t,e,s){let n=this.getConversationPath(t,e),a=this.vault.getAbstractFileByPath(n);if(!(a instanceof x.TFile))return;let i=await this.vault.cachedRead(a),o;try{o=JSON.parse(i)}catch{return}o.name=s.trim(),await this.vault.modify(a,JSON.stringify(o,null,2))}async deleteConversation(t,e){let s=this.getConversationPath(t,e),n=this.vault.getAbstractFileByPath(s);n instanceof x.TFile&&await this.app.fileManager.trashFile(n);let a=s.replace(/\.json$/,".threads"),i=this.vault.getAbstractFileByPath(a);i instanceof x.TFolder&&await this.app.fileManager.trashFile(i);let o=this.getConversationsDir(t),c=this.vault.getAbstractFileByPath(o);c instanceof x.TFolder&&c.children.length===0&&await this.app.fileManager.trashFile(c)}async getMemory(t){let e=await this.readWorkingMemory(t);return e?{filePath:e.filePath,agent:e.agent,lastUpdated:e.lastUpdated,body:Ke(e.sections)}:null}async appendMemory(t,e){if(e.length===0)return;let s=this.getMemoryPath(t),n=this.vault.getAbstractFileByPath(s),a=new Date().toISOString(),i=e.map(o=>`- ${o.trim()}`).join(`
11756
11663
  `);if(n instanceof x.TFile){let c=`${(await this.getMemory(t))?.body.trim()||"## Learned Context"}
11757
11664
 
11758
- ${r}`.trim();await this.vault.modify(n,H({agent:t,last_updated:a},c));return}await this.createFileIfMissing(s,H({agent:t,last_updated:a},`## Learned Context
11665
+ ${i}`.trim();await this.vault.modify(n,H({agent:t,last_updated:a},c));return}await this.createFileIfMissing(s,H({agent:t,last_updated:a},`## Learned Context
11759
11666
 
11760
- ${r}`))}async listRecentRuns(t=50){let e=this.vault.getAbstractFileByPath(this.getRunsRoot());if(!(e instanceof x.TFolder))return[];let s=[];this.collectMarkdownChildren(e,s),s.sort((r,o)=>o.path.localeCompare(r.path));let n=s.slice(0,t),a=[];for(let r of n){let o=await this.readRunLog(r);o&&a.push(o)}return a.sort((r,o)=>o.started.localeCompare(r.started))}async listRunsSince(t){let e=this.vault.getAbstractFileByPath(this.getRunsRoot());if(!(e instanceof x.TFolder))return[];let s=`${t.getFullYear()}-${String(t.getMonth()+1).padStart(2,"0")}-${String(t.getDate()).padStart(2,"0")}`,n=[];for(let r of e.children)r instanceof x.TFolder&&(r.name<s||this.collectMarkdownChildren(r,n));let a=[];for(let r of n){let o=await this.readRunLog(r);o&&a.push(o)}return a.sort((r,o)=>o.started.localeCompare(r.started))}usageLedgerPath(t){return(0,x.normalizePath)(`${this.getSubfolder("usage")}/${t.slice(0,10)}.jsonl`)}async appendUsage(t){await this.ensureFolder(this.getSubfolder("usage"));let e=this.usageLedgerPath(t.ts),s=`${JSON.stringify(t)}
11761
- `,n=this.vault.adapter;await n.exists(e)?await n.append(e,s):await n.write(e,s)}async readUsageSince(t){let e=this.getSubfolder("usage"),s=this.vault.adapter;if(!await s.exists(e))return[];let n=`${t.getFullYear()}-${String(t.getMonth()+1).padStart(2,"0")}-${String(t.getDate()).padStart(2,"0")}`,a=[],r=await s.list(e);for(let o of r.files){if(!o.endsWith(".jsonl")||(o.split("/").pop()??"").replace(/\.jsonl$/,"")<n)continue;let l;try{l=await s.read(o)}catch{continue}for(let h of l.split(`
11762
- `)){let d=h.trim();if(d)try{a.push(JSON.parse(d))}catch{}}}return a}async readRunLog(t){let e=await this.vault.cachedRead(t),{frontmatter:s,body:n}=J(e),a=n.match(/## Prompt\n([\s\S]*?)(?:\n## Result\n|\n## Output\n|$)/),r=n.match(/## Result\n([\s\S]*?)(?:\n## Output\n|$)/),o=n.match(/## Output\n([\s\S]*?)(?:\n## Tools Used\n|$)/),c=n.match(/## Tools Used\n([\s\S]*?)(?:\n## STDERR\n|$)/);return{filePath:t.path,runId:A(s.run_id)??t.basename,agent:A(s.agent)??"unknown",task:A(s.task)??"unknown",status:A(s.status)??"failure",started:A(s.started)??new Date(t.stat.ctime).toISOString(),completed:A(s.completed),durationSeconds:Re(s.duration_seconds,0),tokensUsed:typeof s.tokens_used=="number"?s.tokens_used:void 0,costUsd:typeof s.cost_usd=="number"?s.cost_usd:void 0,model:A(s.model)??rt.defaultModel,modelSource:(()=>{let l=A(s.model_source);if(l==="task"||l==="agent"||l==="settings"||l==="cli-default")return l})(),concreteModel:A(s.resolved_concrete_model),exitCode:typeof s.exit_code=="number"?s.exit_code:null,tags:ue(s.tags),prompt:a?.[1]?.trim()??"",output:o?.[1]?.trim()??"",finalResult:r?.[1]?.trim()||void 0,toolsUsed:c?.[1]?ye(c[1]).map(l=>l.replace(/^- /,"").trim()).filter(Boolean):[],approvals:this.parseApprovals(s.approvals)}}async writeRunLog(t){let e=new Date(t.started),s=(0,x.normalizePath)(`${this.getRunsRoot()}/${e.toISOString().slice(0,10)}`);await this.ensureFolder(s);let n=`${e.toISOString().slice(11,19).replace(/:/g,"")}-${oe(t.agent)}-${oe(t.task)}.md`,a=(0,x.normalizePath)(`${s}/${n}`),r=H({run_id:t.runId,agent:t.agent,task:t.task,status:t.status,started:t.started,completed:t.completed,duration_seconds:t.durationSeconds,tokens_used:t.tokensUsed,cost_usd:t.costUsd,model:t.model,model_source:t.modelSource,resolved_concrete_model:t.concreteModel,exit_code:t.exitCode,tags:t.tags,approvals:t.approvals},["## Prompt","",t.prompt.trim(),"",...t.finalResult&&t.finalResult.trim()?["## Result","",t.finalResult.trim(),""]:[],"## Output","",t.output.trim()||"(no output)","","## Tools Used","",...t.toolsUsed.length>0?t.toolsUsed.map(c=>`- ${c}`):["- none"],...t.stderr?["","## STDERR","",t.stderr.trim()]:[]].join(`
11763
- `)),o=this.vault.getAbstractFileByPath(a);return o instanceof x.TFile?await this.vault.modify(o,r):await this.vault.create(a,r),a}async updateTaskRunMetadata(t,e){let s=this.vault.getAbstractFileByPath(t.filePath);if(!(s instanceof x.TFile))return;let n=await this.vault.cachedRead(s),{frontmatter:a,body:r}=J(n),o={...a,last_run:e.lastRun??t.lastRun,next_run:e.nextRun??t.nextRun,run_count:e.runCount??t.runCount};await this.vault.modify(s,H(o,r)),await this.loadFile(s)}async setApprovalDecision(t,e,s){let n=this.vault.getAbstractFileByPath(t);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:r,body:o}=J(a),c=(this.parseApprovals(r.approvals)??[]).map(l=>l.tool===e?{...l,status:s,resolvedAt:new Date().toISOString()}:l);await this.vault.modify(n,H({...r,approvals:c},o))}async createAgentTemplate(t){let e=await this.getAvailablePath(this.getSubfolder("agents"),oe(t)),s=`---
11667
+ ${i}`))}async listRecentRuns(t=50){let e=this.vault.getAbstractFileByPath(this.getRunsRoot());if(!(e instanceof x.TFolder))return[];let s=[];this.collectMarkdownChildren(e,s),s.sort((i,o)=>o.path.localeCompare(i.path));let n=s.slice(0,t),a=[];for(let i of n){let o=await this.readRunLog(i);o&&a.push(o)}return a.sort((i,o)=>o.started.localeCompare(i.started))}async listRunsSince(t){let e=this.vault.getAbstractFileByPath(this.getRunsRoot());if(!(e instanceof x.TFolder))return[];let s=`${t.getFullYear()}-${String(t.getMonth()+1).padStart(2,"0")}-${String(t.getDate()).padStart(2,"0")}`,n=[];for(let i of e.children)i instanceof x.TFolder&&(i.name<s||this.collectMarkdownChildren(i,n));let a=[];for(let i of n){let o=await this.readRunLog(i);o&&a.push(o)}return a.sort((i,o)=>o.started.localeCompare(i.started))}usageLedgerPath(t){return(0,x.normalizePath)(`${this.getSubfolder("usage")}/${t.slice(0,10)}.jsonl`)}async appendUsage(t){await this.ensureFolder(this.getSubfolder("usage"));let e=this.usageLedgerPath(t.ts),s=`${JSON.stringify(t)}
11668
+ `,n=this.vault.adapter;await n.exists(e)?await n.append(e,s):await n.write(e,s)}async readUsageSince(t){let e=this.getSubfolder("usage"),s=this.vault.adapter;if(!await s.exists(e))return[];let n=`${t.getFullYear()}-${String(t.getMonth()+1).padStart(2,"0")}-${String(t.getDate()).padStart(2,"0")}`,a=[],i=await s.list(e);for(let o of i.files){if(!o.endsWith(".jsonl")||(o.split("/").pop()??"").replace(/\.jsonl$/,"")<n)continue;let l;try{l=await s.read(o)}catch{continue}for(let h of l.split(`
11669
+ `)){let d=h.trim();if(d)try{a.push(JSON.parse(d))}catch{}}}return a}async migrateUsageLedgerCosts(){let t=this.getSubfolder("usage"),e=this.vault.adapter;if(!await e.exists(t))return null;let s=(0,x.normalizePath)(`${t}/.cost-delta-v1`);if(await e.exists(s))return null;try{let n=await e.list(t),a=[],i=[];for(let c of n.files){if(!c.endsWith(".jsonl"))continue;let l;try{l=await e.read(c)}catch{continue}let h=[];for(let d of l.split(`
11670
+ `)){let u=d.trim();if(u)try{h.push(JSON.parse(u))}catch{}}a.push({path:c,records:h}),i.push(...h)}let o=_r(i);if(o>0)for(let{path:c,records:l}of a){if(l.length===0)continue;let h=l.map(d=>JSON.stringify(d)).join(`
11671
+ `)+`
11672
+ `;await e.write(c,h)}return await e.write(s,`migrated ${i.length} rows; corrected ${o}
11673
+ `),{files:a.length,rows:i.length,changed:o}}catch(n){return console.error("Agent Fleet: usage-ledger cost migration failed (ledger left untouched)",n),null}}async readRunLog(t){let e=await this.vault.cachedRead(t),{frontmatter:s,body:n}=J(e),a=n.match(/## Prompt\n([\s\S]*?)(?:\n## Result\n|\n## Output\n|$)/),i=n.match(/## Result\n([\s\S]*?)(?:\n## Output\n|$)/),o=n.match(/## Output\n([\s\S]*?)(?:\n## Tools Used\n|$)/),c=n.match(/## Tools Used\n([\s\S]*?)(?:\n## STDERR\n|$)/);return{filePath:t.path,runId:A(s.run_id)??t.basename,agent:A(s.agent)??"unknown",task:A(s.task)??"unknown",status:A(s.status)??"failure",started:A(s.started)??new Date(t.stat.ctime).toISOString(),completed:A(s.completed),durationSeconds:Re(s.duration_seconds,0),tokensUsed:typeof s.tokens_used=="number"?s.tokens_used:void 0,costUsd:typeof s.cost_usd=="number"?s.cost_usd:void 0,model:A(s.model)??it.defaultModel,modelSource:(()=>{let l=A(s.model_source);if(l==="task"||l==="agent"||l==="settings"||l==="cli-default")return l})(),concreteModel:A(s.resolved_concrete_model),exitCode:typeof s.exit_code=="number"?s.exit_code:null,tags:ue(s.tags),prompt:a?.[1]?.trim()??"",output:o?.[1]?.trim()??"",finalResult:i?.[1]?.trim()||void 0,toolsUsed:c?.[1]?ye(c[1]).map(l=>l.replace(/^- /,"").trim()).filter(Boolean):[],approvals:this.parseApprovals(s.approvals)}}async writeRunLog(t){let e=new Date(t.started),s=(0,x.normalizePath)(`${this.getRunsRoot()}/${e.toISOString().slice(0,10)}`);await this.ensureFolder(s);let n=`${e.toISOString().slice(11,19).replace(/:/g,"")}-${oe(t.agent)}-${oe(t.task)}.md`,a=(0,x.normalizePath)(`${s}/${n}`),i=H({run_id:t.runId,agent:t.agent,task:t.task,status:t.status,started:t.started,completed:t.completed,duration_seconds:t.durationSeconds,tokens_used:t.tokensUsed,cost_usd:t.costUsd,model:t.model,model_source:t.modelSource,resolved_concrete_model:t.concreteModel,exit_code:t.exitCode,tags:t.tags,approvals:t.approvals},["## Prompt","",t.prompt.trim(),"",...t.finalResult&&t.finalResult.trim()?["## Result","",t.finalResult.trim(),""]:[],"## Output","",t.output.trim()||"(no output)","","## Tools Used","",...t.toolsUsed.length>0?t.toolsUsed.map(c=>`- ${c}`):["- none"],...t.stderr?["","## STDERR","",t.stderr.trim()]:[]].join(`
11674
+ `)),o=this.vault.getAbstractFileByPath(a);return o instanceof x.TFile?await this.vault.modify(o,i):await this.vault.create(a,i),a}async updateTaskRunMetadata(t,e){let s=this.vault.getAbstractFileByPath(t.filePath);if(!(s instanceof x.TFile))return;let n=await this.vault.cachedRead(s),{frontmatter:a,body:i}=J(n),o={...a,last_run:e.lastRun??t.lastRun,next_run:e.nextRun??t.nextRun,run_count:e.runCount??t.runCount};await this.vault.modify(s,H(o,i)),await this.loadFile(s)}async setApprovalDecision(t,e,s){let n=this.vault.getAbstractFileByPath(t);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:i,body:o}=J(a),c=(this.parseApprovals(i.approvals)??[]).map(l=>l.tool===e?{...l,status:s,resolvedAt:new Date().toISOString()}:l);await this.vault.modify(n,H({...i,approvals:c},o))}async createAgentTemplate(t){let e=await this.getAvailablePath(this.getSubfolder("agents"),oe(t)),s=`---
11764
11675
  name: ${oe(t)}
11765
11676
  description:
11766
11677
  enabled: true
@@ -11769,7 +11680,7 @@ tags: []
11769
11680
  ---
11770
11681
 
11771
11682
  Agent instructions go here.
11772
- `;return await this.vault.create(e,s)}async createAgentFolder(t){let e=oe(t.name),s=(0,x.normalizePath)(`${this.getSubfolder("agents")}/${e}`);await this.ensureFolder(s);let n={name:t.name,description:t.description||void 0,avatar:t.avatar||void 0,enabled:t.enabled??!0,tags:t.tags,skills:t.skills,mcp_servers:t.mcpServers?.length?t.mcpServers:void 0};t.model&&t.model!=="default"&&(n.model=t.model);let a=(0,x.normalizePath)(`${s}/agent.md`);await this.vault.create(a,H(n,t.systemPrompt||""));let r={model:t.model||"default",adapter:t.adapter||"claude-code",timeout:t.timeout,max_retries:1,cwd:t.cwd||"",permission_mode:t.permissionMode||"bypassPermissions",effort:t.effort||void 0,approval_required:t.approvalRequired,memory:t.memory,memory_max_entries:t.memoryMaxEntries};typeof t.autoCompactThreshold=="number"&&(r.auto_compact_threshold=t.autoCompactThreshold),t.wikiReferences&&t.wikiReferences.length>0&&(r.wiki_references=t.wikiReferences.map(d=>({agent:d})));let o=(0,x.normalizePath)(`${s}/config.md`);await this.vault.create(o,H(r,""));let c=(0,x.normalizePath)(`${s}/SKILLS.md`);await this.vault.create(c,H({},t.skillsBody||""));let l=(0,x.normalizePath)(`${s}/CONTEXT.md`);await this.vault.create(l,H({},t.contextBody||""));let h=t.permissionRules;if(h&&(h.allow.length>0||h.deny.length>0)){let d=(0,x.normalizePath)(`${s}/permissions.json`);await this.vault.create(d,JSON.stringify(h,null,2)+`
11683
+ `;return await this.vault.create(e,s)}async createAgentFolder(t){let e=oe(t.name),s=(0,x.normalizePath)(`${this.getSubfolder("agents")}/${e}`);await this.ensureFolder(s);let n={name:t.name,description:t.description||void 0,avatar:t.avatar||void 0,enabled:t.enabled??!0,tags:t.tags,skills:t.skills,mcp_servers:t.mcpServers?.length?t.mcpServers:void 0};t.model&&t.model!=="default"&&(n.model=t.model);let a=(0,x.normalizePath)(`${s}/agent.md`);await this.vault.create(a,H(n,t.systemPrompt||""));let i={model:t.model||"default",adapter:t.adapter||"claude-code",timeout:t.timeout,max_retries:1,cwd:t.cwd||"",permission_mode:t.permissionMode||"bypassPermissions",effort:t.effort||void 0,approval_required:t.approvalRequired,memory:t.memory,memory_max_entries:t.memoryMaxEntries};typeof t.autoCompactThreshold=="number"&&(i.auto_compact_threshold=t.autoCompactThreshold),t.wikiReferences&&t.wikiReferences.length>0&&(i.wiki_references=t.wikiReferences.map(d=>({agent:d})));let o=(0,x.normalizePath)(`${s}/config.md`);await this.vault.create(o,H(i,""));let c=(0,x.normalizePath)(`${s}/SKILLS.md`);await this.vault.create(c,H({},t.skillsBody||""));let l=(0,x.normalizePath)(`${s}/CONTEXT.md`);await this.vault.create(l,H({},t.contextBody||""));let h=t.permissionRules;if(h&&(h.allow.length>0||h.deny.length>0)){let d=(0,x.normalizePath)(`${s}/permissions.json`);await this.vault.create(d,JSON.stringify(h,null,2)+`
11773
11684
  `)}return a}async createSkillTemplate(t){let e=await this.getAvailablePath(this.getSubfolder("skills"),oe(t)),s=`---
11774
11685
  name: ${oe(t)}
11775
11686
  description:
@@ -11783,23 +11694,23 @@ ${t.toolsBody}`)}if(t.referencesBody){let a=(0,x.normalizePath)(`${e}/references
11783
11694
 
11784
11695
  ${t.referencesBody}`)}if(t.examplesBody){let a=(0,x.normalizePath)(`${e}/examples.md`);await this.createFileIfMissing(a,`# Examples
11785
11696
 
11786
- ${t.examplesBody}`)}}async updateAgent(t,e){let s=this.getAgentByName(t);if(s)if(s.isFolder){let n=(0,x.normalizePath)(s.filePath.replace(/\/agent\.md$/,"")),a=this.vault.getAbstractFileByPath(s.filePath);if(a instanceof x.TFile){let c=await this.vault.cachedRead(a),{frontmatter:l,body:h}=J(c);e.description!==void 0&&(l.description=e.description||void 0),e.avatar!==void 0&&(l.avatar=e.avatar||void 0),e.tags!==void 0&&(l.tags=e.tags),e.skills!==void 0&&(l.skills=e.skills),e.mcpServers!==void 0&&(l.mcp_servers=e.mcpServers.length>0?e.mcpServers:void 0),e.enabled!==void 0&&(l.enabled=e.enabled),e.model!==void 0&&e.model!=="default"&&(l.model=e.model);let d=e.systemPrompt!==void 0?e.systemPrompt:h;await this.vault.modify(a,H(l,d))}let r=(0,x.normalizePath)(`${n}/config.md`),o=this.vault.getAbstractFileByPath(r);if(o instanceof x.TFile){let c=await this.vault.cachedRead(o),{frontmatter:l,body:h}=J(c);e.model!==void 0&&(l.model=e.model),e.adapter!==void 0&&(l.adapter=e.adapter),e.timeout!==void 0&&(l.timeout=e.timeout),e.cwd!==void 0&&(l.cwd=e.cwd),e.permissionMode!==void 0&&(l.permission_mode=e.permissionMode),e.effort!==void 0&&(l.effort=e.effort||void 0),e.approvalRequired!==void 0&&(l.approval_required=e.approvalRequired),e.memory!==void 0&&(l.memory=e.memory),e.memoryTokenBudget!==void 0&&(l.memory_token_budget=e.memoryTokenBudget),e.reflectionEnabled!==void 0&&(l.reflection_enabled=e.reflectionEnabled),e.reflectionSchedule!==void 0&&(l.reflection_schedule=e.reflectionSchedule),e.reflectionProposeSkills!==void 0&&(l.reflection_propose_skills=e.reflectionProposeSkills),e.autoCompactThreshold!==void 0&&(l.auto_compact_threshold=e.autoCompactThreshold),e.wikiReferences!==void 0&&(l.wiki_references=e.wikiReferences.length>0?e.wikiReferences.map(d=>({agent:d})):void 0),delete l.allowed_tools,delete l.blocked_tools,await this.vault.modify(o,H(l,h))}if(e.skillsBody!==void 0){let c=(0,x.normalizePath)(`${n}/SKILLS.md`),l=this.vault.getAbstractFileByPath(c);l instanceof x.TFile?await this.vault.modify(l,H({},e.skillsBody)):await this.vault.create(c,H({},e.skillsBody))}if(e.contextBody!==void 0){let c=(0,x.normalizePath)(`${n}/CONTEXT.md`),l=this.vault.getAbstractFileByPath(c);l instanceof x.TFile?await this.vault.modify(l,H({},e.contextBody)):await this.vault.create(c,H({},e.contextBody))}if(e.permissionRules!==void 0){let c=(0,x.normalizePath)(`${n}/permissions.json`),l=this.vault.getAbstractFileByPath(c),h=e.permissionRules;if(h.allow.length>0||h.deny.length>0){let d=JSON.stringify(h,null,2)+`
11787
- `;l instanceof x.TFile?await this.vault.modify(l,d):await this.vault.create(c,d)}else l instanceof x.TFile&&await this.app.fileManager.trashFile(l)}}else{let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:r,body:o}=J(a);e.description!==void 0&&(r.description=e.description||void 0),e.avatar!==void 0&&(r.avatar=e.avatar||void 0),e.tags!==void 0&&(r.tags=e.tags),e.skills!==void 0&&(r.skills=e.skills),e.mcpServers!==void 0&&(r.mcp_servers=e.mcpServers.length>0?e.mcpServers:void 0),e.enabled!==void 0&&(r.enabled=e.enabled),e.model!==void 0&&(r.model=e.model),e.adapter!==void 0&&(r.adapter=e.adapter),e.timeout!==void 0&&(r.timeout=e.timeout),e.cwd!==void 0&&(r.cwd=e.cwd),e.permissionMode!==void 0&&(r.permission_mode=e.permissionMode),e.effort!==void 0&&(r.effort=e.effort||void 0),e.approvalRequired!==void 0&&(r.approval_required=e.approvalRequired),e.memory!==void 0&&(r.memory=e.memory),e.autoCompactThreshold!==void 0&&(r.auto_compact_threshold=e.autoCompactThreshold),e.wikiReferences!==void 0&&(r.wiki_references=e.wikiReferences.length>0?e.wikiReferences.map(l=>({agent:l})):void 0),delete r.allowed_tools,delete r.blocked_tools;let c=e.systemPrompt!==void 0?e.systemPrompt:o;if(await this.vault.modify(n,H(r,c)),e.permissionRules!==void 0){let l=Ri(s.filePath),h=this.vault.getAbstractFileByPath(l),d=e.permissionRules;if(d.allow.length>0||d.deny.length>0){let u=JSON.stringify(d,null,2)+`
11788
- `;h instanceof x.TFile?await this.vault.modify(h,u):await this.vault.create(l,u)}else h instanceof x.TFile&&await this.app.fileManager.trashFile(h)}}}async updateTask(t,e){let s=this.getTaskById(t);if(!s)return;let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:r,body:o}=J(a);e.agent!==void 0&&(r.agent=e.agent),e.type!==void 0&&(r.type=e.type),e.schedule!==void 0&&(r.schedule=e.schedule||void 0),e.runAt!==void 0&&(r.run_at=e.runAt||void 0),e.enabled!==void 0&&(r.enabled=e.enabled),e.priority!==void 0&&(r.priority=e.priority),e.catch_up!==void 0&&(r.catch_up=e.catch_up),e.effort!==void 0&&(r.effort=e.effort||void 0),e.model!==void 0&&(r.model=e.model||void 0),e.channel!==void 0&&(r.channel=e.channel||void 0),e.channelTarget!==void 0&&(r.channel_target=e.channelTarget||void 0),e.tags!==void 0&&(r.tags=e.tags);let c=e.body!==void 0?e.body:o;await this.vault.modify(n,H(r,c))}async updateSkill(t,e){let s=this.getSkillByName(t);if(s)if(s.isFolder){let n=(0,x.normalizePath)(s.filePath.replace(/\/skill\.md$/,"")),a=this.vault.getAbstractFileByPath(s.filePath);if(a instanceof x.TFile){let r=await this.vault.cachedRead(a),{frontmatter:o,body:c}=J(r);e.description!==void 0&&(o.description=e.description||void 0),e.tags!==void 0&&(o.tags=e.tags.length>0?e.tags:void 0);let l=e.body!==void 0?e.body:c;await this.vault.modify(a,H(o,l))}if(e.toolsBody!==void 0){let r=(0,x.normalizePath)(`${n}/tools.md`),o=this.vault.getAbstractFileByPath(r);e.toolsBody&&(o instanceof x.TFile?await this.vault.modify(o,`# Tools
11697
+ ${t.examplesBody}`)}}async updateAgent(t,e){let s=this.getAgentByName(t);if(s)if(s.isFolder){let n=(0,x.normalizePath)(s.filePath.replace(/\/agent\.md$/,"")),a=this.vault.getAbstractFileByPath(s.filePath);if(a instanceof x.TFile){let c=await this.vault.cachedRead(a),{frontmatter:l,body:h}=J(c);e.description!==void 0&&(l.description=e.description||void 0),e.avatar!==void 0&&(l.avatar=e.avatar||void 0),e.tags!==void 0&&(l.tags=e.tags),e.skills!==void 0&&(l.skills=e.skills),e.mcpServers!==void 0&&(l.mcp_servers=e.mcpServers.length>0?e.mcpServers:void 0),e.enabled!==void 0&&(l.enabled=e.enabled),e.model!==void 0&&e.model!=="default"&&(l.model=e.model);let d=e.systemPrompt!==void 0?e.systemPrompt:h;await this.vault.modify(a,H(l,d))}let i=(0,x.normalizePath)(`${n}/config.md`),o=this.vault.getAbstractFileByPath(i);if(o instanceof x.TFile){let c=await this.vault.cachedRead(o),{frontmatter:l,body:h}=J(c);e.model!==void 0&&(l.model=e.model),e.adapter!==void 0&&(l.adapter=e.adapter),e.timeout!==void 0&&(l.timeout=e.timeout),e.cwd!==void 0&&(l.cwd=e.cwd),e.permissionMode!==void 0&&(l.permission_mode=e.permissionMode),e.effort!==void 0&&(l.effort=e.effort||void 0),e.approvalRequired!==void 0&&(l.approval_required=e.approvalRequired),e.memory!==void 0&&(l.memory=e.memory),e.memoryTokenBudget!==void 0&&(l.memory_token_budget=e.memoryTokenBudget),e.reflectionEnabled!==void 0&&(l.reflection_enabled=e.reflectionEnabled),e.reflectionSchedule!==void 0&&(l.reflection_schedule=e.reflectionSchedule),e.reflectionProposeSkills!==void 0&&(l.reflection_propose_skills=e.reflectionProposeSkills),e.autoCompactThreshold!==void 0&&(l.auto_compact_threshold=e.autoCompactThreshold),e.wikiReferences!==void 0&&(l.wiki_references=e.wikiReferences.length>0?e.wikiReferences.map(d=>({agent:d})):void 0),delete l.allowed_tools,delete l.blocked_tools,await this.vault.modify(o,H(l,h))}if(e.skillsBody!==void 0){let c=(0,x.normalizePath)(`${n}/SKILLS.md`),l=this.vault.getAbstractFileByPath(c);l instanceof x.TFile?await this.vault.modify(l,H({},e.skillsBody)):await this.vault.create(c,H({},e.skillsBody))}if(e.contextBody!==void 0){let c=(0,x.normalizePath)(`${n}/CONTEXT.md`),l=this.vault.getAbstractFileByPath(c);l instanceof x.TFile?await this.vault.modify(l,H({},e.contextBody)):await this.vault.create(c,H({},e.contextBody))}if(e.permissionRules!==void 0){let c=(0,x.normalizePath)(`${n}/permissions.json`),l=this.vault.getAbstractFileByPath(c),h=e.permissionRules;if(h.allow.length>0||h.deny.length>0){let d=JSON.stringify(h,null,2)+`
11698
+ `;l instanceof x.TFile?await this.vault.modify(l,d):await this.vault.create(c,d)}else l instanceof x.TFile&&await this.app.fileManager.trashFile(l)}}else{let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:i,body:o}=J(a);e.description!==void 0&&(i.description=e.description||void 0),e.avatar!==void 0&&(i.avatar=e.avatar||void 0),e.tags!==void 0&&(i.tags=e.tags),e.skills!==void 0&&(i.skills=e.skills),e.mcpServers!==void 0&&(i.mcp_servers=e.mcpServers.length>0?e.mcpServers:void 0),e.enabled!==void 0&&(i.enabled=e.enabled),e.model!==void 0&&(i.model=e.model),e.adapter!==void 0&&(i.adapter=e.adapter),e.timeout!==void 0&&(i.timeout=e.timeout),e.cwd!==void 0&&(i.cwd=e.cwd),e.permissionMode!==void 0&&(i.permission_mode=e.permissionMode),e.effort!==void 0&&(i.effort=e.effort||void 0),e.approvalRequired!==void 0&&(i.approval_required=e.approvalRequired),e.memory!==void 0&&(i.memory=e.memory),e.autoCompactThreshold!==void 0&&(i.auto_compact_threshold=e.autoCompactThreshold),e.wikiReferences!==void 0&&(i.wiki_references=e.wikiReferences.length>0?e.wikiReferences.map(l=>({agent:l})):void 0),delete i.allowed_tools,delete i.blocked_tools;let c=e.systemPrompt!==void 0?e.systemPrompt:o;if(await this.vault.modify(n,H(i,c)),e.permissionRules!==void 0){let l=Dr(s.filePath),h=this.vault.getAbstractFileByPath(l),d=e.permissionRules;if(d.allow.length>0||d.deny.length>0){let u=JSON.stringify(d,null,2)+`
11699
+ `;h instanceof x.TFile?await this.vault.modify(h,u):await this.vault.create(l,u)}else h instanceof x.TFile&&await this.app.fileManager.trashFile(h)}}}async updateTask(t,e){let s=this.getTaskById(t);if(!s)return;let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:i,body:o}=J(a);e.agent!==void 0&&(i.agent=e.agent),e.type!==void 0&&(i.type=e.type),e.schedule!==void 0&&(i.schedule=e.schedule||void 0),e.runAt!==void 0&&(i.run_at=e.runAt||void 0),e.enabled!==void 0&&(i.enabled=e.enabled),e.priority!==void 0&&(i.priority=e.priority),e.catch_up!==void 0&&(i.catch_up=e.catch_up),e.effort!==void 0&&(i.effort=e.effort||void 0),e.model!==void 0&&(i.model=e.model||void 0),e.channel!==void 0&&(i.channel=e.channel||void 0),e.channelTarget!==void 0&&(i.channel_target=e.channelTarget||void 0),e.tags!==void 0&&(i.tags=e.tags);let c=e.body!==void 0?e.body:o;await this.vault.modify(n,H(i,c))}async updateSkill(t,e){let s=this.getSkillByName(t);if(s)if(s.isFolder){let n=(0,x.normalizePath)(s.filePath.replace(/\/skill\.md$/,"")),a=this.vault.getAbstractFileByPath(s.filePath);if(a instanceof x.TFile){let i=await this.vault.cachedRead(a),{frontmatter:o,body:c}=J(i);e.description!==void 0&&(o.description=e.description||void 0),e.tags!==void 0&&(o.tags=e.tags.length>0?e.tags:void 0);let l=e.body!==void 0?e.body:c;await this.vault.modify(a,H(o,l))}if(e.toolsBody!==void 0){let i=(0,x.normalizePath)(`${n}/tools.md`),o=this.vault.getAbstractFileByPath(i);e.toolsBody&&(o instanceof x.TFile?await this.vault.modify(o,`# Tools
11789
11700
 
11790
- ${e.toolsBody}`):await this.vault.create(r,`# Tools
11701
+ ${e.toolsBody}`):await this.vault.create(i,`# Tools
11791
11702
 
11792
- ${e.toolsBody}`))}if(e.referencesBody!==void 0){let r=(0,x.normalizePath)(`${n}/references.md`),o=this.vault.getAbstractFileByPath(r);e.referencesBody&&(o instanceof x.TFile?await this.vault.modify(o,`# References
11703
+ ${e.toolsBody}`))}if(e.referencesBody!==void 0){let i=(0,x.normalizePath)(`${n}/references.md`),o=this.vault.getAbstractFileByPath(i);e.referencesBody&&(o instanceof x.TFile?await this.vault.modify(o,`# References
11793
11704
 
11794
- ${e.referencesBody}`):await this.vault.create(r,`# References
11705
+ ${e.referencesBody}`):await this.vault.create(i,`# References
11795
11706
 
11796
- ${e.referencesBody}`))}if(e.examplesBody!==void 0){let r=(0,x.normalizePath)(`${n}/examples.md`),o=this.vault.getAbstractFileByPath(r);e.examplesBody&&(o instanceof x.TFile?await this.vault.modify(o,`# Examples
11707
+ ${e.referencesBody}`))}if(e.examplesBody!==void 0){let i=(0,x.normalizePath)(`${n}/examples.md`),o=this.vault.getAbstractFileByPath(i);e.examplesBody&&(o instanceof x.TFile?await this.vault.modify(o,`# Examples
11797
11708
 
11798
- ${e.examplesBody}`):await this.vault.create(r,`# Examples
11709
+ ${e.examplesBody}`):await this.vault.create(i,`# Examples
11799
11710
 
11800
- ${e.examplesBody}`))}}else{let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:r,body:o}=J(a);e.description!==void 0&&(r.description=e.description||void 0),e.tags!==void 0&&(r.tags=e.tags.length>0?e.tags:void 0);let c=e.body!==void 0?e.body:o;await this.vault.modify(n,H(r,c))}}async deleteSkill(t){let e=this.getSkillByName(t);if(e)if(e.isFolder){let s=(0,x.normalizePath)(e.filePath.replace(/\/skill\.md$/,"")),n=this.vault.getAbstractFileByPath(s);n instanceof x.TFolder&&await this.app.fileManager.trashFile(n)}else await this.trashFile(e.filePath)}async deleteTask(t){let e=this.getTaskById(t);e&&await this.trashFile(e.filePath)}async updateChannel(t,e){let s=this.getChannelByName(t);if(!s)return;let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:r,body:o}=J(a);e.default_agent!==void 0&&(r.default_agent=e.default_agent,delete r.agent),e.allowed_agents!==void 0&&(r.allowed_agents=e.allowed_agents),e.enabled!==void 0&&(r.enabled=e.enabled),e.credential_ref!==void 0&&(r.credential_ref=e.credential_ref),e.allowed_users!==void 0&&(r.allowed_users=e.allowed_users),e.per_user_sessions!==void 0&&(r.per_user_sessions=e.per_user_sessions),e.channel_context!==void 0&&(r.channel_context=e.channel_context||void 0),e.tags!==void 0&&(r.tags=e.tags),e.type!==void 0&&(r.type=e.type),e.transport!==void 0&&(r.transport=e.transport);let c=e.body!==void 0?e.body:o;await this.vault.modify(n,H(r,c))}async deleteChannel(t){let e=this.getChannelByName(t);if(!e)return;await this.trashFile(e.filePath);let s=(0,x.normalizePath)(`${this.getSubfolder("channels")}/${oe(t)}/sessions`),n=this.vault.getAbstractFileByPath(s);n instanceof x.TFolder&&await this.app.fileManager.trashFile(n)}mcpServerFrontmatter(t){let e={name:t.name,transport:t.type,enabled:t.enabled};return t.source&&(e.source=t.source),t.type==="stdio"?(t.command&&(e.command=t.command),t.args&&t.args.length>0&&(e.args=t.args),t.env&&Object.keys(t.env).length>0&&(e.env=t.env),t.envSecretKeys&&t.envSecretKeys.length>0&&(e.env_secret_keys=t.envSecretKeys)):(t.url&&(e.url=t.url),t.headers&&Object.keys(t.headers).length>0&&(e.headers=t.headers),t.auth&&(e.auth=t.auth),t.oauth&&(t.oauth.clientId||t.oauth.resource||(t.oauth.scopes?.length??0)>0)&&(e.oauth={client_id:t.oauth.clientId||void 0,resource:t.oauth.resource||void 0,scopes:t.oauth.scopes&&t.oauth.scopes.length>0?t.oauth.scopes:void 0})),e}async saveMcpServer(t,e=""){let s=this.mcpServerFrontmatter(t),n=H(s,e.trim()),a=t.filePath?this.vault.getAbstractFileByPath(t.filePath):null;if(a instanceof x.TFile)return await this.vault.modify(a,n),a.path;let r=await this.getAvailablePath(this.getSubfolder("mcp"),oe(t.name));return await this.ensureFolder(this.getSubfolder("mcp")),await this.vault.create(r,n),r}async setMcpServerEnabled(t,e){let s=this.getMcpServerByName(t);if(!s?.filePath)return;let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:r,body:o}=J(a);r.enabled=e,await this.vault.modify(n,H(r,o))}async deleteMcpServer(t){let e=this.getMcpServerByName(t);e?.filePath&&await this.trashFile(e.filePath)}async updateHeartbeat(t,e){let s=this.getAgentByName(t);if(!s||!s.isFolder)return;let n=(0,x.normalizePath)(s.filePath.replace(/\/agent\.md$/,"")),a=(0,x.normalizePath)(`${n}/HEARTBEAT.md`),r=this.vault.getAbstractFileByPath(a);if(r instanceof x.TFile){let o=await this.vault.cachedRead(r),{frontmatter:c,body:l}=J(o);e.enabled!==void 0&&(c.enabled=e.enabled),e.schedule!==void 0&&(c.schedule=e.schedule||void 0),e.notify!==void 0&&(c.notify=e.notify),e.channel!==void 0&&(c.channel=e.channel||void 0),e.channelTarget!==void 0&&(c.channel_target=e.channelTarget||void 0);let h=e.body!==void 0?e.body:l;await this.vault.modify(r,H(c,h))}else{let o={enabled:e.enabled??!1};e.schedule&&(o.schedule=e.schedule),e.notify!==void 0&&(o.notify=e.notify),e.channel&&(o.channel=e.channel),e.channelTarget&&(o.channel_target=e.channelTarget);let c=e.body??"";await this.vault.create(a,H(o,c))}}async deleteAgent(t,e){let s=[],n=this.getAgentByName(t);if(!n)return{trashedFiles:s};if(n.isFolder){let c=(0,x.normalizePath)(n.filePath.replace(/\/agent\.md$/,"")),l=this.vault.getAbstractFileByPath(c);if(l instanceof x.TFolder){let h=[];this.collectMarkdownChildren(l,h);for(let d of h)s.push(d.path);await this.app.fileManager.trashFile(l)}}else await this.trashFile(n.filePath),s.push(n.filePath);let a=this.getMemoryPath(t);this.vault.getAbstractFileByPath(a)&&(await this.trashFile(a),s.push(a));let r=this.getMemoryDir(t),o=this.vault.getAbstractFileByPath(r);if(o instanceof x.TFolder&&(await this.app.fileManager.trashFile(o),s.push(r)),e){let c=this.getTasksForAgent(t);for(let l of c)await this.trashFile(l.filePath),s.push(l.filePath)}return{trashedFiles:s}}async trashFile(t){let e=this.vault.getAbstractFileByPath(t);e&&await this.app.fileManager.trashFile(e)}async ensureFolder(t){if(!this.vault.getAbstractFileByPath(t))try{await this.vault.createFolder(t)}catch(e){if(!(e instanceof Error?e.message:String(e)).includes("Folder already exists"))throw e}}async getAvailablePath(t,e){let s=0;for(;;){let n=s===0?"":`-${s+1}`,a=(0,x.normalizePath)(`${t}/${e}${n}.md`);if(!this.vault.getAbstractFileByPath(a))return a;s+=1}}async createFileIfMissing(t,e){if(!this.vault.getAbstractFileByPath(t))try{await this.vault.create(t,e)}catch(s){if(!(s instanceof Error?s.message:String(s)).includes("File already exists"))throw s}}collectMarkdownChildren(t,e){for(let s of t.children)s instanceof x.TFile&&s.extension==="md"&&e.push(s),s instanceof x.TFolder&&this.collectMarkdownChildren(s,e)}clearStoredFile(t){this.agents.delete(t),this.skills.delete(t),this.tasks.delete(t),this.channels.delete(t),this.mcpServers.delete(t),this.validationIssues.delete(t)}setIssue(t,e){let s=this.validationIssues.get(t)??[];s.push({path:t,message:e}),this.validationIssues.set(t,s)}parseFile(t,e){return t.startsWith(`${this.getSubfolder("agents")}/`)?this.parseAgent(t,e):t.startsWith(`${this.getSubfolder("skills")}/`)?this.parseSkill(t,e):t.startsWith(`${this.getSubfolder("tasks")}/`)?this.parseTask(t,e):null}parseAgent(t,e){let{frontmatter:s,body:n}=J(e);if(!Wt(s))return this.setIssue(t,"Invalid frontmatter."),null;let a=A(s.name),r=A(s.model)??this.settings.defaultModel;if(!a||!r)return this.setIssue(t,"Agent requires string field `name` and a valid model or default model setting."),null;let o=ue(s.allowed_tools),c=ue(s.blocked_tools);return(o.length>0||c.length>0)&&!this.warnedLegacyPerms.has(a)&&(this.warnedLegacyPerms.add(a),console.warn(`Agent Fleet: "${a}" still uses legacy allowed_tools/blocked_tools in its frontmatter. Permission rules now live in <name>.permissions.json beside the .md. Open Edit and Save to migrate.`)),{filePath:t,name:a,description:A(s.description),model:r,adapter:A(s.adapter)??"claude-code",permissionMode:A(s.permission_mode)??"bypassPermissions",effort:A(s.effort),maxRetries:Re(s.max_retries,1),skills:ue(s.skills),mcpServers:ue(s.mcp_servers),cwd:A(s.cwd),enabled:Be(s.enabled,!0),timeout:Re(s.timeout,300),approvalRequired:ue(s.approval_required),memory:Be(s.memory,!1),memoryMaxEntries:Re(s.memory_max_entries,100),memoryTokenBudget:Ei(s),reflection:Pi(s),autoCompactThreshold:Re(s.auto_compact_threshold,85),tags:ue(s.tags),avatar:A(s.avatar)??"",body:n,contextBody:"",skillsBody:"",env:this.parseEnvMap(s.env),permissionRules:{allow:o,deny:c},isFolder:!1,heartbeatEnabled:!1,heartbeatSchedule:"",heartbeatBody:"",heartbeatNotify:!0,heartbeatChannel:"",heartbeatChannelTarget:""}}parseSkill(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.name);return a?{filePath:t,name:a,description:A(s.description),tags:ue(s.tags),body:n,toolsBody:"",referencesBody:"",examplesBody:"",isFolder:!1}:(this.setIssue(t,"Skill requires string field `name`."),null)}parseTask(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.task_id),r=A(s.agent),o=A(s.type);if(!a||!r||!o)return this.setIssue(t,"Task requires `task_id`, `agent`, and `type`."),null;if(o==="recurring"&&!A(s.schedule))return this.setIssue(t,"Recurring task requires `schedule`."),null;if(o==="once"&&!A(s.run_at))return this.setIssue(t,"One-time task requires `run_at`."),null;let c=A(s.priority),h=c&&["low","medium","high","critical"].includes(c)?c:"medium";return{filePath:t,taskId:a,agent:r,schedule:A(s.schedule),runAt:A(s.run_at),type:o,priority:h,enabled:Be(s.enabled,!0),created:A(s.created)??new Date().toISOString(),lastRun:A(s.last_run),nextRun:A(s.next_run),runCount:Re(s.run_count,0),catchUp:Be(s.catch_up,this.settings.catchUpMissedTasks),effort:A(s.effort),model:A(s.model),channel:A(s.channel),channelTarget:A(s.channel_target),tags:ue(s.tags),body:n}}parseChannelFile(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.name);if(!a)return this.setIssue(t,"Channel requires string field `name`."),null;let r=A(s.type),o=["slack","telegram","discord"];if(!r||!o.includes(r))return this.setIssue(t,`Channel \`${a}\` requires \`type\` to be one of: ${o.join(", ")}.`),null;let c=r,l=A(s.default_agent)??A(s.agent);if(!l)return this.setIssue(t,`Channel \`${a}\` requires \`default_agent\` (or \`agent\`).`),null;let h=ue(s.allowed_agents),d=A(s.credential_ref);if(!d)return this.setIssue(t,`Channel \`${a}\` requires \`credential_ref\` pointing at a configured credential.`),null;let u=Wt(s.transport)?s.transport:{};return{filePath:t,name:a,type:c,defaultAgent:l,allowedAgents:h,enabled:Be(s.enabled,!0),credentialRef:d,allowedUsers:ue(s.allowed_users),perUserSessions:Be(s.per_user_sessions,!0),channelContext:A(s.channel_context)??"",transport:u,tags:ue(s.tags),body:n}}parseMcpServerFile(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.name);if(!a)return this.setIssue(t,"MCP server requires string field `name`."),null;let r=(A(s.transport)??A(s.type)??"").toLowerCase(),o=["stdio","http","sse"];if(!o.includes(r))return this.setIssue(t,`MCP server \`${a}\` requires \`transport\` to be one of: ${o.join(", ")}.`),null;let c=r;if(c==="stdio"){if(!A(s.command))return this.setIssue(t,`MCP server \`${a}\` (stdio) requires a \`command\`.`),null}else if(!A(s.url))return this.setIssue(t,`MCP server \`${a}\` (${c}) requires a \`url\`.`),null;let l=(A(s.auth)??"").toLowerCase(),h=l==="bearer"||l==="oauth"||l==="none"?l:void 0,d;if(Wt(s.oauth)){let u=s.oauth;d={clientId:A(u.client_id)??A(u.clientId),resource:A(u.resource),scopes:ue(u.scopes)}}return{name:a,filePath:t,type:c,enabled:Be(s.enabled,!0),description:A(s.description)??(n.trim()||void 0),source:A(s.source)==="imported"?"imported":"manual",command:A(s.command),args:ue(s.args),env:_i(s.env),envSecretKeys:ue(s.env_secret_keys),url:A(s.url),headers:_i(s.headers),auth:h,oauth:d,status:"disconnected",scope:"user",tools:[],toolDetails:[]}}validateReferences(){let t=new Set;for(let o of this.skills.values())t.has(o.name)&&this.setIssue(o.filePath,`Duplicate skill name \`${o.name}\`.`),t.add(o.name);let e=new Set;for(let o of this.agents.values()){e.has(o.name)&&this.setIssue(o.filePath,`Duplicate agent name \`${o.name}\`.`),e.add(o.name);for(let c of o.skills)t.has(c)||this.setIssue(o.filePath,`Agent references missing skill \`${c}\`.`)}for(let o of this.tasks.values())e.has(o.agent)||this.setIssue(o.filePath,`Task references missing agent \`${o.agent}\`.`);let s=new Set,n=new Map;for(let o of this.agents.values())n.set(o.name,o);let a=this.channelCredentialGetter?.()??this.settings.channelCredentials??{};for(let o of this.channels.values()){s.has(o.name)&&this.setIssue(o.filePath,`Duplicate channel name \`${o.name}\`.`),s.add(o.name);let c=n.get(o.defaultAgent);c?c.approvalRequired.length>0&&this.setIssue(o.filePath,`Channel \`${o.name}\` cannot bind to agent \`${c.name}\` because the agent has \`approval_required\` set (${c.approvalRequired.join(", ")}). Clear the approval list on the agent or pick a different agent.`):this.setIssue(o.filePath,`Channel \`${o.name}\` references missing agent \`${o.defaultAgent}\`.`);let l=a[o.credentialRef];l?l.type!==o.type&&this.setIssue(o.filePath,`Channel \`${o.name}\` is type \`${o.type}\` but credential \`${o.credentialRef}\` is type \`${l.type}\`.`):this.setIssue(o.filePath,`Channel \`${o.name}\` references missing credential \`${o.credentialRef}\`. Add it under Settings \u2192 Channel Credentials.`)}let r=new Set;for(let o of this.mcpServers.values())r.has(o.name)&&this.setIssue(o.filePath??o.name,`Duplicate MCP server name \`${o.name}\`.`),r.add(o.name)}parseEnvMap(t){if(!Wt(t))return{};let e={};for(let[s,n]of Object.entries(t))typeof n=="string"?e[s]=n:(typeof n=="number"||typeof n=="boolean")&&(e[s]=String(n));return e}parseApprovals(t){if(Array.isArray(t))return t.flatMap(e=>{if(!Wt(e)||!A(e.tool))return[];let s=A(e.tool);return s?[{tool:s,command:A(e.command),reason:A(e.reason),status:A(e.status)??"pending",resolvedAt:A(e.resolvedAt),note:A(e.note)}]:[]})}};var Et=require("obsidian"),Ks=class extends Et.Modal{constructor(e,s,n){super(e);this.info=s;this.onConfirm=n}deleteTasks=!0;onOpen(){let{contentEl:e}=this;e.empty(),e.addClass("af-confirm-delete-modal");let s=e.createDiv({cls:"af-delete-header"}),n=s.createSpan({cls:"af-delete-header-icon"});(0,Et.setIcon)(n,"alert-triangle"),s.createEl("h3",{text:`Delete agent "${this.info.agentName}"?`});let a=e.createDiv({cls:"af-delete-summary"});a.createDiv({text:"This action will:"});let r=a.createEl("ul",{cls:"af-delete-impact-list"});r.createEl("li",{text:"Move the agent definition to trash"}),this.info.hasMemory&&r.createEl("li",{text:"Move the agent's memory file to trash"}),this.info.taskCount>0&&r.createEl("li").setText(`${this.info.taskCount} task${this.info.taskCount!==1?"s":""} reference this agent`),this.info.runCount>0&&r.createEl("li",{cls:"af-delete-preserved"}).setText(`${this.info.runCount} run log${this.info.runCount!==1?"s":""} will be preserved`),e.createDiv({cls:"af-delete-note",text:"Files are moved to your system trash and can be recovered."}),this.info.taskCount>0&&new Et.Setting(e).setName("Also delete associated tasks").setDesc(`Delete ${this.info.taskCount} task${this.info.taskCount!==1?"s":""} that reference this agent`).addToggle(d=>{d.setValue(this.deleteTasks),d.onChange(u=>{this.deleteTasks=u})});let o=e.createDiv({cls:"af-delete-actions"}),c=o.createEl("button",{text:"Cancel"});c.onclick=()=>this.close();let l=o.createEl("button",{cls:"af-delete-confirm-btn",text:"Delete Agent"}),h=l.createSpan({cls:"af-delete-btn-icon"});(0,Et.setIcon)(h,"trash-2"),l.onclick=()=>{this.onConfirm(this.deleteTasks),this.close()}}};var F=require("obsidian");var Js=require("obsidian"),Ht=class extends Js.Modal{constructor(e,s){super(e);this.opts=s}onOpen(){let{contentEl:e}=this;e.empty(),e.createEl("h3",{text:this.opts.title});for(let s of this.opts.body.split(`
11711
+ ${e.examplesBody}`))}}else{let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:i,body:o}=J(a);e.description!==void 0&&(i.description=e.description||void 0),e.tags!==void 0&&(i.tags=e.tags.length>0?e.tags:void 0);let c=e.body!==void 0?e.body:o;await this.vault.modify(n,H(i,c))}}async deleteSkill(t){let e=this.getSkillByName(t);if(e)if(e.isFolder){let s=(0,x.normalizePath)(e.filePath.replace(/\/skill\.md$/,"")),n=this.vault.getAbstractFileByPath(s);n instanceof x.TFolder&&await this.app.fileManager.trashFile(n)}else await this.trashFile(e.filePath)}async deleteTask(t){let e=this.getTaskById(t);e&&await this.trashFile(e.filePath)}async updateChannel(t,e){let s=this.getChannelByName(t);if(!s)return;let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:i,body:o}=J(a);e.default_agent!==void 0&&(i.default_agent=e.default_agent,delete i.agent),e.allowed_agents!==void 0&&(i.allowed_agents=e.allowed_agents),e.enabled!==void 0&&(i.enabled=e.enabled),e.credential_ref!==void 0&&(i.credential_ref=e.credential_ref),e.allowed_users!==void 0&&(i.allowed_users=e.allowed_users),e.per_user_sessions!==void 0&&(i.per_user_sessions=e.per_user_sessions),e.channel_context!==void 0&&(i.channel_context=e.channel_context||void 0),e.tags!==void 0&&(i.tags=e.tags),e.type!==void 0&&(i.type=e.type),e.transport!==void 0&&(i.transport=e.transport);let c=e.body!==void 0?e.body:o;await this.vault.modify(n,H(i,c))}async deleteChannel(t){let e=this.getChannelByName(t);if(!e)return;await this.trashFile(e.filePath);let s=(0,x.normalizePath)(`${this.getSubfolder("channels")}/${oe(t)}/sessions`),n=this.vault.getAbstractFileByPath(s);n instanceof x.TFolder&&await this.app.fileManager.trashFile(n)}mcpServerFrontmatter(t){let e={name:t.name,transport:t.type,enabled:t.enabled};return t.source&&(e.source=t.source),t.type==="stdio"?(t.command&&(e.command=t.command),t.args&&t.args.length>0&&(e.args=t.args),t.env&&Object.keys(t.env).length>0&&(e.env=t.env),t.envSecretKeys&&t.envSecretKeys.length>0&&(e.env_secret_keys=t.envSecretKeys)):(t.url&&(e.url=t.url),t.headers&&Object.keys(t.headers).length>0&&(e.headers=t.headers),t.auth&&(e.auth=t.auth),t.oauth&&(t.oauth.clientId||t.oauth.resource||(t.oauth.scopes?.length??0)>0)&&(e.oauth={client_id:t.oauth.clientId||void 0,resource:t.oauth.resource||void 0,scopes:t.oauth.scopes&&t.oauth.scopes.length>0?t.oauth.scopes:void 0})),e}async saveMcpServer(t,e=""){let s=this.mcpServerFrontmatter(t),n=H(s,e.trim()),a=t.filePath?this.vault.getAbstractFileByPath(t.filePath):null;if(a instanceof x.TFile)return await this.vault.modify(a,n),a.path;let i=await this.getAvailablePath(this.getSubfolder("mcp"),oe(t.name));return await this.ensureFolder(this.getSubfolder("mcp")),await this.vault.create(i,n),i}async setMcpServerEnabled(t,e){let s=this.getMcpServerByName(t);if(!s?.filePath)return;let n=this.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof x.TFile))return;let a=await this.vault.cachedRead(n),{frontmatter:i,body:o}=J(a);i.enabled=e,await this.vault.modify(n,H(i,o))}async deleteMcpServer(t){let e=this.getMcpServerByName(t);e?.filePath&&await this.trashFile(e.filePath)}async updateHeartbeat(t,e){let s=this.getAgentByName(t);if(!s||!s.isFolder)return;let n=(0,x.normalizePath)(s.filePath.replace(/\/agent\.md$/,"")),a=(0,x.normalizePath)(`${n}/HEARTBEAT.md`),i=this.vault.getAbstractFileByPath(a);if(i instanceof x.TFile){let o=await this.vault.cachedRead(i),{frontmatter:c,body:l}=J(o);e.enabled!==void 0&&(c.enabled=e.enabled),e.schedule!==void 0&&(c.schedule=e.schedule||void 0),e.notify!==void 0&&(c.notify=e.notify),e.channel!==void 0&&(c.channel=e.channel||void 0),e.channelTarget!==void 0&&(c.channel_target=e.channelTarget||void 0);let h=e.body!==void 0?e.body:l;await this.vault.modify(i,H(c,h))}else{let o={enabled:e.enabled??!1};e.schedule&&(o.schedule=e.schedule),e.notify!==void 0&&(o.notify=e.notify),e.channel&&(o.channel=e.channel),e.channelTarget&&(o.channel_target=e.channelTarget);let c=e.body??"";await this.vault.create(a,H(o,c))}}async deleteAgent(t,e){let s=[],n=this.getAgentByName(t);if(!n)return{trashedFiles:s};if(n.isFolder){let c=(0,x.normalizePath)(n.filePath.replace(/\/agent\.md$/,"")),l=this.vault.getAbstractFileByPath(c);if(l instanceof x.TFolder){let h=[];this.collectMarkdownChildren(l,h);for(let d of h)s.push(d.path);await this.app.fileManager.trashFile(l)}}else await this.trashFile(n.filePath),s.push(n.filePath);let a=this.getMemoryPath(t);this.vault.getAbstractFileByPath(a)&&(await this.trashFile(a),s.push(a));let i=this.getMemoryDir(t),o=this.vault.getAbstractFileByPath(i);if(o instanceof x.TFolder&&(await this.app.fileManager.trashFile(o),s.push(i)),e){let c=this.getTasksForAgent(t);for(let l of c)await this.trashFile(l.filePath),s.push(l.filePath)}return{trashedFiles:s}}async trashFile(t){let e=this.vault.getAbstractFileByPath(t);e&&await this.app.fileManager.trashFile(e)}async ensureFolder(t){if(!this.vault.getAbstractFileByPath(t))try{await this.vault.createFolder(t)}catch(e){if(!(e instanceof Error?e.message:String(e)).includes("Folder already exists"))throw e}}async getAvailablePath(t,e){let s=0;for(;;){let n=s===0?"":`-${s+1}`,a=(0,x.normalizePath)(`${t}/${e}${n}.md`);if(!this.vault.getAbstractFileByPath(a))return a;s+=1}}async createFileIfMissing(t,e){if(!this.vault.getAbstractFileByPath(t))try{await this.vault.create(t,e)}catch(s){if(!(s instanceof Error?s.message:String(s)).includes("File already exists"))throw s}}collectMarkdownChildren(t,e){for(let s of t.children)s instanceof x.TFile&&s.extension==="md"&&e.push(s),s instanceof x.TFolder&&this.collectMarkdownChildren(s,e)}clearStoredFile(t){this.agents.delete(t),this.skills.delete(t),this.tasks.delete(t),this.channels.delete(t),this.mcpServers.delete(t),this.validationIssues.delete(t)}setIssue(t,e){let s=this.validationIssues.get(t)??[];s.push({path:t,message:e}),this.validationIssues.set(t,s)}parseFile(t,e){return t.startsWith(`${this.getSubfolder("agents")}/`)?this.parseAgent(t,e):t.startsWith(`${this.getSubfolder("skills")}/`)?this.parseSkill(t,e):t.startsWith(`${this.getSubfolder("tasks")}/`)?this.parseTask(t,e):null}parseAgent(t,e){let{frontmatter:s,body:n}=J(e);if(!Ht(s))return this.setIssue(t,"Invalid frontmatter."),null;let a=A(s.name),i=A(s.model)??this.settings.defaultModel;if(!a||!i)return this.setIssue(t,"Agent requires string field `name` and a valid model or default model setting."),null;let o=ue(s.allowed_tools),c=ue(s.blocked_tools);return(o.length>0||c.length>0)&&!this.warnedLegacyPerms.has(a)&&(this.warnedLegacyPerms.add(a),console.warn(`Agent Fleet: "${a}" still uses legacy allowed_tools/blocked_tools in its frontmatter. Permission rules now live in <name>.permissions.json beside the .md. Open Edit and Save to migrate.`)),{filePath:t,name:a,description:A(s.description),model:i,adapter:A(s.adapter)??"claude-code",permissionMode:A(s.permission_mode)??"bypassPermissions",effort:A(s.effort),maxRetries:Re(s.max_retries,1),skills:ue(s.skills),mcpServers:ue(s.mcp_servers),cwd:A(s.cwd),enabled:Be(s.enabled,!0),timeout:Re(s.timeout,300),approvalRequired:ue(s.approval_required),memory:Be(s.memory,!1),memoryMaxEntries:Re(s.memory_max_entries,100),memoryTokenBudget:Pr(s),reflection:Rr(s),autoCompactThreshold:Re(s.auto_compact_threshold,85),tags:ue(s.tags),avatar:A(s.avatar)??"",body:n,contextBody:"",skillsBody:"",env:this.parseEnvMap(s.env),permissionRules:{allow:o,deny:c},isFolder:!1,heartbeatEnabled:!1,heartbeatSchedule:"",heartbeatBody:"",heartbeatNotify:!0,heartbeatChannel:"",heartbeatChannelTarget:""}}parseSkill(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.name);return a?{filePath:t,name:a,description:A(s.description),tags:ue(s.tags),body:n,toolsBody:"",referencesBody:"",examplesBody:"",isFolder:!1}:(this.setIssue(t,"Skill requires string field `name`."),null)}parseTask(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.task_id),i=A(s.agent),o=A(s.type);if(!a||!i||!o)return this.setIssue(t,"Task requires `task_id`, `agent`, and `type`."),null;if(o==="recurring"&&!A(s.schedule))return this.setIssue(t,"Recurring task requires `schedule`."),null;if(o==="once"&&!A(s.run_at))return this.setIssue(t,"One-time task requires `run_at`."),null;let c=A(s.priority),h=c&&["low","medium","high","critical"].includes(c)?c:"medium";return{filePath:t,taskId:a,agent:i,schedule:A(s.schedule),runAt:A(s.run_at),type:o,priority:h,enabled:Be(s.enabled,!0),created:A(s.created)??new Date().toISOString(),lastRun:A(s.last_run),nextRun:A(s.next_run),runCount:Re(s.run_count,0),catchUp:Be(s.catch_up,this.settings.catchUpMissedTasks),effort:A(s.effort),model:A(s.model),channel:A(s.channel),channelTarget:A(s.channel_target),tags:ue(s.tags),body:n}}parseChannelFile(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.name);if(!a)return this.setIssue(t,"Channel requires string field `name`."),null;let i=A(s.type),o=["slack","telegram","discord"];if(!i||!o.includes(i))return this.setIssue(t,`Channel \`${a}\` requires \`type\` to be one of: ${o.join(", ")}.`),null;let c=i,l=A(s.default_agent)??A(s.agent);if(!l)return this.setIssue(t,`Channel \`${a}\` requires \`default_agent\` (or \`agent\`).`),null;let h=ue(s.allowed_agents),d=A(s.credential_ref);if(!d)return this.setIssue(t,`Channel \`${a}\` requires \`credential_ref\` pointing at a configured credential.`),null;let u=Ht(s.transport)?s.transport:{};return{filePath:t,name:a,type:c,defaultAgent:l,allowedAgents:h,enabled:Be(s.enabled,!0),credentialRef:d,allowedUsers:ue(s.allowed_users),perUserSessions:Be(s.per_user_sessions,!0),channelContext:A(s.channel_context)??"",transport:u,tags:ue(s.tags),body:n}}parseMcpServerFile(t,e){let{frontmatter:s,body:n}=J(e),a=A(s.name);if(!a)return this.setIssue(t,"MCP server requires string field `name`."),null;let i=(A(s.transport)??A(s.type)??"").toLowerCase(),o=["stdio","http","sse"];if(!o.includes(i))return this.setIssue(t,`MCP server \`${a}\` requires \`transport\` to be one of: ${o.join(", ")}.`),null;let c=i;if(c==="stdio"){if(!A(s.command))return this.setIssue(t,`MCP server \`${a}\` (stdio) requires a \`command\`.`),null}else if(!A(s.url))return this.setIssue(t,`MCP server \`${a}\` (${c}) requires a \`url\`.`),null;let l=(A(s.auth)??"").toLowerCase(),h=l==="bearer"||l==="oauth"||l==="none"?l:void 0,d;if(Ht(s.oauth)){let u=s.oauth;d={clientId:A(u.client_id)??A(u.clientId),resource:A(u.resource),scopes:ue(u.scopes)}}return{name:a,filePath:t,type:c,enabled:Be(s.enabled,!0),description:A(s.description)??(n.trim()||void 0),source:A(s.source)==="imported"?"imported":"manual",command:A(s.command),args:ue(s.args),env:Ar(s.env),envSecretKeys:ue(s.env_secret_keys),url:A(s.url),headers:Ar(s.headers),auth:h,oauth:d,status:"disconnected",scope:"user",tools:[],toolDetails:[]}}validateReferences(){let t=new Set;for(let o of this.skills.values())t.has(o.name)&&this.setIssue(o.filePath,`Duplicate skill name \`${o.name}\`.`),t.add(o.name);let e=new Set;for(let o of this.agents.values()){e.has(o.name)&&this.setIssue(o.filePath,`Duplicate agent name \`${o.name}\`.`),e.add(o.name);for(let c of o.skills)t.has(c)||this.setIssue(o.filePath,`Agent references missing skill \`${c}\`.`)}for(let o of this.tasks.values())e.has(o.agent)||this.setIssue(o.filePath,`Task references missing agent \`${o.agent}\`.`);let s=new Set,n=new Map;for(let o of this.agents.values())n.set(o.name,o);let a=this.channelCredentialGetter?.()??this.settings.channelCredentials??{};for(let o of this.channels.values()){s.has(o.name)&&this.setIssue(o.filePath,`Duplicate channel name \`${o.name}\`.`),s.add(o.name);let c=n.get(o.defaultAgent);c?c.approvalRequired.length>0&&this.setIssue(o.filePath,`Channel \`${o.name}\` cannot bind to agent \`${c.name}\` because the agent has \`approval_required\` set (${c.approvalRequired.join(", ")}). Clear the approval list on the agent or pick a different agent.`):this.setIssue(o.filePath,`Channel \`${o.name}\` references missing agent \`${o.defaultAgent}\`.`);let l=a[o.credentialRef];l?l.type!==o.type&&this.setIssue(o.filePath,`Channel \`${o.name}\` is type \`${o.type}\` but credential \`${o.credentialRef}\` is type \`${l.type}\`.`):this.setIssue(o.filePath,`Channel \`${o.name}\` references missing credential \`${o.credentialRef}\`. Add it under Settings \u2192 Channel Credentials.`)}let i=new Set;for(let o of this.mcpServers.values())i.has(o.name)&&this.setIssue(o.filePath??o.name,`Duplicate MCP server name \`${o.name}\`.`),i.add(o.name)}parseEnvMap(t){if(!Ht(t))return{};let e={};for(let[s,n]of Object.entries(t))typeof n=="string"?e[s]=n:(typeof n=="number"||typeof n=="boolean")&&(e[s]=String(n));return e}parseApprovals(t){if(Array.isArray(t))return t.flatMap(e=>{if(!Ht(e)||!A(e.tool))return[];let s=A(e.tool);return s?[{tool:s,command:A(e.command),reason:A(e.reason),status:A(e.status)??"pending",resolvedAt:A(e.resolvedAt),note:A(e.note)}]:[]})}};var Pt=require("obsidian"),Ks=class extends Pt.Modal{constructor(e,s,n){super(e);this.info=s;this.onConfirm=n}deleteTasks=!0;onOpen(){let{contentEl:e}=this;e.empty(),e.addClass("af-confirm-delete-modal");let s=e.createDiv({cls:"af-delete-header"}),n=s.createSpan({cls:"af-delete-header-icon"});(0,Pt.setIcon)(n,"alert-triangle"),s.createEl("h3",{text:`Delete agent "${this.info.agentName}"?`});let a=e.createDiv({cls:"af-delete-summary"});a.createDiv({text:"This action will:"});let i=a.createEl("ul",{cls:"af-delete-impact-list"});i.createEl("li",{text:"Move the agent definition to trash"}),this.info.hasMemory&&i.createEl("li",{text:"Move the agent's memory file to trash"}),this.info.taskCount>0&&i.createEl("li").setText(`${this.info.taskCount} task${this.info.taskCount!==1?"s":""} reference this agent`),this.info.runCount>0&&i.createEl("li",{cls:"af-delete-preserved"}).setText(`${this.info.runCount} run log${this.info.runCount!==1?"s":""} will be preserved`),e.createDiv({cls:"af-delete-note",text:"Files are moved to your system trash and can be recovered."}),this.info.taskCount>0&&new Pt.Setting(e).setName("Also delete associated tasks").setDesc(`Delete ${this.info.taskCount} task${this.info.taskCount!==1?"s":""} that reference this agent`).addToggle(d=>{d.setValue(this.deleteTasks),d.onChange(u=>{this.deleteTasks=u})});let o=e.createDiv({cls:"af-delete-actions"}),c=o.createEl("button",{text:"Cancel"});c.onclick=()=>this.close();let l=o.createEl("button",{cls:"af-delete-confirm-btn",text:"Delete Agent"}),h=l.createSpan({cls:"af-delete-btn-icon"});(0,Pt.setIcon)(h,"trash-2"),l.onclick=()=>{this.onConfirm(this.deleteTasks),this.close()}}};var F=require("obsidian");var Js=require("obsidian"),qt=class extends Js.Modal{constructor(e,s){super(e);this.opts=s}onOpen(){let{contentEl:e}=this;e.empty(),e.createEl("h3",{text:this.opts.title});for(let s of this.opts.body.split(`
11801
11712
 
11802
- `))s.trim()&&e.createEl("p",{text:s});new Js.Setting(e).addButton(s=>s.setButtonText("Cancel").onClick(()=>this.close())).addButton(s=>{s.setButtonText(this.opts.confirmText??"Confirm").onClick(()=>{this.close(),this.opts.onConfirm()}),this.opts.danger&&s.setWarning()})}onClose(){this.contentEl.empty()}};var Qs=[{value:"opus",label:"opus \u2014 latest Opus"},{value:"sonnet",label:"sonnet \u2014 latest Sonnet"},{value:"haiku",label:"haiku \u2014 latest Haiku"},{value:"opusplan",label:"opusplan \u2014 plan-mode Opus"}],Zs=[{value:"gpt-5.5",label:"gpt-5.5 \u2014 frontier"},{value:"gpt-5.4",label:"gpt-5.4"},{value:"gpt-5.4-mini",label:"gpt-5.4-mini \u2014 fast"},{value:"gpt-5.3-codex",label:"gpt-5.3-codex"}],Xs="__custom__";function Mi(i){let t=(i??"").trim().toLowerCase();return t==="codex"||t==="openai-codex"}function Li(i){return Mi(i)?Zs:Qs}function kl(i,t){let e=i.trim();return!e||e==="default"||e==="subscription"?"inherit":Li(t).some(s=>s.value===e)?"alias":"custom"}function Pt(i,t){i.empty(),i.addClass("af-model-picker");let e=Mi(t.adapter),s=Li(t.adapter),n=kl(t.value,t.adapter),a=i.createEl("select",{cls:"af-form-select af-mp-select"}),r=t.allowInherit?t.inheritPlaceholder??"Inherit from agent":e?"Default (let Codex pick)":"Default (let Claude Code pick)";a.createEl("option",{text:r,attr:{value:""}});let o=a.createEl("optgroup",{attr:{label:e?"Codex models":"Aliases (any backend)"}});for(let l of s)o.createEl("option",{text:l.label,attr:{value:l.value}});a.createEl("option",{text:"Custom\u2026",attr:{value:Xs}});let c=i.createEl("input",{cls:"af-form-input af-mp-custom-input",attr:{type:"text",placeholder:e?"e.g. gpt-5.5 \xB7 gpt-5.4-mini":"e.g. claude-opus-4-7 \xB7 us.anthropic.claude-opus-4-7 \xB7 claude-opus-4-7@20251101",spellcheck:"false"}});n==="inherit"?(a.value="",c.value="",c.setCssStyles({display:"none"})):n==="alias"?(a.value=t.value.trim(),c.value="",c.setCssStyles({display:"none"})):(a.value=Xs,c.value=t.value.trim(),c.setCssStyles({display:""})),a.addEventListener("change",()=>{a.value===Xs?(c.setCssStyles({display:""}),c.focus(),t.onChange(c.value.trim())):(c.setCssStyles({display:"none"}),t.onChange(a.value))}),c.addEventListener("input",()=>{a.value===Xs&&t.onChange(c.value.trim())})}var fe=require("obsidian");function xl(){let i=new Date,t=i.getFullYear(),e=String(i.getMonth()+1).padStart(2,"0"),s=String(i.getDate()).padStart(2,"0");return`${t}-${e}-${s}`}var Sl="0 3 * * *",Cl="0 9 * * 0";function na(){return{scopeSlug:"",scopeRoot:"",inboxPath:"_sources/inbox",archivePath:"_sources/archive",topicsRoot:"_topics",indexPath:"index.md",logPath:"log.md",watchedFolders:[],excludePatterns:[],watchedSince:xl(),heartbeatChannel:"",fileSubstantiveAnswers:!0,obsidianUrlScheme:!0,maxTokensPerIngest:6e4,maxTokensPerRefresh:3e4,indexSplitThreshold:100,dedupSimilarityThreshold:.82,summaryStaleDays:30,failedPath:"_sources/failed",stateFile:".wiki-keeper-state.json"}}var Ni=na();function aa(i){let t=oe(i).trim();return t?`wiki-keeper-${t}`:"wiki-keeper"}function Tl(i,t){let e=t||"the whole vault",s={name:i,description:`Cultivates ${e} as a self-maintaining wiki. Ingests new sources, extracts durable claims from watched folders, answers questions with citations, and periodically lints.`,avatar:"library",enabled:!0,skills:["wiki-ingest","wiki-query","wiki-lint","wiki-refresh"],tags:t?["wiki-keeper",`scope:${oe(t)}`]:["wiki-keeper"]},n=`You are the Wiki Keeper for the \`${e}\` scope. Maintain it as a persistent, interlinked knowledge base (Karpathy's "LLM wiki" pattern).
11713
+ `))s.trim()&&e.createEl("p",{text:s});new Js.Setting(e).addButton(s=>s.setButtonText("Cancel").onClick(()=>this.close())).addButton(s=>{s.setButtonText(this.opts.confirmText??"Confirm").onClick(()=>{this.close(),this.opts.onConfirm()}),this.opts.danger&&s.setWarning()})}onClose(){this.contentEl.empty()}};var Qs=[{value:"opus",label:"opus \u2014 latest Opus"},{value:"sonnet",label:"sonnet \u2014 latest Sonnet"},{value:"haiku",label:"haiku \u2014 latest Haiku"},{value:"opusplan",label:"opusplan \u2014 plan-mode Opus"}],Zs=[{value:"gpt-5.5",label:"gpt-5.5 \u2014 frontier"},{value:"gpt-5.4",label:"gpt-5.4"},{value:"gpt-5.4-mini",label:"gpt-5.4-mini \u2014 fast"},{value:"gpt-5.3-codex",label:"gpt-5.3-codex"}],Xs="__custom__";function Lr(r){let t=(r??"").trim().toLowerCase();return t==="codex"||t==="openai-codex"}function Fr(r){return Lr(r)?Zs:Qs}function xl(r,t){let e=r.trim();return!e||e==="default"||e==="subscription"?"inherit":Fr(t).some(s=>s.value===e)?"alias":"custom"}function Rt(r,t){r.empty(),r.addClass("af-model-picker");let e=Lr(t.adapter),s=Fr(t.adapter),n=xl(t.value,t.adapter),a=r.createEl("select",{cls:"af-form-select af-mp-select"}),i=t.allowInherit?t.inheritPlaceholder??"Inherit from agent":e?"Default (let Codex pick)":"Default (let Claude Code pick)";a.createEl("option",{text:i,attr:{value:""}});let o=a.createEl("optgroup",{attr:{label:e?"Codex models":"Aliases (any backend)"}});for(let l of s)o.createEl("option",{text:l.label,attr:{value:l.value}});a.createEl("option",{text:"Custom\u2026",attr:{value:Xs}});let c=r.createEl("input",{cls:"af-form-input af-mp-custom-input",attr:{type:"text",placeholder:e?"e.g. gpt-5.5 \xB7 gpt-5.4-mini":"e.g. claude-opus-4-7 \xB7 us.anthropic.claude-opus-4-7 \xB7 claude-opus-4-7@20251101",spellcheck:"false"}});n==="inherit"?(a.value="",c.value="",c.setCssStyles({display:"none"})):n==="alias"?(a.value=t.value.trim(),c.value="",c.setCssStyles({display:"none"})):(a.value=Xs,c.value=t.value.trim(),c.setCssStyles({display:""})),a.addEventListener("change",()=>{a.value===Xs?(c.setCssStyles({display:""}),c.focus(),t.onChange(c.value.trim())):(c.setCssStyles({display:"none"}),t.onChange(a.value))}),c.addEventListener("input",()=>{a.value===Xs&&t.onChange(c.value.trim())})}var fe=require("obsidian");function Sl(){let r=new Date,t=r.getFullYear(),e=String(r.getMonth()+1).padStart(2,"0"),s=String(r.getDate()).padStart(2,"0");return`${t}-${e}-${s}`}var Cl="0 3 * * *",Tl="0 9 * * 0";function na(){return{scopeSlug:"",scopeRoot:"",inboxPath:"_sources/inbox",archivePath:"_sources/archive",topicsRoot:"_topics",indexPath:"index.md",logPath:"log.md",watchedFolders:[],excludePatterns:[],watchedSince:Sl(),heartbeatChannel:"",fileSubstantiveAnswers:!0,obsidianUrlScheme:!0,maxTokensPerIngest:6e4,maxTokensPerRefresh:3e4,indexSplitThreshold:100,dedupSimilarityThreshold:.82,summaryStaleDays:30,failedPath:"_sources/failed",stateFile:".wiki-keeper-state.json"}}var Br=na();function aa(r){let t=oe(r).trim();return t?`wiki-keeper-${t}`:"wiki-keeper"}function _l(r,t){let e=t||"the whole vault",s={name:r,description:`Cultivates ${e} as a self-maintaining wiki. Ingests new sources, extracts durable claims from watched folders, answers questions with citations, and periodically lints.`,avatar:"library",enabled:!0,skills:["wiki-ingest","wiki-query","wiki-lint","wiki-refresh"],tags:t?["wiki-keeper",`scope:${oe(t)}`]:["wiki-keeper"]},n=`You are the Wiki Keeper for the \`${e}\` scope. Maintain it as a persistent, interlinked knowledge base (Karpathy's "LLM wiki" pattern).
11803
11714
 
11804
11715
  ## Scope isolation
11805
11716
 
@@ -11841,7 +11752,7 @@ Use \`memory\` for procedural learning ("user prefers concept pages under sub-fo
11841
11752
  - Never write outside the scope root.
11842
11753
  - Never edit \`## Claims\` history (append only).
11843
11754
  - Never edit \`## Summary\` blocks by hand \u2014 \`wiki-refresh\` owns them.
11844
- `;return H(s,n)}function _l(i){let t={model:"opus",adapter:"claude-code",timeout:900,max_retries:1,cwd:"",permission_mode:"acceptEdits",approval_required:["Write"],memory:!0,memory_max_entries:200,wiki_keeper:{scope_root:i.scopeRoot,inbox_path:i.inboxPath,archive_path:i.archivePath,failed_path:i.failedPath,topics_root:i.topicsRoot,index_path:i.indexPath,log_path:i.logPath,watched_folders:i.watchedFolders,exclude_patterns:i.excludePatterns,watched_since:i.watchedSince,file_substantive_answers:i.fileSubstantiveAnswers,obsidian_url_scheme:i.obsidianUrlScheme,max_tokens_per_ingest:i.maxTokensPerIngest,max_tokens_per_refresh:i.maxTokensPerRefresh,index_split_threshold:i.indexSplitThreshold,dedup_similarity_threshold:i.dedupSimilarityThreshold,summary_stale_days:i.summaryStaleDays,state_file:i.stateFile}};return H(t,"")}function Al(i){let t={enabled:!0,schedule:Sl,notify:!0};return i.heartbeatChannel&&(t.channel=i.heartbeatChannel),H(t,`Run wiki-ingest in both modes:
11755
+ `;return H(s,n)}function Al(r){let t={model:"opus",adapter:"claude-code",timeout:900,max_retries:1,cwd:"",permission_mode:"acceptEdits",approval_required:["Write"],memory:!0,memory_max_entries:200,wiki_keeper:{scope_root:r.scopeRoot,inbox_path:r.inboxPath,archive_path:r.archivePath,failed_path:r.failedPath,topics_root:r.topicsRoot,index_path:r.indexPath,log_path:r.logPath,watched_folders:r.watchedFolders,exclude_patterns:r.excludePatterns,watched_since:r.watchedSince,file_substantive_answers:r.fileSubstantiveAnswers,obsidian_url_scheme:r.obsidianUrlScheme,max_tokens_per_ingest:r.maxTokensPerIngest,max_tokens_per_refresh:r.maxTokensPerRefresh,index_split_threshold:r.indexSplitThreshold,dedup_similarity_threshold:r.dedupSimilarityThreshold,summary_stale_days:r.summaryStaleDays,state_file:r.stateFile}};return H(t,"")}function El(r){let t={enabled:!0,schedule:Cl,notify:!0};return r.heartbeatChannel&&(t.channel=r.heartbeatChannel),H(t,`Run wiki-ingest in both modes:
11845
11756
 
11846
11757
  1. Drain every unprocessed file in the configured inbox (inbox mode).
11847
11758
  2. Diff watched folders against the state file; process changed or new files (watched mode).
@@ -11849,37 +11760,37 @@ Use \`memory\` for procedural learning ("user prefers concept pages under sub-fo
11849
11760
  Lint runs on its own schedule via the sibling \`*-lint\` task.
11850
11761
 
11851
11762
  Change the schedule by editing this file's \`schedule:\` frontmatter directly, or via the agent editor in the dashboard.
11852
- `)}function Bi(i){let e={task_id:`${i}-lint`,agent:i,type:"recurring",schedule:Cl,priority:"low",enabled:!0,created:new Date().toISOString(),run_count:0,catch_up:!1,tags:["wiki-keeper","lint"]};return H(e,"Run the `wiki-lint` skill on the current scope.\n\nWrite the report as a new `## Lint YYYY-MM-DD` section inside the fenced block in log.md.\n")}function ia(i,t){return(0,fe.normalizePath)(`${i}/tasks/${t}-lint.md`)}var El="# Wiki Schema\n\n## Page types\n\n- **Entity pages** \u2014 people, orgs, products. Frontmatter: `type: entity`.\n- **Concept pages** \u2014 ideas, techniques, patterns. `type: concept`.\n- **Event pages** \u2014 meetings, releases, decisions. `type: event, date: YYYY-MM-DD`.\n- **Summary pages** (inbox-mode only) \u2014 one per ingested inbox source. `type: summary, source: <filename>`.\n- **Synthesis pages** \u2014 filed answers from `wiki-query` substantive responses. `type: synthesis, question: <q>, refreshed: <ISO>`. Live under `_topics/syntheses/`.\n\nTopic pages of type entity/concept/event SHOULD also carry:\n- `summary_refreshed: <ISO>` \u2014 last time `wiki-refresh` regenerated this page's summary block. Empty string on creation.\n- `claims_at_refresh: <int>` \u2014 how many claims existed at last refresh. Drives the staleness threshold.\n\n## Page anatomy\n\nEvery entity/concept/event page has this structure:\n\n```markdown\n---\ntype: entity\nname: Vendor X\nsummary_refreshed: 2026-04-26\nclaims_at_refresh: 14\n---\n\n<!-- wiki-keeper:summary:begin -->\n## Summary\n\n- 3-7 bullet synthesis maintained by wiki-refresh.\n- Forward-links to other topics with [[wikilinks]].\n<!-- wiki-keeper:summary:end -->\n\n## Claims\n\n- 2026-04-18 [meeting]: from [[meetings/2026-04-18-vendor-sync|vendor-sync]]: Vendor X raised prices 15%\n- 2026-04-12 [doc]: from [[summaries/2026-04-12-renewal-brief|renewal-brief]]: contract auto-renews 2026-12-01\n\n## Contradictions\n\n(only present when contradictions exist)\n```\n\nThe fenced summary block is auto-managed by `wiki-refresh`. The `## Claims` section is append-only history written by `wiki-ingest` and `wiki-query` compounding. `## Contradictions` is created on demand. **Anything outside these structures is user-authored** and must be preserved.\n\n## Source-type tags\n\nEvery dated bullet in `## Claims` carries a tag indicating the source type:\n\n- `[doc]` \u2014 PDF, DOCX, XLSX, or other document\n- `[meeting]` \u2014 meeting note or transcript\n- `[email]` \u2014 email forward\n- `[note]` \u2014 markdown daily note or in-vault note\n- `[web]` \u2014 web clipping or URL drop\n- `[synthesis]` \u2014 bullet emitted by `wiki-query` compounding (links to a synthesis page)\n- `[other]` \u2014 fallback\n\nTags help `wiki-query` weight evidence quality and help `wiki-lint` identify drift.\n\n## Naming\n\n- Slug-case filenames: `vendor-x.md`, not `Vendor X.md`.\n- Group by type under `_topics/<type>/` when there are >5 pages of a type.\n\n## Links\n\n- Every entity/concept page MUST have \u22651 inbound link from `index.md` (or a sub-MOC) or a sibling.\n- Summary pages MUST forward-link to every entity/concept they mention.\n- Watched-mode extractions append dated entries to topic pages; forward-link to the source file path so readers can find the raw note.\n- Synthesis pages forward-link to every cited topic and earn their place that way (lint exempts them from orphan flagging if they have any outbound links to topics).\n\n## Index split\n\nWhen the topic count exceeds the `index_split_threshold` config value (default 100) or any single type exceeds 30 pages, `index.md` becomes a hub of hubs and per-type sub-MOCs live at `<scope>/index/<type>.md`. After split, new topic-page entries go into the matching sub-MOC, not the root index.\n\n## Conflict resolution\n\n- New claim contradicts existing? Add a `## Contradictions` section with a dated entry. Do NOT overwrite.\n- Flag in `log.md` for user review.\n- `wiki-refresh` reflects unresolved contradictions in the summary so query-time readers see them.\n\n## Failed sources\n\nFiles that fail ingest 3 times in a row are quarantined to `<failed_path>/` (default `_sources/failed/`) with a sidecar `.error.md`. Lint surfaces the quarantine count weekly.\n",Pl=["Read","Write","Edit","Glob","Grep","Bash(mv *)","Bash(mkdir *)"],Rl=["Bash(rm -rf *)","Bash(git push *)","Bash(rm -rf /*)","Bash(mv * /*)","Bash(cp -r * /*)"];async function Ui(i,t){let e=(0,fe.normalizePath)(`${t}/permissions.json`),s=i.getAbstractFileByPath(e),n={allow:[],deny:[]};if(s instanceof fe.TFile)try{let o=await i.cachedRead(s),c=JSON.parse(o),l=Array.isArray(c.allow)?c.allow.filter(d=>typeof d=="string"):[],h=Array.isArray(c.deny)?c.deny.filter(d=>typeof d=="string"):[];n={allow:l,deny:h}}catch{}let a={allow:Fi(n.allow,Pl),deny:Fi(n.deny,Rl)},r=JSON.stringify(a,null,2)+`
11853
- `;return s instanceof fe.TFile?await i.modify(s,r):await i.create(e,r),a}function Fi(i,t){let e=new Set,s=[];for(let n of i)e.has(n)||(e.add(n),s.push(n));for(let n of t)e.has(n)||(e.add(n),s.push(n));return s}async function $i(i,t,e){let s=aa(e.scopeSlug||e.scopeRoot),n=(0,fe.normalizePath)(`${t}/agents/${s}`);if(await qt(i,n),i.getAbstractFileByPath((0,fe.normalizePath)(`${n}/agent.md`)))throw new Error(`Wiki Keeper agent already exists at ${n}. Delete it first or choose a different scope slug.`);await i.create((0,fe.normalizePath)(`${n}/agent.md`),Tl(s,e.scopeRoot)),await i.create((0,fe.normalizePath)(`${n}/config.md`),_l(e)),await i.create((0,fe.normalizePath)(`${n}/HEARTBEAT.md`),Al(e)),await i.create((0,fe.normalizePath)(`${n}/CONTEXT.md`),El),await Ui(i,n);let r=ia(t,s);i.getAbstractFileByPath(r)||(await qt(i,(0,fe.normalizePath)(`${t}/tasks`)),await i.create(r,Bi(s)));let o=e.scopeRoot.trim(),c=o?`${o}/`:"";return await qt(i,(0,fe.normalizePath)(`${c}${e.inboxPath}`)),await qt(i,(0,fe.normalizePath)(`${c}${e.topicsRoot}`)),await Oi(i,(0,fe.normalizePath)(`${c}${e.indexPath}`),`# Index
11763
+ `)}function Ur(r){let e={task_id:`${r}-lint`,agent:r,type:"recurring",schedule:Tl,priority:"low",enabled:!0,created:new Date().toISOString(),run_count:0,catch_up:!1,tags:["wiki-keeper","lint"]};return H(e,"Run the `wiki-lint` skill on the current scope.\n\nWrite the report as a new `## Lint YYYY-MM-DD` section inside the fenced block in log.md.\n")}function ra(r,t){return(0,fe.normalizePath)(`${r}/tasks/${t}-lint.md`)}var Pl="# Wiki Schema\n\n## Page types\n\n- **Entity pages** \u2014 people, orgs, products. Frontmatter: `type: entity`.\n- **Concept pages** \u2014 ideas, techniques, patterns. `type: concept`.\n- **Event pages** \u2014 meetings, releases, decisions. `type: event, date: YYYY-MM-DD`.\n- **Summary pages** (inbox-mode only) \u2014 one per ingested inbox source. `type: summary, source: <filename>`.\n- **Synthesis pages** \u2014 filed answers from `wiki-query` substantive responses. `type: synthesis, question: <q>, refreshed: <ISO>`. Live under `_topics/syntheses/`.\n\nTopic pages of type entity/concept/event SHOULD also carry:\n- `summary_refreshed: <ISO>` \u2014 last time `wiki-refresh` regenerated this page's summary block. Empty string on creation.\n- `claims_at_refresh: <int>` \u2014 how many claims existed at last refresh. Drives the staleness threshold.\n\n## Page anatomy\n\nEvery entity/concept/event page has this structure:\n\n```markdown\n---\ntype: entity\nname: Vendor X\nsummary_refreshed: 2026-04-26\nclaims_at_refresh: 14\n---\n\n<!-- wiki-keeper:summary:begin -->\n## Summary\n\n- 3-7 bullet synthesis maintained by wiki-refresh.\n- Forward-links to other topics with [[wikilinks]].\n<!-- wiki-keeper:summary:end -->\n\n## Claims\n\n- 2026-04-18 [meeting]: from [[meetings/2026-04-18-vendor-sync|vendor-sync]]: Vendor X raised prices 15%\n- 2026-04-12 [doc]: from [[summaries/2026-04-12-renewal-brief|renewal-brief]]: contract auto-renews 2026-12-01\n\n## Contradictions\n\n(only present when contradictions exist)\n```\n\nThe fenced summary block is auto-managed by `wiki-refresh`. The `## Claims` section is append-only history written by `wiki-ingest` and `wiki-query` compounding. `## Contradictions` is created on demand. **Anything outside these structures is user-authored** and must be preserved.\n\n## Source-type tags\n\nEvery dated bullet in `## Claims` carries a tag indicating the source type:\n\n- `[doc]` \u2014 PDF, DOCX, XLSX, or other document\n- `[meeting]` \u2014 meeting note or transcript\n- `[email]` \u2014 email forward\n- `[note]` \u2014 markdown daily note or in-vault note\n- `[web]` \u2014 web clipping or URL drop\n- `[synthesis]` \u2014 bullet emitted by `wiki-query` compounding (links to a synthesis page)\n- `[other]` \u2014 fallback\n\nTags help `wiki-query` weight evidence quality and help `wiki-lint` identify drift.\n\n## Naming\n\n- Slug-case filenames: `vendor-x.md`, not `Vendor X.md`.\n- Group by type under `_topics/<type>/` when there are >5 pages of a type.\n\n## Links\n\n- Every entity/concept page MUST have \u22651 inbound link from `index.md` (or a sub-MOC) or a sibling.\n- Summary pages MUST forward-link to every entity/concept they mention.\n- Watched-mode extractions append dated entries to topic pages; forward-link to the source file path so readers can find the raw note.\n- Synthesis pages forward-link to every cited topic and earn their place that way (lint exempts them from orphan flagging if they have any outbound links to topics).\n\n## Index split\n\nWhen the topic count exceeds the `index_split_threshold` config value (default 100) or any single type exceeds 30 pages, `index.md` becomes a hub of hubs and per-type sub-MOCs live at `<scope>/index/<type>.md`. After split, new topic-page entries go into the matching sub-MOC, not the root index.\n\n## Conflict resolution\n\n- New claim contradicts existing? Add a `## Contradictions` section with a dated entry. Do NOT overwrite.\n- Flag in `log.md` for user review.\n- `wiki-refresh` reflects unresolved contradictions in the summary so query-time readers see them.\n\n## Failed sources\n\nFiles that fail ingest 3 times in a row are quarantined to `<failed_path>/` (default `_sources/failed/`) with a sidecar `.error.md`. Lint surfaces the quarantine count weekly.\n",Rl=["Read","Write","Edit","Glob","Grep","Bash(mv *)","Bash(mkdir *)"],Dl=["Bash(rm -rf *)","Bash(git push *)","Bash(rm -rf /*)","Bash(mv * /*)","Bash(cp -r * /*)"];async function $r(r,t){let e=(0,fe.normalizePath)(`${t}/permissions.json`),s=r.getAbstractFileByPath(e),n={allow:[],deny:[]};if(s instanceof fe.TFile)try{let o=await r.cachedRead(s),c=JSON.parse(o),l=Array.isArray(c.allow)?c.allow.filter(d=>typeof d=="string"):[],h=Array.isArray(c.deny)?c.deny.filter(d=>typeof d=="string"):[];n={allow:l,deny:h}}catch{}let a={allow:Or(n.allow,Rl),deny:Or(n.deny,Dl)},i=JSON.stringify(a,null,2)+`
11764
+ `;return s instanceof fe.TFile?await r.modify(s,i):await r.create(e,i),a}function Or(r,t){let e=new Set,s=[];for(let n of r)e.has(n)||(e.add(n),s.push(n));for(let n of t)e.has(n)||(e.add(n),s.push(n));return s}async function jr(r,t,e){let s=aa(e.scopeSlug||e.scopeRoot),n=(0,fe.normalizePath)(`${t}/agents/${s}`);if(await zt(r,n),r.getAbstractFileByPath((0,fe.normalizePath)(`${n}/agent.md`)))throw new Error(`Wiki Keeper agent already exists at ${n}. Delete it first or choose a different scope slug.`);await r.create((0,fe.normalizePath)(`${n}/agent.md`),_l(s,e.scopeRoot)),await r.create((0,fe.normalizePath)(`${n}/config.md`),Al(e)),await r.create((0,fe.normalizePath)(`${n}/HEARTBEAT.md`),El(e)),await r.create((0,fe.normalizePath)(`${n}/CONTEXT.md`),Pl),await $r(r,n);let i=ra(t,s);r.getAbstractFileByPath(i)||(await zt(r,(0,fe.normalizePath)(`${t}/tasks`)),await r.create(i,Ur(s)));let o=e.scopeRoot.trim(),c=o?`${o}/`:"";return await zt(r,(0,fe.normalizePath)(`${c}${e.inboxPath}`)),await zt(r,(0,fe.normalizePath)(`${c}${e.topicsRoot}`)),await Nr(r,(0,fe.normalizePath)(`${c}${e.indexPath}`),`# Index
11854
11765
 
11855
11766
  <!-- wiki-keeper:begin -->
11856
11767
  <!-- wiki-keeper:end -->
11857
- `),await Oi(i,(0,fe.normalizePath)(`${c}${e.logPath}`),`# Log
11768
+ `),await Nr(r,(0,fe.normalizePath)(`${c}${e.logPath}`),`# Log
11858
11769
 
11859
11770
  <!-- wiki-keeper:begin -->
11860
11771
  <!-- wiki-keeper:end -->
11861
- `),{path:n,name:s}}async function qt(i,t){if(i.getAbstractFileByPath(t))return;let e=t.split("/"),s="";for(let n of e)if(s=s?`${s}/${n}`:n,!i.getAbstractFileByPath(s))try{await i.createFolder(s)}catch(a){if(!(a instanceof Error?a.message:String(a)).includes("already exists"))throw a}}async function Oi(i,t,e){if(i.getAbstractFileByPath(t))return;let s=t.replace(/\/[^/]+$/,"");s&&s!==t&&await qt(i,s),await i.create(t,e)}async function ji(i,t,e,s){let n=(0,fe.normalizePath)(`${t}/agents/${e}`),a=(0,fe.normalizePath)(`${n}/config.md`),r=i.getAbstractFileByPath(a);if(!(r instanceof fe.TFile))throw new Error(`Config file not found for ${e} at ${a}`);let o=await i.cachedRead(r),{frontmatter:c,body:l}=J(o),h=c.wiki_keeper??{};h.watched_folders=s.watchedFolders,h.exclude_patterns=s.excludePatterns,s.watchedSince?h.watched_since=s.watchedSince:delete h.watched_since,delete h.ingest_schedule,delete h.lint_schedule,delete h.lint_day,h.file_substantive_answers=s.fileSubstantiveAnswers,h.obsidian_url_scheme=s.obsidianUrlScheme,h.max_tokens_per_ingest=s.maxTokensPerIngest,h.max_tokens_per_refresh=s.maxTokensPerRefresh,h.index_split_threshold=s.indexSplitThreshold,h.dedup_similarity_threshold=s.dedupSimilarityThreshold,h.summary_stale_days=s.summaryStaleDays,(typeof h.failed_path!="string"||!h.failed_path)&&(h.failed_path="_sources/failed"),c.wiki_keeper=h,delete c.allowed_tools,delete c.blocked_tools,await i.modify(r,H(c,l)),await Ui(i,n);let d=(0,fe.normalizePath)(`${n}/HEARTBEAT.md`),u=i.getAbstractFileByPath(d);if(u instanceof fe.TFile){let f=await i.cachedRead(u),{frontmatter:g,body:y}=J(f);s.heartbeatChannel?g.channel=s.heartbeatChannel:delete g.channel,await i.modify(u,H(g,y))}let p=ia(t,e);i.getAbstractFileByPath(p)instanceof fe.TFile||(await qt(i,(0,fe.normalizePath)(`${t}/tasks`)),await i.create(p,Bi(e)))}async function Wi(i,t,e){let s=i.vault,n=(0,fe.normalizePath)(`${t}/agents/${e}`),a=s.getAbstractFileByPath(n);if(!a)return;if(!(a instanceof fe.TFolder))throw new Error(`Expected folder at ${n}`);await i.fileManager.trashFile(a);let r=ia(t,e),o=s.getAbstractFileByPath(r);o instanceof fe.TFile&&await i.fileManager.trashFile(o)}var en=class extends F.PluginSettingTab{constructor(e){super(e.app,e);this.plugin=e}display(){let{containerEl:e}=this;e.empty(),new F.Setting(e).setName("Fleet folder").addText(a=>a.setValue(this.plugin.settings.fleetFolder).onChange(async r=>{this.plugin.settings.fleetFolder=r.trim()||rt.fleetFolder,await this.plugin.saveSettings()})),new F.Setting(e).setName("Claude CLI path").addText(a=>a.setValue(this.plugin.settings.claudeCliPath).onChange(async r=>{this.plugin.settings.claudeCliPath=r.trim()||rt.claudeCliPath,await this.plugin.saveSettings()})),new F.Setting(e).setName("Codex CLI path").setDesc("Used by agents with adapter \u201Ccodex\u201D. Install via `npm i -g @openai/codex` and authenticate with `codex login` in a terminal first.").addText(a=>a.setValue(this.plugin.settings.codexCliPath).onChange(async r=>{this.plugin.settings.codexCliPath=r.trim()||rt.codexCliPath,await this.plugin.saveSettings()}));let n=new F.Setting(e).setName("Default model").setDesc("Fallback for agents that don\u2019t set their own. Aliases (opus/sonnet/haiku) work on any backend; use Custom for pinned IDs or Bedrock/Vertex/Foundry.").controlEl.createDiv();Pt(n,{value:this.plugin.settings.defaultModel,onChange:async a=>{this.plugin.settings.defaultModel=a||rt.defaultModel,await this.plugin.saveSettings()}}),new F.Setting(e).setName("AWS region").addText(a=>a.setValue(this.plugin.settings.awsRegion).onChange(async r=>{this.plugin.settings.awsRegion=r.trim()||rt.awsRegion,await this.plugin.saveSettings()})),new F.Setting(e).setName("Max concurrent runs").addSlider(a=>a.setLimits(1,10,1).setValue(this.plugin.settings.maxConcurrentRuns).setDynamicTooltip().onChange(async r=>{this.plugin.settings.maxConcurrentRuns=r,await this.plugin.saveSettings()})),new F.Setting(e).setName("Run log retention").setDesc("Days to keep run logs before auto-prune.").addSlider(a=>a.setLimits(1,365,1).setValue(this.plugin.settings.runLogRetentionDays).setDynamicTooltip().onChange(async r=>{this.plugin.settings.runLogRetentionDays=r,await this.plugin.saveSettings()})),new F.Setting(e).setName("Catch up missed tasks").addToggle(a=>a.setValue(this.plugin.settings.catchUpMissedTasks).onChange(async r=>{this.plugin.settings.catchUpMissedTasks=r,await this.plugin.saveSettings()})),new F.Setting(e).setName("Notification level").addDropdown(a=>a.addOption("all","All").addOption("failures-only","Failures only").addOption("none","None").setValue(this.plugin.settings.notificationLevel).onChange(async r=>{this.plugin.settings.notificationLevel=r,await this.plugin.saveSettings()})),new F.Setting(e).setName("Status bar").addToggle(a=>a.setValue(this.plugin.settings.showStatusBar).onChange(async r=>{this.plugin.settings.showStatusBar=r,await this.plugin.saveSettings(),this.plugin.refreshStatusBar()})),new F.Setting(e).setName("Chat watchdog timeout (minutes)").setDesc("How long a chat session waits for any output from the Claude CLI before killing the subprocess and surfacing a timeout error. Raise this if your agents run long tools (e.g. large web fetches, builds) and you're seeing spurious 'no response' errors.").addSlider(a=>a.setLimits(1,60,1).setValue(this.plugin.settings.chatWatchdogMinutes).setDynamicTooltip().onChange(async r=>{this.plugin.settings.chatWatchdogMinutes=r,await this.plugin.saveSettings()})),new F.Setting(e).setName("Verify Claude CLI").setDesc("Checks that the configured binary is reachable.").addButton(a=>a.setButtonText("Verify").onClick(async()=>{let r=await this.plugin.verifyClaudeCli();new F.Notice(r?"Claude CLI detected.":"Claude CLI check failed. See console for details.")})),new F.Setting(e).setName("Verify Codex CLI").setDesc("Checks that the configured Codex binary is reachable.").addButton(a=>a.setButtonText("Verify").onClick(async()=>{let r=await this.plugin.verifyCodexCli();new F.Notice(r?"Codex CLI detected.":"Codex CLI check failed. See console for details.")})),this.renderWikiKeepersSection(e),this.renderChannelsSection(e)}renderChannelsSection(e){new F.Setting(e).setName("Channels").setHeading();let s=e.createDiv({cls:"af-settings-warning"});s.setCssStyles({padding:"12px"}),s.setCssStyles({margin:"8px 0 16px 0"}),s.setCssStyles({border:"1px solid var(--background-modifier-border)"}),s.setCssStyles({borderRadius:"6px"}),s.setCssStyles({background:"var(--background-secondary)"}),s.createEl("strong",{text:"Credential storage: "});let n=this.plugin.app.vault.configDir;s.createSpan({text:`Channel credentials are stored in this plugin's data.json inside your vault's ${n} folder. If you sync your ${n} folder across devices, credentials will sync with it. Do not commit this file to a public git repository.`}),new F.Setting(e).setName("Max concurrent channel sessions").setDesc("Hard cap on live claude subprocesses across all channels. Oldest idle session is hibernated when exceeded.").addSlider(d=>d.setLimits(1,20,1).setValue(this.plugin.settings.maxConcurrentChannelSessions).setDynamicTooltip().onChange(async u=>{this.plugin.settings.maxConcurrentChannelSessions=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Idle timeout (minutes)").setDesc("Channel sessions with no activity for this long get their subprocess hibernated. State is preserved and the next message resumes transparently.").addSlider(d=>d.setLimits(1,120,1).setValue(this.plugin.settings.channelIdleTimeoutMinutes).setDynamicTooltip().onChange(async u=>{this.plugin.settings.channelIdleTimeoutMinutes=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Rate limit per conversation").setDesc("Maximum messages allowed per external conversation within the rolling window.").addSlider(d=>d.setLimits(1,100,1).setValue(this.plugin.settings.channelRateLimitPerConversation).setDynamicTooltip().onChange(async u=>{this.plugin.settings.channelRateLimitPerConversation=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Rate limit window (minutes)").addSlider(d=>d.setLimits(1,60,1).setValue(this.plugin.settings.channelRateLimitWindowMinutes).setDynamicTooltip().onChange(async u=>{this.plugin.settings.channelRateLimitWindowMinutes=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Channel credentials").setHeading();let a=e.createDiv({cls:"af-channel-credentials"});this.renderCredentialList(a);let r=e.createDiv({cls:"af-channel-credential-add"});r.setCssStyles({marginTop:"12px"}),r.setCssStyles({padding:"12px"}),r.setCssStyles({border:"1px dashed var(--background-modifier-border)"}),r.setCssStyles({borderRadius:"6px"}),r.createEl("strong",{text:"Add a channel credential"});let o={ref:"",type:"slack",botToken:"",appToken:""};new F.Setting(r).setName("Reference name").setDesc("Used by `credential_ref` in _fleet/channels/*.md files.").addText(d=>d.setPlaceholder("my-creds").onChange(u=>{o.ref=u.trim()})),new F.Setting(r).setName("Type").addDropdown(d=>d.addOption("slack","Slack").addOption("telegram","Telegram").addOption("discord","Discord").setValue("slack").onChange(u=>{o.type=u,c.setCssStyles({display:u==="slack"?"":"none"}),l.setCssStyles({display:u==="telegram"?"":"none"}),h.setCssStyles({display:u==="discord"?"":"none"})}));let c=r.createDiv();new F.Setting(c).setName("Bot token (xoxb-...)").addText(d=>{d.inputEl.type="password",d.setPlaceholder("xoxb-...").onChange(u=>{o.botToken=u.trim()})}),new F.Setting(c).setName("App-level token (xapp-...)").setDesc("Generated in your Slack app's Basic Information \u2192 App-Level Tokens.").addText(d=>{d.inputEl.type="password",d.setPlaceholder("xapp-...").onChange(u=>{o.appToken=u.trim()})});let l=r.createDiv();l.setCssStyles({display:"none"}),new F.Setting(l).setName("Bot token").setDesc("From @BotFather on Telegram.").addText(d=>{d.inputEl.type="password",d.setPlaceholder("123456:ABC-DEF1234...").onChange(u=>{o.botToken=u.trim()})});let h=r.createDiv();h.setCssStyles({display:"none"}),new F.Setting(h).setName("Bot token").setDesc("From the Discord Developer Portal \u2192 your application \u2192 Bot \u2192 Reset Token. Enable the Message Content intent on the same page.").addText(d=>{d.inputEl.type="password",d.setPlaceholder("MTA...").onChange(u=>{o.botToken=u.trim()})}),new F.Setting(r).addButton(d=>d.setButtonText("Add credential").setCta().onClick(async()=>{if(!o.ref||!o.botToken){new F.Notice("Fill in the reference name and bot token.");return}let u;if(o.type==="telegram")u={type:"telegram",botToken:o.botToken};else if(o.type==="discord")u={type:"discord",botToken:o.botToken};else{if(!o.appToken){new F.Notice("Slack requires both bot token and app-level token.");return}u={type:"slack",botToken:o.botToken,appToken:o.appToken}}this.plugin.channelCredentials.set(o.ref,u),new F.Notice(`Added credential \`${o.ref}\`.`),this.display()}))}renderCredentialList(e){let s=this.plugin.channelCredentials.list();if(s.length===0){e.createDiv({text:"No channel credentials configured yet.",cls:"af-muted"}).setCssStyles({color:"var(--text-muted)"});return}for(let{ref:n,entry:a}of s){let r=e.createDiv({cls:"af-channel-credential-row"});r.setCssStyles({display:"flex"}),r.setCssStyles({justifyContent:"space-between"}),r.setCssStyles({alignItems:"center"}),r.setCssStyles({padding:"8px 12px"}),r.setCssStyles({border:"1px solid var(--background-modifier-border)"}),r.setCssStyles({borderRadius:"6px"}),r.setCssStyles({marginBottom:"6px"});let o=r.createDiv();o.createEl("strong",{text:n}),o.createEl("span",{text:` \xB7 ${a.type} \xB7 ${Il(Dl(a))}`,cls:"af-muted"}).setCssStyles({color:"var(--text-muted)"});let c=r.createEl("button",{text:"Remove"});c.onclick=()=>{this.plugin.channelCredentials.delete(n),new F.Notice(`Removed credential \`${n}\`.`),this.display()}}}renderWikiKeepersSection(e){new F.Setting(e).setName("Wiki Keepers").setHeading();let s=e.createEl("p",{cls:"af-settings-hint",text:"Per-scope wiki agents. Each runs its own inbox + watched folders + topics + heartbeat. All fields on the Add form are optional \u2014 click Create with everything blank to get a whole-vault keeper with defaults."});s.setCssStyles({color:"var(--af-text-secondary)"}),s.setCssStyles({fontSize:"12px"});let a=this.plugin.runtime.getSnapshot().agents.filter(o=>o.wikiKeeper!==void 0),r=e.createDiv({cls:"af-wk-list"});if(a.length===0)r.createDiv({cls:"af-wk-empty",text:"No Wiki Keepers yet. Click Add to create one."});else for(let o of a){let c=r.createDiv({cls:"af-wk-row"}),l=c.createDiv({cls:"af-wk-row-left"});l.createSpan({cls:"af-wk-name",text:o.name});let h=o.wikiKeeper.scopeRoot||"(whole vault)";l.createSpan({cls:"af-wk-scope",text:`scope: ${h}`});let d=c.createDiv({cls:"af-wk-row-actions"}),u=d.createEl("button",{text:"Edit",cls:"af-wk-row-btn"});u.onclick=()=>{new oa(this.plugin,o,()=>{this.scheduleRerender()}).open()};let p=d.createEl("button",{text:"Delete",cls:"af-wk-row-btn af-wk-row-btn-danger"});p.onclick=()=>{new Ht(this.plugin.app,{title:`Delete Wiki Keeper "${o.name}"?`,body:`This removes the agent folder at _fleet/agents/${o.name}/.
11772
+ `),{path:n,name:s}}async function zt(r,t){if(r.getAbstractFileByPath(t))return;let e=t.split("/"),s="";for(let n of e)if(s=s?`${s}/${n}`:n,!r.getAbstractFileByPath(s))try{await r.createFolder(s)}catch(a){if(!(a instanceof Error?a.message:String(a)).includes("already exists"))throw a}}async function Nr(r,t,e){if(r.getAbstractFileByPath(t))return;let s=t.replace(/\/[^/]+$/,"");s&&s!==t&&await zt(r,s),await r.create(t,e)}async function Wr(r,t,e,s){let n=(0,fe.normalizePath)(`${t}/agents/${e}`),a=(0,fe.normalizePath)(`${n}/config.md`),i=r.getAbstractFileByPath(a);if(!(i instanceof fe.TFile))throw new Error(`Config file not found for ${e} at ${a}`);let o=await r.cachedRead(i),{frontmatter:c,body:l}=J(o),h=c.wiki_keeper??{};h.watched_folders=s.watchedFolders,h.exclude_patterns=s.excludePatterns,s.watchedSince?h.watched_since=s.watchedSince:delete h.watched_since,delete h.ingest_schedule,delete h.lint_schedule,delete h.lint_day,h.file_substantive_answers=s.fileSubstantiveAnswers,h.obsidian_url_scheme=s.obsidianUrlScheme,h.max_tokens_per_ingest=s.maxTokensPerIngest,h.max_tokens_per_refresh=s.maxTokensPerRefresh,h.index_split_threshold=s.indexSplitThreshold,h.dedup_similarity_threshold=s.dedupSimilarityThreshold,h.summary_stale_days=s.summaryStaleDays,(typeof h.failed_path!="string"||!h.failed_path)&&(h.failed_path="_sources/failed"),c.wiki_keeper=h,delete c.allowed_tools,delete c.blocked_tools,await r.modify(i,H(c,l)),await $r(r,n);let d=(0,fe.normalizePath)(`${n}/HEARTBEAT.md`),u=r.getAbstractFileByPath(d);if(u instanceof fe.TFile){let f=await r.cachedRead(u),{frontmatter:g,body:y}=J(f);s.heartbeatChannel?g.channel=s.heartbeatChannel:delete g.channel,await r.modify(u,H(g,y))}let p=ra(t,e);r.getAbstractFileByPath(p)instanceof fe.TFile||(await zt(r,(0,fe.normalizePath)(`${t}/tasks`)),await r.create(p,Ur(e)))}async function Hr(r,t,e){let s=r.vault,n=(0,fe.normalizePath)(`${t}/agents/${e}`),a=s.getAbstractFileByPath(n);if(!a)return;if(!(a instanceof fe.TFolder))throw new Error(`Expected folder at ${n}`);await r.fileManager.trashFile(a);let i=ra(t,e),o=s.getAbstractFileByPath(i);o instanceof fe.TFile&&await r.fileManager.trashFile(o)}var en=class extends F.PluginSettingTab{constructor(e){super(e.app,e);this.plugin=e}display(){let{containerEl:e}=this;e.empty(),new F.Setting(e).setName("Fleet folder").addText(a=>a.setValue(this.plugin.settings.fleetFolder).onChange(async i=>{this.plugin.settings.fleetFolder=i.trim()||it.fleetFolder,await this.plugin.saveSettings()})),new F.Setting(e).setName("Claude CLI path").addText(a=>a.setValue(this.plugin.settings.claudeCliPath).onChange(async i=>{this.plugin.settings.claudeCliPath=i.trim()||it.claudeCliPath,await this.plugin.saveSettings()})),new F.Setting(e).setName("Codex CLI path").setDesc("Used by agents with adapter \u201Ccodex\u201D. Install via `npm i -g @openai/codex` and authenticate with `codex login` in a terminal first.").addText(a=>a.setValue(this.plugin.settings.codexCliPath).onChange(async i=>{this.plugin.settings.codexCliPath=i.trim()||it.codexCliPath,await this.plugin.saveSettings()}));let n=new F.Setting(e).setName("Default model").setDesc("Fallback for agents that don\u2019t set their own. Aliases (opus/sonnet/haiku) work on any backend; use Custom for pinned IDs or Bedrock/Vertex/Foundry.").controlEl.createDiv();Rt(n,{value:this.plugin.settings.defaultModel,onChange:async a=>{this.plugin.settings.defaultModel=a||it.defaultModel,await this.plugin.saveSettings()}}),new F.Setting(e).setName("AWS region").addText(a=>a.setValue(this.plugin.settings.awsRegion).onChange(async i=>{this.plugin.settings.awsRegion=i.trim()||it.awsRegion,await this.plugin.saveSettings()})),new F.Setting(e).setName("Max concurrent runs").addSlider(a=>a.setLimits(1,10,1).setValue(this.plugin.settings.maxConcurrentRuns).setDynamicTooltip().onChange(async i=>{this.plugin.settings.maxConcurrentRuns=i,await this.plugin.saveSettings()})),new F.Setting(e).setName("Run log retention").setDesc("Days to keep run logs before auto-prune.").addSlider(a=>a.setLimits(1,365,1).setValue(this.plugin.settings.runLogRetentionDays).setDynamicTooltip().onChange(async i=>{this.plugin.settings.runLogRetentionDays=i,await this.plugin.saveSettings()})),new F.Setting(e).setName("Catch up missed tasks").addToggle(a=>a.setValue(this.plugin.settings.catchUpMissedTasks).onChange(async i=>{this.plugin.settings.catchUpMissedTasks=i,await this.plugin.saveSettings()})),new F.Setting(e).setName("Notification level").addDropdown(a=>a.addOption("all","All").addOption("failures-only","Failures only").addOption("none","None").setValue(this.plugin.settings.notificationLevel).onChange(async i=>{this.plugin.settings.notificationLevel=i,await this.plugin.saveSettings()})),new F.Setting(e).setName("Status bar").addToggle(a=>a.setValue(this.plugin.settings.showStatusBar).onChange(async i=>{this.plugin.settings.showStatusBar=i,await this.plugin.saveSettings(),this.plugin.refreshStatusBar()})),new F.Setting(e).setName("Chat watchdog timeout (minutes)").setDesc("How long a chat session waits for any output from the Claude CLI before killing the subprocess and surfacing a timeout error. Raise this if your agents run long tools (e.g. large web fetches, builds) and you're seeing spurious 'no response' errors.").addSlider(a=>a.setLimits(1,60,1).setValue(this.plugin.settings.chatWatchdogMinutes).setDynamicTooltip().onChange(async i=>{this.plugin.settings.chatWatchdogMinutes=i,await this.plugin.saveSettings()})),new F.Setting(e).setName("Verify Claude CLI").setDesc("Checks that the configured binary is reachable.").addButton(a=>a.setButtonText("Verify").onClick(async()=>{let i=await this.plugin.verifyClaudeCli();new F.Notice(i?"Claude CLI detected.":"Claude CLI check failed. See console for details.")})),new F.Setting(e).setName("Verify Codex CLI").setDesc("Checks that the configured Codex binary is reachable.").addButton(a=>a.setButtonText("Verify").onClick(async()=>{let i=await this.plugin.verifyCodexCli();new F.Notice(i?"Codex CLI detected.":"Codex CLI check failed. See console for details.")})),this.renderWikiKeepersSection(e),this.renderChannelsSection(e)}renderChannelsSection(e){new F.Setting(e).setName("Channels").setHeading();let s=e.createDiv({cls:"af-settings-warning"});s.setCssStyles({padding:"12px"}),s.setCssStyles({margin:"8px 0 16px 0"}),s.setCssStyles({border:"1px solid var(--background-modifier-border)"}),s.setCssStyles({borderRadius:"6px"}),s.setCssStyles({background:"var(--background-secondary)"}),s.createEl("strong",{text:"Credential storage: "});let n=this.plugin.app.vault.configDir;s.createSpan({text:`Channel credentials are stored in this plugin's data.json inside your vault's ${n} folder. If you sync your ${n} folder across devices, credentials will sync with it. Do not commit this file to a public git repository.`}),new F.Setting(e).setName("Max concurrent channel sessions").setDesc("Hard cap on live claude subprocesses across all channels. Oldest idle session is hibernated when exceeded.").addSlider(d=>d.setLimits(1,20,1).setValue(this.plugin.settings.maxConcurrentChannelSessions).setDynamicTooltip().onChange(async u=>{this.plugin.settings.maxConcurrentChannelSessions=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Idle timeout (minutes)").setDesc("Channel sessions with no activity for this long get their subprocess hibernated. State is preserved and the next message resumes transparently.").addSlider(d=>d.setLimits(1,120,1).setValue(this.plugin.settings.channelIdleTimeoutMinutes).setDynamicTooltip().onChange(async u=>{this.plugin.settings.channelIdleTimeoutMinutes=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Rate limit per conversation").setDesc("Maximum messages allowed per external conversation within the rolling window.").addSlider(d=>d.setLimits(1,100,1).setValue(this.plugin.settings.channelRateLimitPerConversation).setDynamicTooltip().onChange(async u=>{this.plugin.settings.channelRateLimitPerConversation=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Rate limit window (minutes)").addSlider(d=>d.setLimits(1,60,1).setValue(this.plugin.settings.channelRateLimitWindowMinutes).setDynamicTooltip().onChange(async u=>{this.plugin.settings.channelRateLimitWindowMinutes=u,await this.plugin.saveSettings()})),new F.Setting(e).setName("Channel credentials").setHeading();let a=e.createDiv({cls:"af-channel-credentials"});this.renderCredentialList(a);let i=e.createDiv({cls:"af-channel-credential-add"});i.setCssStyles({marginTop:"12px"}),i.setCssStyles({padding:"12px"}),i.setCssStyles({border:"1px dashed var(--background-modifier-border)"}),i.setCssStyles({borderRadius:"6px"}),i.createEl("strong",{text:"Add a channel credential"});let o={ref:"",type:"slack",botToken:"",appToken:""};new F.Setting(i).setName("Reference name").setDesc("Used by `credential_ref` in _fleet/channels/*.md files.").addText(d=>d.setPlaceholder("my-creds").onChange(u=>{o.ref=u.trim()})),new F.Setting(i).setName("Type").addDropdown(d=>d.addOption("slack","Slack").addOption("telegram","Telegram").addOption("discord","Discord").setValue("slack").onChange(u=>{o.type=u,c.setCssStyles({display:u==="slack"?"":"none"}),l.setCssStyles({display:u==="telegram"?"":"none"}),h.setCssStyles({display:u==="discord"?"":"none"})}));let c=i.createDiv();new F.Setting(c).setName("Bot token (xoxb-...)").addText(d=>{d.inputEl.type="password",d.setPlaceholder("xoxb-...").onChange(u=>{o.botToken=u.trim()})}),new F.Setting(c).setName("App-level token (xapp-...)").setDesc("Generated in your Slack app's Basic Information \u2192 App-Level Tokens.").addText(d=>{d.inputEl.type="password",d.setPlaceholder("xapp-...").onChange(u=>{o.appToken=u.trim()})});let l=i.createDiv();l.setCssStyles({display:"none"}),new F.Setting(l).setName("Bot token").setDesc("From @BotFather on Telegram.").addText(d=>{d.inputEl.type="password",d.setPlaceholder("123456:ABC-DEF1234...").onChange(u=>{o.botToken=u.trim()})});let h=i.createDiv();h.setCssStyles({display:"none"}),new F.Setting(h).setName("Bot token").setDesc("From the Discord Developer Portal \u2192 your application \u2192 Bot \u2192 Reset Token. Enable the Message Content intent on the same page.").addText(d=>{d.inputEl.type="password",d.setPlaceholder("MTA...").onChange(u=>{o.botToken=u.trim()})}),new F.Setting(i).addButton(d=>d.setButtonText("Add credential").setCta().onClick(async()=>{if(!o.ref||!o.botToken){new F.Notice("Fill in the reference name and bot token.");return}let u;if(o.type==="telegram")u={type:"telegram",botToken:o.botToken};else if(o.type==="discord")u={type:"discord",botToken:o.botToken};else{if(!o.appToken){new F.Notice("Slack requires both bot token and app-level token.");return}u={type:"slack",botToken:o.botToken,appToken:o.appToken}}this.plugin.channelCredentials.set(o.ref,u),new F.Notice(`Added credential \`${o.ref}\`.`),this.display()}))}renderCredentialList(e){let s=this.plugin.channelCredentials.list();if(s.length===0){e.createDiv({text:"No channel credentials configured yet.",cls:"af-muted"}).setCssStyles({color:"var(--text-muted)"});return}for(let{ref:n,entry:a}of s){let i=e.createDiv({cls:"af-channel-credential-row"});i.setCssStyles({display:"flex"}),i.setCssStyles({justifyContent:"space-between"}),i.setCssStyles({alignItems:"center"}),i.setCssStyles({padding:"8px 12px"}),i.setCssStyles({border:"1px solid var(--background-modifier-border)"}),i.setCssStyles({borderRadius:"6px"}),i.setCssStyles({marginBottom:"6px"});let o=i.createDiv();o.createEl("strong",{text:n}),o.createEl("span",{text:` \xB7 ${a.type} \xB7 ${Ml(Il(a))}`,cls:"af-muted"}).setCssStyles({color:"var(--text-muted)"});let c=i.createEl("button",{text:"Remove"});c.onclick=()=>{this.plugin.channelCredentials.delete(n),new F.Notice(`Removed credential \`${n}\`.`),this.display()}}}renderWikiKeepersSection(e){new F.Setting(e).setName("Wiki Keepers").setHeading();let s=e.createEl("p",{cls:"af-settings-hint",text:"Per-scope wiki agents. Each runs its own inbox + watched folders + topics + heartbeat. All fields on the Add form are optional \u2014 click Create with everything blank to get a whole-vault keeper with defaults."});s.setCssStyles({color:"var(--af-text-secondary)"}),s.setCssStyles({fontSize:"12px"});let a=this.plugin.runtime.getSnapshot().agents.filter(o=>o.wikiKeeper!==void 0),i=e.createDiv({cls:"af-wk-list"});if(a.length===0)i.createDiv({cls:"af-wk-empty",text:"No Wiki Keepers yet. Click Add to create one."});else for(let o of a){let c=i.createDiv({cls:"af-wk-row"}),l=c.createDiv({cls:"af-wk-row-left"});l.createSpan({cls:"af-wk-name",text:o.name});let h=o.wikiKeeper.scopeRoot||"(whole vault)";l.createSpan({cls:"af-wk-scope",text:`scope: ${h}`});let d=c.createDiv({cls:"af-wk-row-actions"}),u=d.createEl("button",{text:"Edit",cls:"af-wk-row-btn"});u.onclick=()=>{new oa(this.plugin,o,()=>{this.scheduleRerender()}).open()};let p=d.createEl("button",{text:"Delete",cls:"af-wk-row-btn af-wk-row-btn-danger"});p.onclick=()=>{new qt(this.plugin.app,{title:`Delete Wiki Keeper "${o.name}"?`,body:`This removes the agent folder at _fleet/agents/${o.name}/.
11862
11773
 
11863
- Your scope's inbox, topics, index, and log are NOT deleted.`,confirmText:"Delete",danger:!0,onConfirm:async()=>{try{await Wi(this.plugin.app,this.plugin.settings.fleetFolder,o.name),await this.plugin.refreshFromVault(),new F.Notice(`Wiki Keeper "${o.name}" deleted.`),this.scheduleRerender()}catch(m){let f=m instanceof Error?m.message:String(m);new F.Notice(`Failed to delete: ${f}`)}}}).open()}}new F.Setting(e).setName("Add Wiki Keeper").setDesc("Create a new scoped wiki agent. All fields optional.").addButton(o=>o.setButtonText("+ Add").onClick(()=>{new ra(this.plugin,()=>{this.scheduleRerender()}).open()}))}scheduleRerender(){window.setTimeout(()=>{(async()=>{try{await this.plugin.refreshFromVault()}catch{}this.display()})()},600)}},ra=class extends F.Modal{constructor(e,s){super(e.app);this.plugin=e;this.onCreated=s;this.input=na()}input;onOpen(){let{contentEl:e}=this;e.empty(),e.addClass("af-wk-modal"),e.createEl("h2",{text:"Add Wiki Keeper"});let s=e.createDiv({cls:"af-wk-banner"});s.createEl("strong",{text:"All fields are optional. "}),s.createSpan({text:"Click Create with everything blank and you'll get a whole-vault keeper with sensible defaults (inbox at _sources/inbox, topics at topics/, nightly ingest at 3am, Sunday lint)."}),e.createEl("h3",{text:"Scope",cls:"af-wk-section-h"}),new F.Setting(e).setName("Scope folder").setDesc("Optional. Vault-relative path. Empty = whole vault.").addText(h=>h.setPlaceholder("(empty = whole vault)").setValue(this.input.scopeRoot).onChange(d=>{this.input.scopeRoot=d.trim(),this.renderPreview()})),new F.Setting(e).setName("Slug").setDesc("Optional. Agent name suffix. Default: derived from scope folder name, or 'wiki-keeper' for whole-vault.").addText(h=>h.setPlaceholder("(auto)").setValue(this.input.scopeSlug).onChange(d=>{this.input.scopeSlug=d.trim(),this.renderPreview()})),this.renderPreview(),e.createEl("h3",{text:"Sources",cls:"af-wk-section-h"}),new F.Setting(e).setName("Watched folders").setDesc("Optional. Comma- or newline-separated vault-relative paths. Read-only \u2014 claims are extracted but files are never moved. Leave empty for inbox-only mode (you drop files into _sources/inbox/ and they get archived after processing).").addTextArea(h=>h.setPlaceholder("(empty = inbox-only)").setValue(this.input.watchedFolders.join(", ")).onChange(d=>{this.input.watchedFolders=d.split(/[,\n]/).map(u=>u.trim()).filter(Boolean)})),new F.Setting(e).setName("Exclude patterns").setDesc("Optional. Glob patterns to skip. Leave empty to process everything under watched + inbox.").addTextArea(h=>h.setPlaceholder("(empty = no excludes)").setValue(this.input.excludePatterns.join(", ")).onChange(d=>{this.input.excludePatterns=d.split(/[,\n]/).map(u=>u.trim()).filter(Boolean)})),new F.Setting(e).setName("Watch files modified since").setDesc("Watched mode only \u2014 files whose last modification date is before this are skipped. Defaults to today so an established vault doesn't flood the keeper on first run. Clear the field to process everything.").addText(h=>{h.inputEl.type="date",h.setValue(this.input.watchedSince).onChange(d=>{this.input.watchedSince=d.trim()})});let n=this.plugin.runtime.getSnapshot().channels.map(h=>h.name);new F.Setting(e).setName("Heartbeat channel").setDesc("Optional. Slack/Telegram channel for ingest + lint summaries. Leave as (none) to disable.").addDropdown(h=>{h.addOption("","(none)");for(let d of n)h.addOption(d,d);h.setValue(this.input.heartbeatChannel).onChange(d=>{this.input.heartbeatChannel=d})}),e.createEl("h3",{text:"Advanced",cls:"af-wk-section-h"}),new F.Setting(e).setName("Max tokens per ingest").setDesc("Hard cap on token spend per run. Unprocessed files resume next cycle.").addText(h=>h.setValue(String(this.input.maxTokensPerIngest)).onChange(d=>{let u=parseInt(d,10);!isNaN(u)&&u>0&&(this.input.maxTokensPerIngest=u)})),new F.Setting(e).setName("File substantive answers").setDesc("If on, wiki-query files long answers under _topics/syntheses/. Off = answers live only in chat.").addToggle(h=>h.setValue(this.input.fileSubstantiveAnswers).onChange(d=>{this.input.fileSubstantiveAnswers=d})),new F.Setting(e).setName("Obsidian URL scheme for citations").setDesc("If on, external-channel (Slack/Telegram) replies rewrite [[wikilinks]] as obsidian:// URLs.").addToggle(h=>h.setValue(this.input.obsidianUrlScheme).onChange(d=>{this.input.obsidianUrlScheme=d}));let a=e.createEl("h3",{text:"Paths (expert)",cls:"af-wk-section-h"});a.title="Paths are relative to scope_root. Defaults work for almost everyone. Override only if your vault layout demands it \u2014 these cannot be changed after creation.";let r=(h,d,u)=>{new F.Setting(e).setName(h).setDesc(d).addText(p=>p.setPlaceholder(String(this.input[u]??"")).setValue(String(this.input[u]??"")).onChange(m=>{this.input[u]=m.trim()||Ni[u]}))};r("Inbox path","Where one-off source drops land.","inboxPath"),r("Archive path","Where processed inbox files are moved after summarization.","archivePath"),r("Topics root","Folder under scope_root that holds the generated topic pages.","topicsRoot"),r("Index path","Relative file path for the curated index.","indexPath"),r("Log path","Relative file path for the keeper's activity log.","logPath"),r("State file","Hidden JSON file that tracks watched-source mtimes.","stateFile");let o=e.createDiv({cls:"af-wk-modal-footer"}),c=o.createEl("button",{text:"Cancel"});c.onclick=()=>this.close();let l=o.createEl("button",{text:"Create",cls:"mod-cta"});l.onclick=async()=>{l.setText("Creating\u2026"),l.disabled=!0;try{let{name:h}=await $i(this.app.vault,this.plugin.settings.fleetFolder,this.input);await this.plugin.refreshFromVault(),this.close(),new F.Notice(`Wiki Keeper "${h}" created.`),this.onCreated()}catch(h){let d=h instanceof Error?h.message:String(h);new F.Notice(`Failed to create Wiki Keeper: ${d}`),l.setText("Create"),l.disabled=!1}}}renderPreview(){let e=this.contentEl.querySelector(".af-wk-name-preview");e||(e=this.contentEl.createDiv({cls:"af-wk-name-preview"}));let s=this.input.scopeSlug||this.input.scopeRoot;e.setText(`\u2192 Will create agent: ${aa(s)}`)}onClose(){this.contentEl.empty()}},oa=class extends F.Modal{constructor(e,s,n){super(e.app);this.plugin=e;this.agent=s;this.onSaved=n;let a=s.wikiKeeper;this.edit={watchedFolders:[...a.watchedFolders],excludePatterns:[...a.excludePatterns],watchedSince:a.watchedSince,heartbeatChannel:s.heartbeatChannel??"",fileSubstantiveAnswers:a.fileSubstantiveAnswers,obsidianUrlScheme:a.obsidianUrlScheme,maxTokensPerIngest:a.maxTokensPerIngest,maxTokensPerRefresh:a.maxTokensPerRefresh,indexSplitThreshold:a.indexSplitThreshold,dedupSimilarityThreshold:a.dedupSimilarityThreshold,summaryStaleDays:a.summaryStaleDays}}edit;onOpen(){let{contentEl:e}=this;e.empty(),e.addClass("af-wk-modal"),e.createEl("h2",{text:`Edit ${this.agent.name}`});let s=e.createDiv({cls:"af-wk-banner"}),n=this.agent.wikiKeeper.scopeRoot||"(whole vault)";s.createEl("strong",{text:"Scope: "}),s.createSpan({text:n}),s.createEl("br"),s.createSpan({text:"Scope and paths are fixed after creation \u2014 changing them would orphan existing topic pages. To move a Wiki Keeper, delete this one and create a new one at the new scope."}),e.createEl("h3",{text:"Sources",cls:"af-wk-section-h"}),new F.Setting(e).setName("Watched folders").setDesc("Comma- or newline-separated vault-relative paths.").addTextArea(l=>l.setValue(this.edit.watchedFolders.join(", ")).onChange(h=>{this.edit.watchedFolders=h.split(/[,\n]/).map(d=>d.trim()).filter(Boolean)})),new F.Setting(e).setName("Exclude patterns").setDesc("Glob patterns to skip.").addTextArea(l=>l.setValue(this.edit.excludePatterns.join(", ")).onChange(h=>{this.edit.excludePatterns=h.split(/[,\n]/).map(d=>d.trim()).filter(Boolean)})),new F.Setting(e).setName("Watch files modified since").setDesc("Watched mode skips files older than this date. Clear to process everything.").addText(l=>{l.inputEl.type="date",l.setValue(this.edit.watchedSince).onChange(h=>{this.edit.watchedSince=h.trim()})});let a=this.plugin.runtime.getSnapshot().channels.map(l=>l.name);new F.Setting(e).setName("Heartbeat channel").addDropdown(l=>{l.addOption("","(none)");for(let h of a)l.addOption(h,h);l.setValue(this.edit.heartbeatChannel).onChange(h=>{this.edit.heartbeatChannel=h})}),e.createEl("h3",{text:"Advanced",cls:"af-wk-section-h"}),new F.Setting(e).setName("Max tokens per ingest").addText(l=>l.setValue(String(this.edit.maxTokensPerIngest)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.maxTokensPerIngest=d)})),new F.Setting(e).setName("Max tokens per refresh").setDesc("Separate budget for the synthesis-refresh phase that regenerates topic-page summaries.").addText(l=>l.setValue(String(this.edit.maxTokensPerRefresh)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.maxTokensPerRefresh=d)})),new F.Setting(e).setName("Index split threshold").setDesc("Topic-page count above which index.md splits into per-type sub-MOCs.").addText(l=>l.setValue(String(this.edit.indexSplitThreshold)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.indexSplitThreshold=d)})),new F.Setting(e).setName("Summary stale (days)").setDesc("Lint flags topic-page summaries older than this and chains wiki-refresh.").addText(l=>l.setValue(String(this.edit.summaryStaleDays)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.summaryStaleDays=d)})),new F.Setting(e).setName("Dedup similarity threshold").setDesc("Levenshtein-ratio above which lint proposes merging two same-type pages (0.0\u20131.0).").addText(l=>l.setValue(String(this.edit.dedupSimilarityThreshold)).onChange(h=>{let d=parseFloat(h);!isNaN(d)&&d>0&&d<=1&&(this.edit.dedupSimilarityThreshold=d)})),new F.Setting(e).setName("File substantive answers").setDesc("Compounding: wiki-query files long answers under _topics/syntheses/ and bullets each cited topic page. Default ON.").addToggle(l=>l.setValue(this.edit.fileSubstantiveAnswers).onChange(h=>{this.edit.fileSubstantiveAnswers=h})),new F.Setting(e).setName("Obsidian URL scheme for citations").setDesc("Rewrite [[wikilinks]] as obsidian:// URLs when replying via Slack/Telegram.").addToggle(l=>l.setValue(this.edit.obsidianUrlScheme).onChange(h=>{this.edit.obsidianUrlScheme=h}));let r=e.createDiv({cls:"af-wk-modal-footer"}),o=r.createEl("button",{text:"Cancel"});o.onclick=()=>this.close();let c=r.createEl("button",{text:"Save",cls:"mod-cta"});c.onclick=async()=>{c.setText("Saving\u2026"),c.disabled=!0;try{await ji(this.app.vault,this.plugin.settings.fleetFolder,this.agent.name,this.edit),await this.plugin.refreshFromVault(),this.close(),new F.Notice("Saved."),this.onSaved()}catch(l){let h=l instanceof Error?l.message:String(l);new F.Notice(`Failed to save: ${h}`),c.setText("Save"),c.disabled=!1}}}onClose(){this.contentEl.empty()}};function Dl(i){return i.type==="slack",i.botToken}function Il(i){return i.length<=10?"***":`${i.slice(0,6)}\u2026${i.slice(-4)}`}var Sa=require("crypto");function He(i,t,e,s,n,a,r,o){return He.fromTZ(He.tp(i,t,e,s,n,a,r),o)}He.fromTZISO=(i,t,e)=>He.fromTZ(Ml(i,t),e);He.fromTZ=function(i,t){let e=new Date(Date.UTC(i.y,i.m-1,i.d,i.h,i.i,i.s)),s=la(i.tz,e),n=new Date(e.getTime()-s),a=la(i.tz,n);if(a-s===0)return n;{let r=new Date(e.getTime()-a),o=la(i.tz,r);if(o-a===0)return r;if(!t&&o-a>0)return r;if(t)throw new Error("Invalid date passed to fromTZ()");return n}};He.toTZ=function(i,t){let e=i.toLocaleString("en-US",{timeZone:t}).replace(/[\u202f]/," "),s=new Date(e);return{y:s.getFullYear(),m:s.getMonth()+1,d:s.getDate(),h:s.getHours(),i:s.getMinutes(),s:s.getSeconds(),tz:t}};He.tp=(i,t,e,s,n,a,r)=>({y:i,m:t,d:e,h:s,i:n,s:a,tz:r});function la(i,t=new Date){let e=t.toLocaleString("en-US",{timeZone:i,timeZoneName:"shortOffset"}).split(" ").slice(-1)[0],s=t.toLocaleString("en-US").replace(/[\u202f]/," ");return Date.parse(`${s} GMT`)-Date.parse(`${s} ${e}`)}function Ml(i,t){let e=new Date(Date.parse(i));if(isNaN(e))throw new Error("minitz: Invalid ISO8601 passed to parser.");let s=i.substring(9);return i.includes("Z")||s.includes("-")||s.includes("+")?He.tp(e.getUTCFullYear(),e.getUTCMonth()+1,e.getUTCDate(),e.getUTCHours(),e.getUTCMinutes(),e.getUTCSeconds(),"Etc/UTC"):He.tp(e.getFullYear(),e.getMonth()+1,e.getDate(),e.getHours(),e.getMinutes(),e.getSeconds(),t)}He.minitz=He;function Ll(i){if(i===void 0&&(i={}),delete i.name,i.legacyMode=i.legacyMode===void 0?!0:i.legacyMode,i.paused=i.paused===void 0?!1:i.paused,i.maxRuns=i.maxRuns===void 0?1/0:i.maxRuns,i.catch=i.catch===void 0?!1:i.catch,i.interval=i.interval===void 0?0:parseInt(i.interval,10),i.utcOffset=i.utcOffset===void 0?void 0:parseInt(i.utcOffset,10),i.unref=i.unref===void 0?!1:i.unref,i.startAt&&(i.startAt=new _e(i.startAt,i.timezone)),i.stopAt&&(i.stopAt=new _e(i.stopAt,i.timezone)),i.interval!==null){if(isNaN(i.interval))throw new Error("CronOptions: Supplied value for interval is not a number");if(i.interval<0)throw new Error("CronOptions: Supplied value for interval can not be negative")}if(i.utcOffset!==void 0){if(isNaN(i.utcOffset))throw new Error("CronOptions: Invalid value passed for utcOffset, should be number representing minutes offset from UTC.");if(i.utcOffset<-870||i.utcOffset>870)throw new Error("CronOptions: utcOffset out of bounds.");if(i.utcOffset!==void 0&&i.timezone)throw new Error("CronOptions: Combining 'utcOffset' with 'timezone' is not allowed.")}if(i.unref!==!0&&i.unref!==!1)throw new Error("CronOptions: Unref should be either true, false or undefined(false).");return i}var ca=32,ms=31|ca,qi=[1,2,4,8,16];function Ge(i,t){this.pattern=i,this.timezone=t,this.second=Array(60).fill(0),this.minute=Array(60).fill(0),this.hour=Array(24).fill(0),this.day=Array(31).fill(0),this.month=Array(12).fill(0),this.dayOfWeek=Array(7).fill(0),this.lastDayOfMonth=!1,this.starDOM=!1,this.starDOW=!1,this.parse()}Ge.prototype.parse=function(){if(!(typeof this.pattern=="string"||this.pattern.constructor===String))throw new TypeError("CronPattern: Pattern has to be of type string.");this.pattern.indexOf("@")>=0&&(this.pattern=this.handleNicknames(this.pattern).trim());let i=this.pattern.replace(/\s+/g," ").split(" ");if(i.length<5||i.length>6)throw new TypeError("CronPattern: invalid configuration format ('"+this.pattern+"'), exactly five or six space separated parts are required.");if(i.length===5&&i.unshift("0"),i[3].indexOf("L")>=0&&(i[3]=i[3].replace("L",""),this.lastDayOfMonth=!0),i[3]=="*"&&(this.starDOM=!0),i[4].length>=3&&(i[4]=this.replaceAlphaMonths(i[4])),i[5].length>=3&&(i[5]=this.replaceAlphaDays(i[5])),i[5]=="*"&&(this.starDOW=!0),this.pattern.indexOf("?")>=0){let t=new _e(new Date,this.timezone).getDate(!0);i[0]=i[0].replace("?",t.getSeconds()),i[1]=i[1].replace("?",t.getMinutes()),i[2]=i[2].replace("?",t.getHours()),this.starDOM||(i[3]=i[3].replace("?",t.getDate())),i[4]=i[4].replace("?",t.getMonth()+1),this.starDOW||(i[5]=i[5].replace("?",t.getDay()))}this.throwAtIllegalCharacters(i),this.partToArray("second",i[0],0,1),this.partToArray("minute",i[1],0,1),this.partToArray("hour",i[2],0,1),this.partToArray("day",i[3],-1,1),this.partToArray("month",i[4],-1,1),this.partToArray("dayOfWeek",i[5],0,ms),this.dayOfWeek[7]&&(this.dayOfWeek[0]=this.dayOfWeek[7])};Ge.prototype.partToArray=function(i,t,e,s){let n=this[i],a=i==="day"&&this.lastDayOfMonth;if(t===""&&!a)throw new TypeError("CronPattern: configuration entry "+i+" ("+t+") is empty, check for trailing spaces.");if(t==="*")return n.fill(s);let r=t.split(",");if(r.length>1)for(let o=0;o<r.length;o++)this.partToArray(i,r[o],e,s);else t.indexOf("-")!==-1&&t.indexOf("/")!==-1?this.handleRangeWithStepping(t,i,e,s):t.indexOf("-")!==-1?this.handleRange(t,i,e,s):t.indexOf("/")!==-1?this.handleStepping(t,i,e,s):t!==""&&this.handleNumber(t,i,e,s)};Ge.prototype.throwAtIllegalCharacters=function(i){for(let t=0;t<i.length;t++)if((t===5?/[^/*0-9,\-#L]+/:/[^/*0-9,-]+/).test(i[t]))throw new TypeError("CronPattern: configuration entry "+t+" ("+i[t]+") contains illegal characters.")};Ge.prototype.handleNumber=function(i,t,e,s){let n=this.extractNth(i,t),a=parseInt(n[0],10)+e;if(isNaN(a))throw new TypeError("CronPattern: "+t+" is not a number: '"+i+"'");this.setPart(t,a,n[1]||s)};Ge.prototype.setPart=function(i,t,e){if(!Object.prototype.hasOwnProperty.call(this,i))throw new TypeError("CronPattern: Invalid part specified: "+i);if(i==="dayOfWeek"){if(t===7&&(t=0),(t<0||t>6)&&t!=="L")throw new RangeError("CronPattern: Invalid value for dayOfWeek: "+t);this.setNthWeekdayOfMonth(t,e);return}if(i==="second"||i==="minute"){if(t<0||t>=60)throw new RangeError("CronPattern: Invalid value for "+i+": "+t)}else if(i==="hour"){if(t<0||t>=24)throw new RangeError("CronPattern: Invalid value for "+i+": "+t)}else if(i==="day"){if(t<0||t>=31)throw new RangeError("CronPattern: Invalid value for "+i+": "+t)}else if(i==="month"&&(t<0||t>=12))throw new RangeError("CronPattern: Invalid value for "+i+": "+t);this[i][t]=e};Ge.prototype.handleRangeWithStepping=function(i,t,e,s){let n=this.extractNth(i,t),a=n[0].match(/^(\d+)-(\d+)\/(\d+)$/);if(a===null)throw new TypeError("CronPattern: Syntax error, illegal range with stepping: '"+i+"'");let[,r,o,c]=a;if(r=parseInt(r,10)+e,o=parseInt(o,10)+e,c=parseInt(c,10),isNaN(r))throw new TypeError("CronPattern: Syntax error, illegal lower range (NaN)");if(isNaN(o))throw new TypeError("CronPattern: Syntax error, illegal upper range (NaN)");if(isNaN(c))throw new TypeError("CronPattern: Syntax error, illegal stepping: (NaN)");if(c===0)throw new TypeError("CronPattern: Syntax error, illegal stepping: 0");if(c>this[t].length)throw new TypeError("CronPattern: Syntax error, steps cannot be greater than maximum value of part ("+this[t].length+")");if(r>o)throw new TypeError("CronPattern: From value is larger than to value: '"+i+"'");for(let l=r;l<=o;l+=c)this.setPart(t,l,n[1]||s)};Ge.prototype.extractNth=function(i,t){let e=i,s;if(e.includes("#")){if(t!=="dayOfWeek")throw new Error("CronPattern: nth (#) only allowed in day-of-week field");s=e.split("#")[1],e=e.split("#")[0]}return[e,s]};Ge.prototype.handleRange=function(i,t,e,s){let n=this.extractNth(i,t),a=n[0].split("-");if(a.length!==2)throw new TypeError("CronPattern: Syntax error, illegal range: '"+i+"'");let r=parseInt(a[0],10)+e,o=parseInt(a[1],10)+e;if(isNaN(r))throw new TypeError("CronPattern: Syntax error, illegal lower range (NaN)");if(isNaN(o))throw new TypeError("CronPattern: Syntax error, illegal upper range (NaN)");if(r>o)throw new TypeError("CronPattern: From value is larger than to value: '"+i+"'");for(let c=r;c<=o;c++)this.setPart(t,c,n[1]||s)};Ge.prototype.handleStepping=function(i,t,e,s){let n=this.extractNth(i,t),a=n[0].split("/");if(a.length!==2)throw new TypeError("CronPattern: Syntax error, illegal stepping: '"+i+"'");let r=0;a[0]!=="*"&&(r=parseInt(a[0],10)+e);let o=parseInt(a[1],10);if(isNaN(o))throw new TypeError("CronPattern: Syntax error, illegal stepping: (NaN)");if(o===0)throw new TypeError("CronPattern: Syntax error, illegal stepping: 0");if(o>this[t].length)throw new TypeError("CronPattern: Syntax error, max steps for part is ("+this[t].length+")");for(let c=r;c<this[t].length;c+=o)this.setPart(t,c,n[1]||s)};Ge.prototype.replaceAlphaDays=function(i){return i.replace(/-sun/gi,"-7").replace(/sun/gi,"0").replace(/mon/gi,"1").replace(/tue/gi,"2").replace(/wed/gi,"3").replace(/thu/gi,"4").replace(/fri/gi,"5").replace(/sat/gi,"6")};Ge.prototype.replaceAlphaMonths=function(i){return i.replace(/jan/gi,"1").replace(/feb/gi,"2").replace(/mar/gi,"3").replace(/apr/gi,"4").replace(/may/gi,"5").replace(/jun/gi,"6").replace(/jul/gi,"7").replace(/aug/gi,"8").replace(/sep/gi,"9").replace(/oct/gi,"10").replace(/nov/gi,"11").replace(/dec/gi,"12")};Ge.prototype.handleNicknames=function(i){let t=i.trim().toLowerCase();return t==="@yearly"||t==="@annually"?"0 0 1 1 *":t==="@monthly"?"0 0 1 * *":t==="@weekly"?"0 0 * * 0":t==="@daily"?"0 0 * * *":t==="@hourly"?"0 * * * *":i};Ge.prototype.setNthWeekdayOfMonth=function(i,t){if(t==="L")this.dayOfWeek[i]=this.dayOfWeek[i]|ca;else if(t<6&&t>0)this.dayOfWeek[i]=this.dayOfWeek[i]|qi[t-1];else if(t===ms)this.dayOfWeek[i]=ms;else throw new TypeError(`CronPattern: nth weekday of of range, should be 1-5 or L. Value: ${t}`)};var zi=[31,28,31,30,31,30,31,31,30,31,30,31],ft=[["month","year",0],["day","month",-1],["hour","day",0],["minute","hour",0],["second","minute",0]];function _e(i,t){if(this.tz=t,i&&i instanceof Date)if(!isNaN(i))this.fromDate(i);else throw new TypeError("CronDate: Invalid date passed to CronDate constructor");else if(i===void 0)this.fromDate(new Date);else if(i&&typeof i=="string")this.fromString(i);else if(i instanceof _e)this.fromCronDate(i);else throw new TypeError("CronDate: Invalid type ("+typeof i+") passed to CronDate constructor")}_e.prototype.isNthWeekdayOfMonth=function(i,t,e,s){let a=new Date(Date.UTC(i,t,e)).getUTCDay(),r=0;for(let o=1;o<=e;o++)new Date(Date.UTC(i,t,o)).getUTCDay()===a&&r++;if(s&ms&&qi[r-1]&s)return!0;if(s&ca){let o=new Date(Date.UTC(i,t+1,0)).getUTCDate();for(let c=e+1;c<=o;c++)if(new Date(Date.UTC(i,t,c)).getUTCDay()===a)return!1;return!0}return!1};_e.prototype.fromDate=function(i){if(this.tz!==void 0)if(typeof this.tz=="number")this.ms=i.getUTCMilliseconds(),this.second=i.getUTCSeconds(),this.minute=i.getUTCMinutes()+this.tz,this.hour=i.getUTCHours(),this.day=i.getUTCDate(),this.month=i.getUTCMonth(),this.year=i.getUTCFullYear(),this.apply();else{let t=He.toTZ(i,this.tz);this.ms=i.getMilliseconds(),this.second=t.s,this.minute=t.i,this.hour=t.h,this.day=t.d,this.month=t.m-1,this.year=t.y}else this.ms=i.getMilliseconds(),this.second=i.getSeconds(),this.minute=i.getMinutes(),this.hour=i.getHours(),this.day=i.getDate(),this.month=i.getMonth(),this.year=i.getFullYear()};_e.prototype.fromCronDate=function(i){this.tz=i.tz,this.year=i.year,this.month=i.month,this.day=i.day,this.hour=i.hour,this.minute=i.minute,this.second=i.second,this.ms=i.ms};_e.prototype.apply=function(){if(this.month>11||this.day>zi[this.month]||this.hour>59||this.minute>59||this.second>59||this.hour<0||this.minute<0||this.second<0){let i=new Date(Date.UTC(this.year,this.month,this.day,this.hour,this.minute,this.second,this.ms));return this.ms=i.getUTCMilliseconds(),this.second=i.getUTCSeconds(),this.minute=i.getUTCMinutes(),this.hour=i.getUTCHours(),this.day=i.getUTCDate(),this.month=i.getUTCMonth(),this.year=i.getUTCFullYear(),!0}else return!1};_e.prototype.fromString=function(i){if(typeof this.tz=="number"){let t=He.fromTZISO(i);this.ms=t.getUTCMilliseconds(),this.second=t.getUTCSeconds(),this.minute=t.getUTCMinutes(),this.hour=t.getUTCHours(),this.day=t.getUTCDate(),this.month=t.getUTCMonth(),this.year=t.getUTCFullYear(),this.apply()}else return this.fromDate(He.fromTZISO(i,this.tz))};_e.prototype.findNext=function(i,t,e,s){let n=this[t],a;e.lastDayOfMonth&&(this.month!==1?a=zi[this.month]:a=new Date(Date.UTC(this.year,this.month+1,0,0,0,0,0)).getUTCDate());let r=!e.starDOW&&t=="day"?new Date(Date.UTC(this.year,this.month,1,0,0,0,0)).getUTCDay():void 0;for(let o=this[t]+s;o<e[t].length;o++){let c=e[t][o];if(t==="day"&&e.lastDayOfMonth&&o-s==a&&(c=!0),t==="day"&&!e.starDOW){let l=e.dayOfWeek[(r+(o-s-1))%7];if(l&&l&ms)l=this.isNthWeekdayOfMonth(this.year,this.month,o-s,l);else if(l)throw new Error(`CronDate: Invalid value for dayOfWeek encountered. ${l}`);i.legacyMode&&!e.starDOM?c=c||l:c=c&&l}if(c)return this[t]=o-s,n!==this[t]?2:1}return 3};_e.prototype.recurse=function(i,t,e){let s=this.findNext(t,ft[e][0],i,ft[e][2]);if(s>1){let n=e+1;for(;n<ft.length;)this[ft[n][0]]=-ft[n][2],n++;if(s===3)return this[ft[e][1]]++,this[ft[e][0]]=-ft[e][2],this.apply(),this.recurse(i,t,0);if(this.apply())return this.recurse(i,t,e-1)}return e+=1,e>=ft.length?this:this.year>=3e3?null:this.recurse(i,t,e)};_e.prototype.increment=function(i,t,e){return this.second+=t.interval>1&&e?t.interval:1,this.ms=0,this.apply(),this.recurse(i,t,0)};_e.prototype.getDate=function(i){return i||this.tz===void 0?new Date(this.year,this.month,this.day,this.hour,this.minute,this.second,this.ms):typeof this.tz=="number"?new Date(Date.UTC(this.year,this.month,this.day,this.hour,this.minute-this.tz,this.second,this.ms)):He(this.year,this.month+1,this.day,this.hour,this.minute,this.second,this.tz)};_e.prototype.getTime=function(){return this.getDate().getTime()};function tn(i){return Object.prototype.toString.call(i)==="[object Function]"||typeof i=="function"||i instanceof Function}function Fl(i){typeof Deno<"u"&&typeof Deno.unrefTimer<"u"?Deno.unrefTimer(i):i&&typeof i.unref<"u"&&i.unref()}var Hi=30*1e3,fs=[];function pe(i,t,e){if(!(this instanceof pe))return new pe(i,t,e);let s,n;if(tn(t))n=t;else if(typeof t=="object")s=t;else if(t!==void 0)throw new Error("Cron: Invalid argument passed for optionsIn. Should be one of function, or object (options).");if(tn(e))n=e;else if(typeof e=="object")s=e;else if(e!==void 0)throw new Error("Cron: Invalid argument passed for funcIn. Should be one of function, or object (options).");if(this.name=s?s.name:void 0,this.options=Ll(s),this._states={kill:!1,blocking:!1,previousRun:void 0,currentRun:void 0,once:void 0,currentTimeout:void 0,maxRuns:s?s.maxRuns:void 0,paused:s?s.paused:!1,pattern:void 0},i&&(i instanceof Date||typeof i=="string"&&i.indexOf(":")>0)?this._states.once=new _e(i,this.options.timezone||this.options.utcOffset):this._states.pattern=new Ge(i,this.options.timezone),this.name){if(fs.find(r=>r.name===this.name))throw new Error("Cron: Tried to initialize new named job '"+this.name+"', but name already taken.");fs.push(this)}return n!==void 0&&(this.fn=n,this.schedule()),this}pe.prototype.nextRun=function(i){let t=this._next(i);return t?t.getDate():null};pe.prototype.nextRuns=function(i,t){i>this._states.maxRuns&&(i=this._states.maxRuns);let e=[],s=t||this._states.currentRun;for(;i--&&(s=this.nextRun(s));)e.push(s);return e};pe.prototype.getPattern=function(){return this._states.pattern?this._states.pattern.pattern:void 0};pe.prototype.isRunning=function(){let i=this.nextRun(this._states.currentRun),t=!this._states.paused,e=this.fn!==void 0,s=!this._states.kill;return t&&e&&s&&i!==null};pe.prototype.isStopped=function(){return this._states.kill};pe.prototype.isBusy=function(){return this._states.blocking};pe.prototype.currentRun=function(){return this._states.currentRun?this._states.currentRun.getDate():null};pe.prototype.previousRun=function(){return this._states.previousRun?this._states.previousRun.getDate():null};pe.prototype.msToNext=function(i){i=i||new Date;let t=this._next(i);return t?t.getTime()-i.getTime():null};pe.prototype.stop=function(){this._states.kill=!0,this._states.currentTimeout&&clearTimeout(this._states.currentTimeout);let i=fs.indexOf(this);i>=0&&fs.splice(i,1)};pe.prototype.pause=function(){return this._states.paused=!0,!this._states.kill};pe.prototype.resume=function(){return this._states.paused=!1,!this._states.kill};pe.prototype.schedule=function(i){if(i&&this.fn)throw new Error("Cron: It is not allowed to schedule two functions using the same Croner instance.");i&&(this.fn=i);let t=this.msToNext(),e=this.nextRun(this._states.currentRun);return t==null||isNaN(t)||e===null?this:(t>Hi&&(t=Hi),this._states.currentTimeout=setTimeout(()=>this._checkTrigger(e),t),this._states.currentTimeout&&this.options.unref&&Fl(this._states.currentTimeout),this)};pe.prototype._trigger=async function(i){if(this._states.blocking=!0,this._states.currentRun=new _e(void 0,this.options.timezone||this.options.utcOffset),this.options.catch)try{await this.fn(this,this.options.context)}catch(t){tn(this.options.catch)&&this.options.catch(t,this)}else await this.fn(this,this.options.context);this._states.previousRun=new _e(i,this.options.timezone||this.options.utcOffset),this._states.blocking=!1};pe.prototype.trigger=async function(){await this._trigger()};pe.prototype._checkTrigger=function(i){let t=new Date,e=!this._states.paused&&t.getTime()>=i,s=this._states.blocking&&this.options.protect;e&&!s?(this._states.maxRuns--,this._trigger()):e&&s&&tn(this.options.protect)&&setTimeout(()=>this.options.protect(this),0),this.schedule()};pe.prototype._next=function(i){let t=!!(i||this._states.currentRun),e=!1;!i&&this.options.startAt&&this.options.interval&&([i,t]=this._calculatePreviousRun(i,t),e=!i),i=new _e(i,this.options.timezone||this.options.utcOffset),this.options.startAt&&i&&i.getTime()<this.options.startAt.getTime()&&(i=this.options.startAt);let s=this._states.once||new _e(i,this.options.timezone||this.options.utcOffset);return!e&&s!==this._states.once&&(s=s.increment(this._states.pattern,this.options,t)),this._states.once&&this._states.once.getTime()<=i.getTime()||s===null||this._states.maxRuns<=0||this._states.kill||this.options.stopAt&&s.getTime()>=this.options.stopAt.getTime()?null:s};pe.prototype._calculatePreviousRun=function(i,t){let e=new _e(void 0,this.options.timezone||this.options.utcOffset);if(this.options.startAt.getTime()<=e.getTime()){i=this.options.startAt;let s=i.getTime()+this.options.interval*1e3;for(;s<=e.getTime();)i=new _e(i,this.options.timezone||this.options.utcOffset).increment(this._states.pattern,this.options,!0),s=i.getTime()+this.options.interval*1e3;t=!0}return[i,t]};pe.Cron=pe;pe.scheduledJobs=fs;var vr=require("obsidian");var dr=require("crypto");var Je=require("fs"),da=require("path");function Ol(i){return`mcp__${i.replace(/[\s.]+/g,"_")}`}function sn(i,t,e={}){let s=t.permissionRules.allow.length>0||t.permissionRules.deny.length>0,n=t.permissionMode?.trim(),a=!!n&&n!=="default",r=e.mcpAllowServers??[],o=r.length>0;if(!(s||a||o))return null;let l=(0,da.join)(i,".claude"),h=(0,da.join)(l,"settings.local.json");(0,Je.existsSync)(l)||(0,Je.mkdirSync)(l,{recursive:!0});let d=null;if((0,Je.existsSync)(h))try{d=(0,Je.readFileSync)(h,"utf-8")}catch{d=null}let u={};if(a&&(u.defaultMode=n),s||o){let p=[...t.permissionRules.allow];for(let m of r)p.push(Ol(m));u.permissions={allow:p,deny:t.permissionRules.deny}}return(0,Je.writeFileSync)(h,JSON.stringify(u,null,2)+`
11864
- `,"utf-8"),{path:h,backupContent:d}}function Rt(i){if(i)try{i.backupContent!==null?(0,Je.writeFileSync)(i.path,i.backupContent,"utf-8"):(0,Je.existsSync)(i.path)&&(0,Je.unlinkSync)(i.path)}catch{}}function Gi(i){if(typeof i=="string")return i;if(Array.isArray(i))return i.map(e=>{if(typeof e=="string")return e;if(e&&typeof e=="object"&&"text"in e){let s=e.text;if(typeof s=="string")return s}return""}).filter(Boolean).join(`
11865
- `);if(i&&typeof i=="object"){for(let t of["output","result","text","message"])if(t in i)return Gi(i[t])}}function ha(i,t=[]){if(Array.isArray(i)){for(let r of i)ha(r,t);return t}if(!i||typeof i!="object")return t;let e=i,s=typeof e.tool_name=="string"&&e.tool_name||typeof e.tool=="string"&&e.tool||typeof e.name=="string"&&e.name,n=typeof e.command=="string"?e.command:typeof e.input=="string"?e.input:typeof e.cmd=="string"?e.cmd:void 0,a=typeof e.reason=="string"?e.reason:void 0;s&&["tool_use","tool","name","tool_name"].some(r=>r in e)&&t.push({tool:s,command:n,reason:a});for(let r of Object.values(e))ha(r,t);return t}function Vi(i){if(!i||typeof i!="object")return;let t=i,e=t.usage;if(e&&typeof e=="object"){let n=typeof e.input_tokens=="number"?e.input_tokens:0,a=typeof e.output_tokens=="number"?e.output_tokens:0,r=typeof e.cache_creation_input_tokens=="number"?e.cache_creation_input_tokens:0,o=typeof e.cache_read_input_tokens=="number"?e.cache_read_input_tokens:0,c=n+a+r+o;if(c>0)return c}let s=t.modelUsage;if(s&&typeof s=="object"){let n=0;for(let a of Object.values(s)){if(!a||typeof a!="object")continue;let r=a;n+=typeof r.inputTokens=="number"?r.inputTokens:0,n+=typeof r.outputTokens=="number"?r.outputTokens:0,n+=typeof r.cacheReadInputTokens=="number"?r.cacheReadInputTokens:0,n+=typeof r.cacheCreationInputTokens=="number"?r.cacheCreationInputTokens:0}if(n>0)return n}for(let n of["tokens_used","total_tokens","totalTokens"])if(typeof t[n]=="number")return t[n];for(let n of Object.values(t)){let a=Vi(n);if(typeof a=="number")return a}}function Yi(i){if(!i||typeof i!="object")return;let t=i;if(typeof t.total_cost_usd=="number")return t.total_cost_usd;for(let e of Object.values(t)){let s=Yi(e);if(typeof s=="number")return s}}function ua(i){if(!i||typeof i!="object")return;let t=i;if(t.type==="result"&&typeof t.result=="string"&&t.result.trim())return t.result}function nn(i){if(!i||typeof i!="object")return;let t=i;if(t.modelUsage&&typeof t.modelUsage=="object"){let s=Object.keys(t.modelUsage);if(s.length>0&&s[0])return s[0]}let e=t.message;if(e&&typeof e.model=="string"&&e.model)return e.model;if(typeof t.model=="string"&&t.model)return t.model;for(let s of Object.values(t)){let n=nn(s);if(n)return n}}function Nl(i){return/^gpt-|codex/i.test(i.trim())}var Ki={id:"claude-code",label:"Claude Code",cliPath(i){return i.claudeCliPath},buildExec(i){let t=["-p",i.prompt,"--output-format",i.streaming?"stream-json":"json"];i.streaming&&t.push("--verbose");let e=i.modelSource==="settings"&&Nl(i.model);i.model&&!e&&t.push("--model",i.model),i.effort&&t.push("--effort",i.effort);let s=i.agent.permissionMode?.trim();return s&&s!=="default"?t.push("--permission-mode",s):t.push("--permission-mode","bypassPermissions"),Promise.resolve({cliPath:i.settings.claudeCliPath,args:t})},parseExecOutput(i,t,e){let s=i.trim(),n;if(e){let c=ye(s);for(let l=c.length-1;l>=0;l--){let h=c[l]?.trim();if(h)try{let d=JSON.parse(h);if(d&&typeof d=="object"){n=d;break}}catch{}}}else if(s.startsWith("{")||s.startsWith("["))try{n=JSON.parse(s)}catch{n=void 0}let a=Gi(n)??"";if(!a&&e){let c=[];for(let l of ye(s)){let h=l.trim();if(h)try{let d=JSON.parse(h);if(d.type==="assistant"&&d.message?.content)for(let u of d.message.content)u.type==="text"&&u.text&&c.push(u.text);else d.type==="result"&&typeof d.result=="string"&&c.push(d.result)}catch{}}a=c.join(`
11866
- `).trim()}a||(a=t.trim()||"(no output)");let r=nn(n),o=ua(n);if((!r||!o)&&e)for(let c of ye(s)){let l=c.trim();if(l)try{let h=JSON.parse(l);if(!r){let d=nn(h);d&&(r=d)}if(!o){let d=ua(h);d&&(o=d)}if(r&&o)break}catch{}}return{outputText:a,finalResult:o,tokensUsed:Vi(n),costUsd:Yi(n),toolsUsed:ha(n),concreteModel:r,rawJson:n}},extractStreamChunk(i){let t=i.trim();if(!t)return null;let e;try{e=JSON.parse(t)}catch{return null}let s=e.type;if(s==="assistant"){let n=e.message;if(n?.content&&Array.isArray(n.content)){let a=[];for(let r of n.content)if(r.type==="text"&&typeof r.text=="string")a.push(r.text);else if(r.type==="tool_use"){let o=String(r.name??"tool"),c=r.input,l=c?.command??c?.content??"";a.push(`
11774
+ Your scope's inbox, topics, index, and log are NOT deleted.`,confirmText:"Delete",danger:!0,onConfirm:async()=>{try{await Hr(this.plugin.app,this.plugin.settings.fleetFolder,o.name),await this.plugin.refreshFromVault(),new F.Notice(`Wiki Keeper "${o.name}" deleted.`),this.scheduleRerender()}catch(m){let f=m instanceof Error?m.message:String(m);new F.Notice(`Failed to delete: ${f}`)}}}).open()}}new F.Setting(e).setName("Add Wiki Keeper").setDesc("Create a new scoped wiki agent. All fields optional.").addButton(o=>o.setButtonText("+ Add").onClick(()=>{new ia(this.plugin,()=>{this.scheduleRerender()}).open()}))}scheduleRerender(){window.setTimeout(()=>{(async()=>{try{await this.plugin.refreshFromVault()}catch{}this.display()})()},600)}},ia=class extends F.Modal{constructor(e,s){super(e.app);this.plugin=e;this.onCreated=s;this.input=na()}input;onOpen(){let{contentEl:e}=this;e.empty(),e.addClass("af-wk-modal"),e.createEl("h2",{text:"Add Wiki Keeper"});let s=e.createDiv({cls:"af-wk-banner"});s.createEl("strong",{text:"All fields are optional. "}),s.createSpan({text:"Click Create with everything blank and you'll get a whole-vault keeper with sensible defaults (inbox at _sources/inbox, topics at topics/, nightly ingest at 3am, Sunday lint)."}),e.createEl("h3",{text:"Scope",cls:"af-wk-section-h"}),new F.Setting(e).setName("Scope folder").setDesc("Optional. Vault-relative path. Empty = whole vault.").addText(h=>h.setPlaceholder("(empty = whole vault)").setValue(this.input.scopeRoot).onChange(d=>{this.input.scopeRoot=d.trim(),this.renderPreview()})),new F.Setting(e).setName("Slug").setDesc("Optional. Agent name suffix. Default: derived from scope folder name, or 'wiki-keeper' for whole-vault.").addText(h=>h.setPlaceholder("(auto)").setValue(this.input.scopeSlug).onChange(d=>{this.input.scopeSlug=d.trim(),this.renderPreview()})),this.renderPreview(),e.createEl("h3",{text:"Sources",cls:"af-wk-section-h"}),new F.Setting(e).setName("Watched folders").setDesc("Optional. Comma- or newline-separated vault-relative paths. Read-only \u2014 claims are extracted but files are never moved. Leave empty for inbox-only mode (you drop files into _sources/inbox/ and they get archived after processing).").addTextArea(h=>h.setPlaceholder("(empty = inbox-only)").setValue(this.input.watchedFolders.join(", ")).onChange(d=>{this.input.watchedFolders=d.split(/[,\n]/).map(u=>u.trim()).filter(Boolean)})),new F.Setting(e).setName("Exclude patterns").setDesc("Optional. Glob patterns to skip. Leave empty to process everything under watched + inbox.").addTextArea(h=>h.setPlaceholder("(empty = no excludes)").setValue(this.input.excludePatterns.join(", ")).onChange(d=>{this.input.excludePatterns=d.split(/[,\n]/).map(u=>u.trim()).filter(Boolean)})),new F.Setting(e).setName("Watch files modified since").setDesc("Watched mode only \u2014 files whose last modification date is before this are skipped. Defaults to today so an established vault doesn't flood the keeper on first run. Clear the field to process everything.").addText(h=>{h.inputEl.type="date",h.setValue(this.input.watchedSince).onChange(d=>{this.input.watchedSince=d.trim()})});let n=this.plugin.runtime.getSnapshot().channels.map(h=>h.name);new F.Setting(e).setName("Heartbeat channel").setDesc("Optional. Slack/Telegram channel for ingest + lint summaries. Leave as (none) to disable.").addDropdown(h=>{h.addOption("","(none)");for(let d of n)h.addOption(d,d);h.setValue(this.input.heartbeatChannel).onChange(d=>{this.input.heartbeatChannel=d})}),e.createEl("h3",{text:"Advanced",cls:"af-wk-section-h"}),new F.Setting(e).setName("Max tokens per ingest").setDesc("Hard cap on token spend per run. Unprocessed files resume next cycle.").addText(h=>h.setValue(String(this.input.maxTokensPerIngest)).onChange(d=>{let u=parseInt(d,10);!isNaN(u)&&u>0&&(this.input.maxTokensPerIngest=u)})),new F.Setting(e).setName("File substantive answers").setDesc("If on, wiki-query files long answers under _topics/syntheses/. Off = answers live only in chat.").addToggle(h=>h.setValue(this.input.fileSubstantiveAnswers).onChange(d=>{this.input.fileSubstantiveAnswers=d})),new F.Setting(e).setName("Obsidian URL scheme for citations").setDesc("If on, external-channel (Slack/Telegram) replies rewrite [[wikilinks]] as obsidian:// URLs.").addToggle(h=>h.setValue(this.input.obsidianUrlScheme).onChange(d=>{this.input.obsidianUrlScheme=d}));let a=e.createEl("h3",{text:"Paths (expert)",cls:"af-wk-section-h"});a.title="Paths are relative to scope_root. Defaults work for almost everyone. Override only if your vault layout demands it \u2014 these cannot be changed after creation.";let i=(h,d,u)=>{new F.Setting(e).setName(h).setDesc(d).addText(p=>p.setPlaceholder(String(this.input[u]??"")).setValue(String(this.input[u]??"")).onChange(m=>{this.input[u]=m.trim()||Br[u]}))};i("Inbox path","Where one-off source drops land.","inboxPath"),i("Archive path","Where processed inbox files are moved after summarization.","archivePath"),i("Topics root","Folder under scope_root that holds the generated topic pages.","topicsRoot"),i("Index path","Relative file path for the curated index.","indexPath"),i("Log path","Relative file path for the keeper's activity log.","logPath"),i("State file","Hidden JSON file that tracks watched-source mtimes.","stateFile");let o=e.createDiv({cls:"af-wk-modal-footer"}),c=o.createEl("button",{text:"Cancel"});c.onclick=()=>this.close();let l=o.createEl("button",{text:"Create",cls:"mod-cta"});l.onclick=async()=>{l.setText("Creating\u2026"),l.disabled=!0;try{let{name:h}=await jr(this.app.vault,this.plugin.settings.fleetFolder,this.input);await this.plugin.refreshFromVault(),this.close(),new F.Notice(`Wiki Keeper "${h}" created.`),this.onCreated()}catch(h){let d=h instanceof Error?h.message:String(h);new F.Notice(`Failed to create Wiki Keeper: ${d}`),l.setText("Create"),l.disabled=!1}}}renderPreview(){let e=this.contentEl.querySelector(".af-wk-name-preview");e||(e=this.contentEl.createDiv({cls:"af-wk-name-preview"}));let s=this.input.scopeSlug||this.input.scopeRoot;e.setText(`\u2192 Will create agent: ${aa(s)}`)}onClose(){this.contentEl.empty()}},oa=class extends F.Modal{constructor(e,s,n){super(e.app);this.plugin=e;this.agent=s;this.onSaved=n;let a=s.wikiKeeper;this.edit={watchedFolders:[...a.watchedFolders],excludePatterns:[...a.excludePatterns],watchedSince:a.watchedSince,heartbeatChannel:s.heartbeatChannel??"",fileSubstantiveAnswers:a.fileSubstantiveAnswers,obsidianUrlScheme:a.obsidianUrlScheme,maxTokensPerIngest:a.maxTokensPerIngest,maxTokensPerRefresh:a.maxTokensPerRefresh,indexSplitThreshold:a.indexSplitThreshold,dedupSimilarityThreshold:a.dedupSimilarityThreshold,summaryStaleDays:a.summaryStaleDays}}edit;onOpen(){let{contentEl:e}=this;e.empty(),e.addClass("af-wk-modal"),e.createEl("h2",{text:`Edit ${this.agent.name}`});let s=e.createDiv({cls:"af-wk-banner"}),n=this.agent.wikiKeeper.scopeRoot||"(whole vault)";s.createEl("strong",{text:"Scope: "}),s.createSpan({text:n}),s.createEl("br"),s.createSpan({text:"Scope and paths are fixed after creation \u2014 changing them would orphan existing topic pages. To move a Wiki Keeper, delete this one and create a new one at the new scope."}),e.createEl("h3",{text:"Sources",cls:"af-wk-section-h"}),new F.Setting(e).setName("Watched folders").setDesc("Comma- or newline-separated vault-relative paths.").addTextArea(l=>l.setValue(this.edit.watchedFolders.join(", ")).onChange(h=>{this.edit.watchedFolders=h.split(/[,\n]/).map(d=>d.trim()).filter(Boolean)})),new F.Setting(e).setName("Exclude patterns").setDesc("Glob patterns to skip.").addTextArea(l=>l.setValue(this.edit.excludePatterns.join(", ")).onChange(h=>{this.edit.excludePatterns=h.split(/[,\n]/).map(d=>d.trim()).filter(Boolean)})),new F.Setting(e).setName("Watch files modified since").setDesc("Watched mode skips files older than this date. Clear to process everything.").addText(l=>{l.inputEl.type="date",l.setValue(this.edit.watchedSince).onChange(h=>{this.edit.watchedSince=h.trim()})});let a=this.plugin.runtime.getSnapshot().channels.map(l=>l.name);new F.Setting(e).setName("Heartbeat channel").addDropdown(l=>{l.addOption("","(none)");for(let h of a)l.addOption(h,h);l.setValue(this.edit.heartbeatChannel).onChange(h=>{this.edit.heartbeatChannel=h})}),e.createEl("h3",{text:"Advanced",cls:"af-wk-section-h"}),new F.Setting(e).setName("Max tokens per ingest").addText(l=>l.setValue(String(this.edit.maxTokensPerIngest)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.maxTokensPerIngest=d)})),new F.Setting(e).setName("Max tokens per refresh").setDesc("Separate budget for the synthesis-refresh phase that regenerates topic-page summaries.").addText(l=>l.setValue(String(this.edit.maxTokensPerRefresh)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.maxTokensPerRefresh=d)})),new F.Setting(e).setName("Index split threshold").setDesc("Topic-page count above which index.md splits into per-type sub-MOCs.").addText(l=>l.setValue(String(this.edit.indexSplitThreshold)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.indexSplitThreshold=d)})),new F.Setting(e).setName("Summary stale (days)").setDesc("Lint flags topic-page summaries older than this and chains wiki-refresh.").addText(l=>l.setValue(String(this.edit.summaryStaleDays)).onChange(h=>{let d=parseInt(h,10);!isNaN(d)&&d>0&&(this.edit.summaryStaleDays=d)})),new F.Setting(e).setName("Dedup similarity threshold").setDesc("Levenshtein-ratio above which lint proposes merging two same-type pages (0.0\u20131.0).").addText(l=>l.setValue(String(this.edit.dedupSimilarityThreshold)).onChange(h=>{let d=parseFloat(h);!isNaN(d)&&d>0&&d<=1&&(this.edit.dedupSimilarityThreshold=d)})),new F.Setting(e).setName("File substantive answers").setDesc("Compounding: wiki-query files long answers under _topics/syntheses/ and bullets each cited topic page. Default ON.").addToggle(l=>l.setValue(this.edit.fileSubstantiveAnswers).onChange(h=>{this.edit.fileSubstantiveAnswers=h})),new F.Setting(e).setName("Obsidian URL scheme for citations").setDesc("Rewrite [[wikilinks]] as obsidian:// URLs when replying via Slack/Telegram.").addToggle(l=>l.setValue(this.edit.obsidianUrlScheme).onChange(h=>{this.edit.obsidianUrlScheme=h}));let i=e.createDiv({cls:"af-wk-modal-footer"}),o=i.createEl("button",{text:"Cancel"});o.onclick=()=>this.close();let c=i.createEl("button",{text:"Save",cls:"mod-cta"});c.onclick=async()=>{c.setText("Saving\u2026"),c.disabled=!0;try{await Wr(this.app.vault,this.plugin.settings.fleetFolder,this.agent.name,this.edit),await this.plugin.refreshFromVault(),this.close(),new F.Notice("Saved."),this.onSaved()}catch(l){let h=l instanceof Error?l.message:String(l);new F.Notice(`Failed to save: ${h}`),c.setText("Save"),c.disabled=!1}}}onClose(){this.contentEl.empty()}};function Il(r){return r.type==="slack",r.botToken}function Ml(r){return r.length<=10?"***":`${r.slice(0,6)}\u2026${r.slice(-4)}`}var Sa=require("crypto");function He(r,t,e,s,n,a,i,o){return He.fromTZ(He.tp(r,t,e,s,n,a,i),o)}He.fromTZISO=(r,t,e)=>He.fromTZ(Ll(r,t),e);He.fromTZ=function(r,t){let e=new Date(Date.UTC(r.y,r.m-1,r.d,r.h,r.i,r.s)),s=la(r.tz,e),n=new Date(e.getTime()-s),a=la(r.tz,n);if(a-s===0)return n;{let i=new Date(e.getTime()-a),o=la(r.tz,i);if(o-a===0)return i;if(!t&&o-a>0)return i;if(t)throw new Error("Invalid date passed to fromTZ()");return n}};He.toTZ=function(r,t){let e=r.toLocaleString("en-US",{timeZone:t}).replace(/[\u202f]/," "),s=new Date(e);return{y:s.getFullYear(),m:s.getMonth()+1,d:s.getDate(),h:s.getHours(),i:s.getMinutes(),s:s.getSeconds(),tz:t}};He.tp=(r,t,e,s,n,a,i)=>({y:r,m:t,d:e,h:s,i:n,s:a,tz:i});function la(r,t=new Date){let e=t.toLocaleString("en-US",{timeZone:r,timeZoneName:"shortOffset"}).split(" ").slice(-1)[0],s=t.toLocaleString("en-US").replace(/[\u202f]/," ");return Date.parse(`${s} GMT`)-Date.parse(`${s} ${e}`)}function Ll(r,t){let e=new Date(Date.parse(r));if(isNaN(e))throw new Error("minitz: Invalid ISO8601 passed to parser.");let s=r.substring(9);return r.includes("Z")||s.includes("-")||s.includes("+")?He.tp(e.getUTCFullYear(),e.getUTCMonth()+1,e.getUTCDate(),e.getUTCHours(),e.getUTCMinutes(),e.getUTCSeconds(),"Etc/UTC"):He.tp(e.getFullYear(),e.getMonth()+1,e.getDate(),e.getHours(),e.getMinutes(),e.getSeconds(),t)}He.minitz=He;function Fl(r){if(r===void 0&&(r={}),delete r.name,r.legacyMode=r.legacyMode===void 0?!0:r.legacyMode,r.paused=r.paused===void 0?!1:r.paused,r.maxRuns=r.maxRuns===void 0?1/0:r.maxRuns,r.catch=r.catch===void 0?!1:r.catch,r.interval=r.interval===void 0?0:parseInt(r.interval,10),r.utcOffset=r.utcOffset===void 0?void 0:parseInt(r.utcOffset,10),r.unref=r.unref===void 0?!1:r.unref,r.startAt&&(r.startAt=new _e(r.startAt,r.timezone)),r.stopAt&&(r.stopAt=new _e(r.stopAt,r.timezone)),r.interval!==null){if(isNaN(r.interval))throw new Error("CronOptions: Supplied value for interval is not a number");if(r.interval<0)throw new Error("CronOptions: Supplied value for interval can not be negative")}if(r.utcOffset!==void 0){if(isNaN(r.utcOffset))throw new Error("CronOptions: Invalid value passed for utcOffset, should be number representing minutes offset from UTC.");if(r.utcOffset<-870||r.utcOffset>870)throw new Error("CronOptions: utcOffset out of bounds.");if(r.utcOffset!==void 0&&r.timezone)throw new Error("CronOptions: Combining 'utcOffset' with 'timezone' is not allowed.")}if(r.unref!==!0&&r.unref!==!1)throw new Error("CronOptions: Unref should be either true, false or undefined(false).");return r}var ca=32,ms=31|ca,zr=[1,2,4,8,16];function Ge(r,t){this.pattern=r,this.timezone=t,this.second=Array(60).fill(0),this.minute=Array(60).fill(0),this.hour=Array(24).fill(0),this.day=Array(31).fill(0),this.month=Array(12).fill(0),this.dayOfWeek=Array(7).fill(0),this.lastDayOfMonth=!1,this.starDOM=!1,this.starDOW=!1,this.parse()}Ge.prototype.parse=function(){if(!(typeof this.pattern=="string"||this.pattern.constructor===String))throw new TypeError("CronPattern: Pattern has to be of type string.");this.pattern.indexOf("@")>=0&&(this.pattern=this.handleNicknames(this.pattern).trim());let r=this.pattern.replace(/\s+/g," ").split(" ");if(r.length<5||r.length>6)throw new TypeError("CronPattern: invalid configuration format ('"+this.pattern+"'), exactly five or six space separated parts are required.");if(r.length===5&&r.unshift("0"),r[3].indexOf("L")>=0&&(r[3]=r[3].replace("L",""),this.lastDayOfMonth=!0),r[3]=="*"&&(this.starDOM=!0),r[4].length>=3&&(r[4]=this.replaceAlphaMonths(r[4])),r[5].length>=3&&(r[5]=this.replaceAlphaDays(r[5])),r[5]=="*"&&(this.starDOW=!0),this.pattern.indexOf("?")>=0){let t=new _e(new Date,this.timezone).getDate(!0);r[0]=r[0].replace("?",t.getSeconds()),r[1]=r[1].replace("?",t.getMinutes()),r[2]=r[2].replace("?",t.getHours()),this.starDOM||(r[3]=r[3].replace("?",t.getDate())),r[4]=r[4].replace("?",t.getMonth()+1),this.starDOW||(r[5]=r[5].replace("?",t.getDay()))}this.throwAtIllegalCharacters(r),this.partToArray("second",r[0],0,1),this.partToArray("minute",r[1],0,1),this.partToArray("hour",r[2],0,1),this.partToArray("day",r[3],-1,1),this.partToArray("month",r[4],-1,1),this.partToArray("dayOfWeek",r[5],0,ms),this.dayOfWeek[7]&&(this.dayOfWeek[0]=this.dayOfWeek[7])};Ge.prototype.partToArray=function(r,t,e,s){let n=this[r],a=r==="day"&&this.lastDayOfMonth;if(t===""&&!a)throw new TypeError("CronPattern: configuration entry "+r+" ("+t+") is empty, check for trailing spaces.");if(t==="*")return n.fill(s);let i=t.split(",");if(i.length>1)for(let o=0;o<i.length;o++)this.partToArray(r,i[o],e,s);else t.indexOf("-")!==-1&&t.indexOf("/")!==-1?this.handleRangeWithStepping(t,r,e,s):t.indexOf("-")!==-1?this.handleRange(t,r,e,s):t.indexOf("/")!==-1?this.handleStepping(t,r,e,s):t!==""&&this.handleNumber(t,r,e,s)};Ge.prototype.throwAtIllegalCharacters=function(r){for(let t=0;t<r.length;t++)if((t===5?/[^/*0-9,\-#L]+/:/[^/*0-9,-]+/).test(r[t]))throw new TypeError("CronPattern: configuration entry "+t+" ("+r[t]+") contains illegal characters.")};Ge.prototype.handleNumber=function(r,t,e,s){let n=this.extractNth(r,t),a=parseInt(n[0],10)+e;if(isNaN(a))throw new TypeError("CronPattern: "+t+" is not a number: '"+r+"'");this.setPart(t,a,n[1]||s)};Ge.prototype.setPart=function(r,t,e){if(!Object.prototype.hasOwnProperty.call(this,r))throw new TypeError("CronPattern: Invalid part specified: "+r);if(r==="dayOfWeek"){if(t===7&&(t=0),(t<0||t>6)&&t!=="L")throw new RangeError("CronPattern: Invalid value for dayOfWeek: "+t);this.setNthWeekdayOfMonth(t,e);return}if(r==="second"||r==="minute"){if(t<0||t>=60)throw new RangeError("CronPattern: Invalid value for "+r+": "+t)}else if(r==="hour"){if(t<0||t>=24)throw new RangeError("CronPattern: Invalid value for "+r+": "+t)}else if(r==="day"){if(t<0||t>=31)throw new RangeError("CronPattern: Invalid value for "+r+": "+t)}else if(r==="month"&&(t<0||t>=12))throw new RangeError("CronPattern: Invalid value for "+r+": "+t);this[r][t]=e};Ge.prototype.handleRangeWithStepping=function(r,t,e,s){let n=this.extractNth(r,t),a=n[0].match(/^(\d+)-(\d+)\/(\d+)$/);if(a===null)throw new TypeError("CronPattern: Syntax error, illegal range with stepping: '"+r+"'");let[,i,o,c]=a;if(i=parseInt(i,10)+e,o=parseInt(o,10)+e,c=parseInt(c,10),isNaN(i))throw new TypeError("CronPattern: Syntax error, illegal lower range (NaN)");if(isNaN(o))throw new TypeError("CronPattern: Syntax error, illegal upper range (NaN)");if(isNaN(c))throw new TypeError("CronPattern: Syntax error, illegal stepping: (NaN)");if(c===0)throw new TypeError("CronPattern: Syntax error, illegal stepping: 0");if(c>this[t].length)throw new TypeError("CronPattern: Syntax error, steps cannot be greater than maximum value of part ("+this[t].length+")");if(i>o)throw new TypeError("CronPattern: From value is larger than to value: '"+r+"'");for(let l=i;l<=o;l+=c)this.setPart(t,l,n[1]||s)};Ge.prototype.extractNth=function(r,t){let e=r,s;if(e.includes("#")){if(t!=="dayOfWeek")throw new Error("CronPattern: nth (#) only allowed in day-of-week field");s=e.split("#")[1],e=e.split("#")[0]}return[e,s]};Ge.prototype.handleRange=function(r,t,e,s){let n=this.extractNth(r,t),a=n[0].split("-");if(a.length!==2)throw new TypeError("CronPattern: Syntax error, illegal range: '"+r+"'");let i=parseInt(a[0],10)+e,o=parseInt(a[1],10)+e;if(isNaN(i))throw new TypeError("CronPattern: Syntax error, illegal lower range (NaN)");if(isNaN(o))throw new TypeError("CronPattern: Syntax error, illegal upper range (NaN)");if(i>o)throw new TypeError("CronPattern: From value is larger than to value: '"+r+"'");for(let c=i;c<=o;c++)this.setPart(t,c,n[1]||s)};Ge.prototype.handleStepping=function(r,t,e,s){let n=this.extractNth(r,t),a=n[0].split("/");if(a.length!==2)throw new TypeError("CronPattern: Syntax error, illegal stepping: '"+r+"'");let i=0;a[0]!=="*"&&(i=parseInt(a[0],10)+e);let o=parseInt(a[1],10);if(isNaN(o))throw new TypeError("CronPattern: Syntax error, illegal stepping: (NaN)");if(o===0)throw new TypeError("CronPattern: Syntax error, illegal stepping: 0");if(o>this[t].length)throw new TypeError("CronPattern: Syntax error, max steps for part is ("+this[t].length+")");for(let c=i;c<this[t].length;c+=o)this.setPart(t,c,n[1]||s)};Ge.prototype.replaceAlphaDays=function(r){return r.replace(/-sun/gi,"-7").replace(/sun/gi,"0").replace(/mon/gi,"1").replace(/tue/gi,"2").replace(/wed/gi,"3").replace(/thu/gi,"4").replace(/fri/gi,"5").replace(/sat/gi,"6")};Ge.prototype.replaceAlphaMonths=function(r){return r.replace(/jan/gi,"1").replace(/feb/gi,"2").replace(/mar/gi,"3").replace(/apr/gi,"4").replace(/may/gi,"5").replace(/jun/gi,"6").replace(/jul/gi,"7").replace(/aug/gi,"8").replace(/sep/gi,"9").replace(/oct/gi,"10").replace(/nov/gi,"11").replace(/dec/gi,"12")};Ge.prototype.handleNicknames=function(r){let t=r.trim().toLowerCase();return t==="@yearly"||t==="@annually"?"0 0 1 1 *":t==="@monthly"?"0 0 1 * *":t==="@weekly"?"0 0 * * 0":t==="@daily"?"0 0 * * *":t==="@hourly"?"0 * * * *":r};Ge.prototype.setNthWeekdayOfMonth=function(r,t){if(t==="L")this.dayOfWeek[r]=this.dayOfWeek[r]|ca;else if(t<6&&t>0)this.dayOfWeek[r]=this.dayOfWeek[r]|zr[t-1];else if(t===ms)this.dayOfWeek[r]=ms;else throw new TypeError(`CronPattern: nth weekday of of range, should be 1-5 or L. Value: ${t}`)};var Gr=[31,28,31,30,31,30,31,31,30,31,30,31],ft=[["month","year",0],["day","month",-1],["hour","day",0],["minute","hour",0],["second","minute",0]];function _e(r,t){if(this.tz=t,r&&r instanceof Date)if(!isNaN(r))this.fromDate(r);else throw new TypeError("CronDate: Invalid date passed to CronDate constructor");else if(r===void 0)this.fromDate(new Date);else if(r&&typeof r=="string")this.fromString(r);else if(r instanceof _e)this.fromCronDate(r);else throw new TypeError("CronDate: Invalid type ("+typeof r+") passed to CronDate constructor")}_e.prototype.isNthWeekdayOfMonth=function(r,t,e,s){let a=new Date(Date.UTC(r,t,e)).getUTCDay(),i=0;for(let o=1;o<=e;o++)new Date(Date.UTC(r,t,o)).getUTCDay()===a&&i++;if(s&ms&&zr[i-1]&s)return!0;if(s&ca){let o=new Date(Date.UTC(r,t+1,0)).getUTCDate();for(let c=e+1;c<=o;c++)if(new Date(Date.UTC(r,t,c)).getUTCDay()===a)return!1;return!0}return!1};_e.prototype.fromDate=function(r){if(this.tz!==void 0)if(typeof this.tz=="number")this.ms=r.getUTCMilliseconds(),this.second=r.getUTCSeconds(),this.minute=r.getUTCMinutes()+this.tz,this.hour=r.getUTCHours(),this.day=r.getUTCDate(),this.month=r.getUTCMonth(),this.year=r.getUTCFullYear(),this.apply();else{let t=He.toTZ(r,this.tz);this.ms=r.getMilliseconds(),this.second=t.s,this.minute=t.i,this.hour=t.h,this.day=t.d,this.month=t.m-1,this.year=t.y}else this.ms=r.getMilliseconds(),this.second=r.getSeconds(),this.minute=r.getMinutes(),this.hour=r.getHours(),this.day=r.getDate(),this.month=r.getMonth(),this.year=r.getFullYear()};_e.prototype.fromCronDate=function(r){this.tz=r.tz,this.year=r.year,this.month=r.month,this.day=r.day,this.hour=r.hour,this.minute=r.minute,this.second=r.second,this.ms=r.ms};_e.prototype.apply=function(){if(this.month>11||this.day>Gr[this.month]||this.hour>59||this.minute>59||this.second>59||this.hour<0||this.minute<0||this.second<0){let r=new Date(Date.UTC(this.year,this.month,this.day,this.hour,this.minute,this.second,this.ms));return this.ms=r.getUTCMilliseconds(),this.second=r.getUTCSeconds(),this.minute=r.getUTCMinutes(),this.hour=r.getUTCHours(),this.day=r.getUTCDate(),this.month=r.getUTCMonth(),this.year=r.getUTCFullYear(),!0}else return!1};_e.prototype.fromString=function(r){if(typeof this.tz=="number"){let t=He.fromTZISO(r);this.ms=t.getUTCMilliseconds(),this.second=t.getUTCSeconds(),this.minute=t.getUTCMinutes(),this.hour=t.getUTCHours(),this.day=t.getUTCDate(),this.month=t.getUTCMonth(),this.year=t.getUTCFullYear(),this.apply()}else return this.fromDate(He.fromTZISO(r,this.tz))};_e.prototype.findNext=function(r,t,e,s){let n=this[t],a;e.lastDayOfMonth&&(this.month!==1?a=Gr[this.month]:a=new Date(Date.UTC(this.year,this.month+1,0,0,0,0,0)).getUTCDate());let i=!e.starDOW&&t=="day"?new Date(Date.UTC(this.year,this.month,1,0,0,0,0)).getUTCDay():void 0;for(let o=this[t]+s;o<e[t].length;o++){let c=e[t][o];if(t==="day"&&e.lastDayOfMonth&&o-s==a&&(c=!0),t==="day"&&!e.starDOW){let l=e.dayOfWeek[(i+(o-s-1))%7];if(l&&l&ms)l=this.isNthWeekdayOfMonth(this.year,this.month,o-s,l);else if(l)throw new Error(`CronDate: Invalid value for dayOfWeek encountered. ${l}`);r.legacyMode&&!e.starDOM?c=c||l:c=c&&l}if(c)return this[t]=o-s,n!==this[t]?2:1}return 3};_e.prototype.recurse=function(r,t,e){let s=this.findNext(t,ft[e][0],r,ft[e][2]);if(s>1){let n=e+1;for(;n<ft.length;)this[ft[n][0]]=-ft[n][2],n++;if(s===3)return this[ft[e][1]]++,this[ft[e][0]]=-ft[e][2],this.apply(),this.recurse(r,t,0);if(this.apply())return this.recurse(r,t,e-1)}return e+=1,e>=ft.length?this:this.year>=3e3?null:this.recurse(r,t,e)};_e.prototype.increment=function(r,t,e){return this.second+=t.interval>1&&e?t.interval:1,this.ms=0,this.apply(),this.recurse(r,t,0)};_e.prototype.getDate=function(r){return r||this.tz===void 0?new Date(this.year,this.month,this.day,this.hour,this.minute,this.second,this.ms):typeof this.tz=="number"?new Date(Date.UTC(this.year,this.month,this.day,this.hour,this.minute-this.tz,this.second,this.ms)):He(this.year,this.month+1,this.day,this.hour,this.minute,this.second,this.tz)};_e.prototype.getTime=function(){return this.getDate().getTime()};function tn(r){return Object.prototype.toString.call(r)==="[object Function]"||typeof r=="function"||r instanceof Function}function Ol(r){typeof Deno<"u"&&typeof Deno.unrefTimer<"u"?Deno.unrefTimer(r):r&&typeof r.unref<"u"&&r.unref()}var qr=30*1e3,fs=[];function pe(r,t,e){if(!(this instanceof pe))return new pe(r,t,e);let s,n;if(tn(t))n=t;else if(typeof t=="object")s=t;else if(t!==void 0)throw new Error("Cron: Invalid argument passed for optionsIn. Should be one of function, or object (options).");if(tn(e))n=e;else if(typeof e=="object")s=e;else if(e!==void 0)throw new Error("Cron: Invalid argument passed for funcIn. Should be one of function, or object (options).");if(this.name=s?s.name:void 0,this.options=Fl(s),this._states={kill:!1,blocking:!1,previousRun:void 0,currentRun:void 0,once:void 0,currentTimeout:void 0,maxRuns:s?s.maxRuns:void 0,paused:s?s.paused:!1,pattern:void 0},r&&(r instanceof Date||typeof r=="string"&&r.indexOf(":")>0)?this._states.once=new _e(r,this.options.timezone||this.options.utcOffset):this._states.pattern=new Ge(r,this.options.timezone),this.name){if(fs.find(i=>i.name===this.name))throw new Error("Cron: Tried to initialize new named job '"+this.name+"', but name already taken.");fs.push(this)}return n!==void 0&&(this.fn=n,this.schedule()),this}pe.prototype.nextRun=function(r){let t=this._next(r);return t?t.getDate():null};pe.prototype.nextRuns=function(r,t){r>this._states.maxRuns&&(r=this._states.maxRuns);let e=[],s=t||this._states.currentRun;for(;r--&&(s=this.nextRun(s));)e.push(s);return e};pe.prototype.getPattern=function(){return this._states.pattern?this._states.pattern.pattern:void 0};pe.prototype.isRunning=function(){let r=this.nextRun(this._states.currentRun),t=!this._states.paused,e=this.fn!==void 0,s=!this._states.kill;return t&&e&&s&&r!==null};pe.prototype.isStopped=function(){return this._states.kill};pe.prototype.isBusy=function(){return this._states.blocking};pe.prototype.currentRun=function(){return this._states.currentRun?this._states.currentRun.getDate():null};pe.prototype.previousRun=function(){return this._states.previousRun?this._states.previousRun.getDate():null};pe.prototype.msToNext=function(r){r=r||new Date;let t=this._next(r);return t?t.getTime()-r.getTime():null};pe.prototype.stop=function(){this._states.kill=!0,this._states.currentTimeout&&clearTimeout(this._states.currentTimeout);let r=fs.indexOf(this);r>=0&&fs.splice(r,1)};pe.prototype.pause=function(){return this._states.paused=!0,!this._states.kill};pe.prototype.resume=function(){return this._states.paused=!1,!this._states.kill};pe.prototype.schedule=function(r){if(r&&this.fn)throw new Error("Cron: It is not allowed to schedule two functions using the same Croner instance.");r&&(this.fn=r);let t=this.msToNext(),e=this.nextRun(this._states.currentRun);return t==null||isNaN(t)||e===null?this:(t>qr&&(t=qr),this._states.currentTimeout=setTimeout(()=>this._checkTrigger(e),t),this._states.currentTimeout&&this.options.unref&&Ol(this._states.currentTimeout),this)};pe.prototype._trigger=async function(r){if(this._states.blocking=!0,this._states.currentRun=new _e(void 0,this.options.timezone||this.options.utcOffset),this.options.catch)try{await this.fn(this,this.options.context)}catch(t){tn(this.options.catch)&&this.options.catch(t,this)}else await this.fn(this,this.options.context);this._states.previousRun=new _e(r,this.options.timezone||this.options.utcOffset),this._states.blocking=!1};pe.prototype.trigger=async function(){await this._trigger()};pe.prototype._checkTrigger=function(r){let t=new Date,e=!this._states.paused&&t.getTime()>=r,s=this._states.blocking&&this.options.protect;e&&!s?(this._states.maxRuns--,this._trigger()):e&&s&&tn(this.options.protect)&&setTimeout(()=>this.options.protect(this),0),this.schedule()};pe.prototype._next=function(r){let t=!!(r||this._states.currentRun),e=!1;!r&&this.options.startAt&&this.options.interval&&([r,t]=this._calculatePreviousRun(r,t),e=!r),r=new _e(r,this.options.timezone||this.options.utcOffset),this.options.startAt&&r&&r.getTime()<this.options.startAt.getTime()&&(r=this.options.startAt);let s=this._states.once||new _e(r,this.options.timezone||this.options.utcOffset);return!e&&s!==this._states.once&&(s=s.increment(this._states.pattern,this.options,t)),this._states.once&&this._states.once.getTime()<=r.getTime()||s===null||this._states.maxRuns<=0||this._states.kill||this.options.stopAt&&s.getTime()>=this.options.stopAt.getTime()?null:s};pe.prototype._calculatePreviousRun=function(r,t){let e=new _e(void 0,this.options.timezone||this.options.utcOffset);if(this.options.startAt.getTime()<=e.getTime()){r=this.options.startAt;let s=r.getTime()+this.options.interval*1e3;for(;s<=e.getTime();)r=new _e(r,this.options.timezone||this.options.utcOffset).increment(this._states.pattern,this.options,!0),s=r.getTime()+this.options.interval*1e3;t=!0}return[r,t]};pe.Cron=pe;pe.scheduledJobs=fs;var wi=require("obsidian");var hi=require("crypto");var Je=require("fs"),da=require("path");function Nl(r){return`mcp__${r.replace(/[\s.]+/g,"_")}`}function sn(r,t,e={}){let s=t.permissionRules.allow.length>0||t.permissionRules.deny.length>0,n=t.permissionMode?.trim(),a=!!n&&n!=="default",i=e.mcpAllowServers??[],o=i.length>0;if(!(s||a||o))return null;let l=(0,da.join)(r,".claude"),h=(0,da.join)(l,"settings.local.json");(0,Je.existsSync)(l)||(0,Je.mkdirSync)(l,{recursive:!0});let d=null;if((0,Je.existsSync)(h))try{d=(0,Je.readFileSync)(h,"utf-8")}catch{d=null}let u={};if(a&&(u.defaultMode=n),s||o){let p=[...t.permissionRules.allow];for(let m of i)p.push(Nl(m));u.permissions={allow:p,deny:t.permissionRules.deny}}return(0,Je.writeFileSync)(h,JSON.stringify(u,null,2)+`
11775
+ `,"utf-8"),{path:h,backupContent:d}}function Dt(r){if(r)try{r.backupContent!==null?(0,Je.writeFileSync)(r.path,r.backupContent,"utf-8"):(0,Je.existsSync)(r.path)&&(0,Je.unlinkSync)(r.path)}catch{}}function Vr(r){if(typeof r=="string")return r;if(Array.isArray(r))return r.map(e=>{if(typeof e=="string")return e;if(e&&typeof e=="object"&&"text"in e){let s=e.text;if(typeof s=="string")return s}return""}).filter(Boolean).join(`
11776
+ `);if(r&&typeof r=="object"){for(let t of["output","result","text","message"])if(t in r)return Vr(r[t])}}function ha(r,t=[]){if(Array.isArray(r)){for(let i of r)ha(i,t);return t}if(!r||typeof r!="object")return t;let e=r,s=typeof e.tool_name=="string"&&e.tool_name||typeof e.tool=="string"&&e.tool||typeof e.name=="string"&&e.name,n=typeof e.command=="string"?e.command:typeof e.input=="string"?e.input:typeof e.cmd=="string"?e.cmd:void 0,a=typeof e.reason=="string"?e.reason:void 0;s&&["tool_use","tool","name","tool_name"].some(i=>i in e)&&t.push({tool:s,command:n,reason:a});for(let i of Object.values(e))ha(i,t);return t}function Yr(r){if(!r||typeof r!="object")return;let t=r,e=t.usage;if(e&&typeof e=="object"){let n=typeof e.input_tokens=="number"?e.input_tokens:0,a=typeof e.output_tokens=="number"?e.output_tokens:0,i=typeof e.cache_creation_input_tokens=="number"?e.cache_creation_input_tokens:0,o=typeof e.cache_read_input_tokens=="number"?e.cache_read_input_tokens:0,c=n+a+i+o;if(c>0)return c}let s=t.modelUsage;if(s&&typeof s=="object"){let n=0;for(let a of Object.values(s)){if(!a||typeof a!="object")continue;let i=a;n+=typeof i.inputTokens=="number"?i.inputTokens:0,n+=typeof i.outputTokens=="number"?i.outputTokens:0,n+=typeof i.cacheReadInputTokens=="number"?i.cacheReadInputTokens:0,n+=typeof i.cacheCreationInputTokens=="number"?i.cacheCreationInputTokens:0}if(n>0)return n}for(let n of["tokens_used","total_tokens","totalTokens"])if(typeof t[n]=="number")return t[n];for(let n of Object.values(t)){let a=Yr(n);if(typeof a=="number")return a}}function Kr(r){if(!r||typeof r!="object")return;let t=r;if(typeof t.total_cost_usd=="number")return t.total_cost_usd;for(let e of Object.values(t)){let s=Kr(e);if(typeof s=="number")return s}}function ua(r){if(!r||typeof r!="object")return;let t=r;if(t.type==="result"&&typeof t.result=="string"&&t.result.trim())return t.result}function nn(r){if(!r||typeof r!="object")return;let t=r;if(t.modelUsage&&typeof t.modelUsage=="object"){let s=Object.keys(t.modelUsage);if(s.length>0&&s[0])return s[0]}let e=t.message;if(e&&typeof e.model=="string"&&e.model)return e.model;if(typeof t.model=="string"&&t.model)return t.model;for(let s of Object.values(t)){let n=nn(s);if(n)return n}}function Bl(r){return/^gpt-|codex/i.test(r.trim())}var Jr={id:"claude-code",label:"Claude Code",cliPath(r){return r.claudeCliPath},buildExec(r){let t=["-p",r.prompt,"--output-format",r.streaming?"stream-json":"json"];r.streaming&&t.push("--verbose");let e=r.modelSource==="settings"&&Bl(r.model);r.model&&!e&&t.push("--model",r.model),r.effort&&t.push("--effort",r.effort);let s=r.agent.permissionMode?.trim();return s&&s!=="default"?t.push("--permission-mode",s):t.push("--permission-mode","bypassPermissions"),Promise.resolve({cliPath:r.settings.claudeCliPath,args:t})},parseExecOutput(r,t,e){let s=r.trim(),n;if(e){let c=ye(s);for(let l=c.length-1;l>=0;l--){let h=c[l]?.trim();if(h)try{let d=JSON.parse(h);if(d&&typeof d=="object"){n=d;break}}catch{}}}else if(s.startsWith("{")||s.startsWith("["))try{n=JSON.parse(s)}catch{n=void 0}let a=Vr(n)??"";if(!a&&e){let c=[];for(let l of ye(s)){let h=l.trim();if(h)try{let d=JSON.parse(h);if(d.type==="assistant"&&d.message?.content)for(let u of d.message.content)u.type==="text"&&u.text&&c.push(u.text);else d.type==="result"&&typeof d.result=="string"&&c.push(d.result)}catch{}}a=c.join(`
11777
+ `).trim()}a||(a=t.trim()||"(no output)");let i=nn(n),o=ua(n);if((!i||!o)&&e)for(let c of ye(s)){let l=c.trim();if(l)try{let h=JSON.parse(l);if(!i){let d=nn(h);d&&(i=d)}if(!o){let d=ua(h);d&&(o=d)}if(i&&o)break}catch{}}return{outputText:a,finalResult:o,tokensUsed:Yr(n),costUsd:Kr(n),toolsUsed:ha(n),concreteModel:i,rawJson:n}},extractStreamChunk(r){let t=r.trim();if(!t)return null;let e;try{e=JSON.parse(t)}catch{return null}let s=e.type;if(s==="assistant"){let n=e.message;if(n?.content&&Array.isArray(n.content)){let a=[];for(let i of n.content)if(i.type==="text"&&typeof i.text=="string")a.push(i.text);else if(i.type==="tool_use"){let o=String(i.name??"tool"),c=i.input,l=c?.command??c?.content??"";a.push(`
11867
11778
  \u25B8 ${o}${l?`: ${String(l).slice(0,200)}`:""}
11868
11779
  `)}if(a.length>0)return a.join("")}}if(s==="result"){let n=typeof e.result=="string"?e.result:null;if(n)return`
11869
- ${n}`}return null},setupPermissions(i,t,e,s){let n=sn(i,t,{mcpAllowServers:s?.mcpAllowServers});return n?Promise.resolve({restore:()=>Rt(n)}):Promise.resolve(null)}};var le=require("fs"),fa=require("crypto"),ys=require("os"),Xe=require("path");var Bl=/^[A-Z][A-Za-z]*$/;function Ul(i){let t=i.trim();if(!t)return{error:"empty Bash() matches every command \u2014 use the read-only sandbox instead"};let e=t.split(/\s+/).filter(Boolean),s=[];for(let n=0;n<e.length;n++){let a=e[n],r=n===e.length-1;if(a==="*"){if(r)continue;return{error:"a wildcard in the middle of a command can't be expressed as a prefix rule"}}if(a.includes("*"))return{error:"embedded wildcards aren't supported (execpolicy matches argv prefixes only)"};s.push(a)}return s.length===0?{error:"a bare wildcard matches every command \u2014 use the read-only sandbox instead"}:{tokens:s}}function $l(i,t){return`prefix_rule(pattern=[${i.map(s=>JSON.stringify(s)).join(", ")}], decision="${t}", justification="agent-fleet")`}function jl(i){let t=[],e=[],s=(a,r)=>{let o=a.trim();if(!o||o.startsWith("mcp__"))return;let c=o.match(/^Bash\((.*)\)$/s);if(!c){e.push({rule:o,reason:Bl.test(o)?"tool-name rules are governed by the sandbox, not execpolicy":"not a Bash(...) command pattern"});return}let l=Ul(c[1]);if("error"in l){e.push({rule:o,reason:l.error});return}t.push($l(l.tokens,r))};for(let a of i.permissionRules.deny)s(a,"forbidden");for(let a of i.permissionRules.allow)s(a,"allow");return t.length===0?{rulesText:null,dropped:e,emitted:0}:{rulesText:`${`# Generated by Agent Fleet for agent "${i.name}". Do not edit \u2014 regenerated each run.
11780
+ ${n}`}return null},setupPermissions(r,t,e,s){let n=sn(r,t,{mcpAllowServers:s?.mcpAllowServers});return n?Promise.resolve({restore:()=>Dt(n)}):Promise.resolve(null)}};var le=require("fs"),fa=require("crypto"),ys=require("os"),Xe=require("path");var Ul=/^[A-Z][A-Za-z]*$/;function $l(r){let t=r.trim();if(!t)return{error:"empty Bash() matches every command \u2014 use the read-only sandbox instead"};let e=t.split(/\s+/).filter(Boolean),s=[];for(let n=0;n<e.length;n++){let a=e[n],i=n===e.length-1;if(a==="*"){if(i)continue;return{error:"a wildcard in the middle of a command can't be expressed as a prefix rule"}}if(a.includes("*"))return{error:"embedded wildcards aren't supported (execpolicy matches argv prefixes only)"};s.push(a)}return s.length===0?{error:"a bare wildcard matches every command \u2014 use the read-only sandbox instead"}:{tokens:s}}function jl(r,t){return`prefix_rule(pattern=[${r.map(s=>JSON.stringify(s)).join(", ")}], decision="${t}", justification="agent-fleet")`}function Wl(r){let t=[],e=[],s=(a,i)=>{let o=a.trim();if(!o||o.startsWith("mcp__"))return;let c=o.match(/^Bash\((.*)\)$/s);if(!c){e.push({rule:o,reason:Ul.test(o)?"tool-name rules are governed by the sandbox, not execpolicy":"not a Bash(...) command pattern"});return}let l=$l(c[1]);if("error"in l){e.push({rule:o,reason:l.error});return}t.push(jl(l.tokens,i))};for(let a of r.permissionRules.deny)s(a,"forbidden");for(let a of r.permissionRules.allow)s(a,"allow");return t.length===0?{rulesText:null,dropped:e,emitted:0}:{rulesText:`${`# Generated by Agent Fleet for agent "${r.name}". Do not edit \u2014 regenerated each run.
11870
11781
  # Source: this agent's permissionRules (allow/deny), translated to Codex execpolicy.
11871
11782
  `}${t.join(`
11872
11783
  `)}
11873
- `,dropped:e,emitted:t.length}}function Wl(){let i=process.env.CODEX_HOME?.trim();return i||(0,Xe.join)((0,ys.homedir)(),".codex")}var Xi=(0,Xe.join)((0,ys.tmpdir)(),"agent-fleet-codex");function Hl(i){let t=i.name.replace(/[^A-Za-z0-9_-]+/g,"-").replace(/^-+|-+$/g,"")||"agent",e=(0,fa.createHash)("sha1").update(i.filePath).digest("hex").slice(0,8);return(0,Xe.join)(Xi,`${t}-${e}`)}function ql(i,t){let e=`${i}.${process.pid}.${Date.now()}.tmp`;(0,le.writeFileSync)(e,t,"utf-8"),(0,le.renameSync)(e,i)}function zl(i,t){try{let e=(0,le.lstatSync)(t);if(e.isSymbolicLink()){try{if((0,le.readlinkSync)(t)===i)return}catch{}(0,le.unlinkSync)(t)}else e.isDirectory()?(0,le.rmSync)(t,{recursive:!0,force:!0}):(0,le.unlinkSync)(t)}catch{}(0,le.symlinkSync)(i,t)}function Gl(i,t,e=Wl()){if(!(0,le.existsSync)(e))return null;let s=Hl(i);(0,le.mkdirSync)(s,{recursive:!0});for(let o of(0,le.readdirSync)(e))o!=="rules"&&zl((0,Xe.join)(e,o),(0,Xe.join)(s,o));let n=(0,Xe.join)(s,"rules");(0,le.mkdirSync)(n,{recursive:!0});let a=(0,Xe.join)(e,"rules");if((0,le.existsSync)(a)){for(let o of(0,le.readdirSync)(a))if(o.endsWith(".rules"))try{(0,le.copyFileSync)((0,Xe.join)(a,o),(0,Xe.join)(n,`user-${o}`))}catch{}}let r=(0,Xe.join)(n,"agent-fleet.rules");return ql(r,t),{codexHome:s,rulesPath:r}}function Qi(){try{(0,le.rmSync)(Xi,{recursive:!0,force:!0})}catch{}}var pa=new Map,ma=new Map;function Zi(){pa.clear(),ma.clear()}function er(i,t,e){return new Promise(s=>{let n=!1,a=r=>{n||(n=!0,s(r))};try{let r=dt(i,["execpolicy","check","--rules",t,...e]),o=window.setTimeout(()=>{try{r.kill()}catch{}a(null)},8e3);r.on("error",()=>{window.clearTimeout(o),a(null)}),r.on("close",c=>{window.clearTimeout(o),a(c)})}catch{a(null)}})}function Vl(i){let t=pa.get(i);if(t)return t;let e=(async()=>{let s=null;try{s=(0,le.mkdtempSync)((0,Xe.join)((0,ys.tmpdir)(),"af-codex-probe-"));let n=(0,Xe.join)(s,"probe.rules");return(0,le.writeFileSync)(n,`prefix_rule(pattern=["ls"], decision="allow")
11874
- `,"utf-8"),await er(i,n,["ls"])===0}catch{return!1}finally{if(s)try{(0,le.rmSync)(s,{recursive:!0,force:!0})}catch{}}})();return pa.set(i,e),e}async function Yl(i,t,e){let s=`${i}:${(0,fa.createHash)("sha1").update(e).digest("hex")}`,n=ma.get(s);if(n!==void 0)return n;let r=await er(i,t,["true"])===0;return ma.set(s,r),r}var Ji=new Set;function gs(i,t){Ji.has(i)||(Ji.add(i),console.warn(`Agent Fleet: ${t}`))}async function tr(i,t){if(!(i.permissionRules.allow.length>0||i.permissionRules.deny.length>0))return null;let{rulesText:s,dropped:n}=jl(i);if(n.length>0&&gs(`dropped:${i.name}:${n.map(l=>l.rule).join("|")}`,`agent "${i.name}": ${n.length} permission rule(s) can't be enforced by Codex and were ignored \u2014 ${n.map(l=>`"${l.rule}" (${l.reason})`).join("; ")}. File/network access is governed by the sandbox (Permission Mode).`),!s)return null;let a=t.codexCliPath;if(!await Vl(a))return gs(`unsupported:${a}`,`this Codex build doesn't support execpolicy rules; agent "${i.name}" runs with sandbox-only enforcement. Update Codex to enforce command allow/deny rules.`),null;let o;try{o=Gl(i,s)}catch(l){return gs(`overlay-fail:${i.name}`,`couldn't build the Codex permission overlay for agent "${i.name}" (${l instanceof Error?l.message:String(l)}); falling back to sandbox-only (symlinks may be unavailable on this platform).`),null}return o?await Yl(a,o.rulesPath,s)?{restore:()=>{},env:{CODEX_HOME:o.codexHome}}:(gs(`invalid-rules:${i.name}`,`generated Codex rules for agent "${i.name}" failed validation; falling back to sandbox-only.`),null):(gs("no-codex-home",`Codex home (~/.codex) not found; agent "${i.name}" runs with sandbox-only enforcement. Run \`codex login\` first.`),null)}function Kl(i){let t=i.trim();return/^(opus|sonnet|haiku|opusplan)$/i.test(t)||/claude|anthropic/i.test(t)}function Jl(i){let t=i.trim().toLowerCase();return t?t==="max"?"xhigh":["low","medium","high","xhigh"].includes(t)?t:"":""}function Xl(i){switch((i??"").trim()){case"plan":case"read-only":return["--sandbox","read-only"];case"acceptEdits":case"default":case"workspace-write":return["--sandbox","workspace-write"];case"danger-full-access":return["--sandbox","danger-full-access"];default:return["--dangerously-bypass-approvals-and-sandbox"]}}function Ql(i){let t=["exec","--json","--skip-git-repo-check"],e=i.modelSource==="settings"&&Kl(i.model);i.model&&!e&&t.push("-m",i.model);let s=Jl(i.effort);return s&&t.push("-c",`model_reasoning_effort="${s}"`),t.push(...Xl(i.agent.permissionMode)),i.resumeSessionId&&t.push("resume",i.resumeSessionId),t.push("-"),{args:t,stdinPayload:i.prompt}}function sr(i){let t=i.trim();if(!t)return null;try{let e=JSON.parse(t);return e&&typeof e=="object"&&!Array.isArray(e)?e:null}catch{return null}}function an(i){let t=i.item;return t&&typeof t=="object"?t:null}function ga(i){let t=typeof i.type=="string"?i.type:"";if(t==="command_execution")return{tool:"shell",command:typeof i.command=="string"?i.command:void 0};if(t==="mcp_tool_call"){let e=typeof i.server=="string"?i.server:"mcp",s=typeof i.tool=="string"?i.tool:"tool";return{tool:`mcp__${e}__${s}`}}return t==="web_search"?{tool:"web_search",command:typeof i.query=="string"?i.query:void 0}:t==="file_change"?{tool:"file_change",command:(Array.isArray(i.changes)?i.changes:[]).map(n=>`${typeof n.kind=="string"?n.kind:"update"} ${typeof n.path=="string"?n.path:""}`.trim()).filter(Boolean).join(", ")||void 0}:null}function nr(i){let t=i.usage;if(!t||typeof t!="object")return 0;let e=typeof t.input_tokens=="number"?t.input_tokens:0,s=typeof t.output_tokens=="number"?t.output_tokens:0;return e+s}function Zl(i){let t=i.usage;return!t||typeof t!="object"?0:typeof t.input_tokens=="number"?t.input_tokens:0}function ya(){return{emittedTextLengths:new Map}}function ar(i,t){let e=typeof i.type=="string"?i.type:"",s=[];if(e==="thread.started"&&typeof i.thread_id=="string"&&i.thread_id)return s.push({kind:"session",sessionId:i.thread_id}),s;if(e==="turn.completed")return s.push({kind:"usage",contextTokens:Zl(i),totalTokens:nr(i)}),s;if(e==="turn.failed"){let n=i.error,a=n&&typeof n.message=="string"?n.message:"turn failed";return s.push({kind:"turn-failed",message:a}),s}if(e==="error"){let n=typeof i.message=="string"?i.message:"stream error";return s.push({kind:"error",message:n}),s}if(e==="item.started"||e==="item.updated"||e==="item.completed"){let n=an(i);if(!n)return s;let a=typeof n.type=="string"?n.type:"";if(a==="agent_message"){let r=typeof n.text=="string"?n.text:"",o=typeof n.id=="string"?n.id:"__single__",c=t.emittedTextLengths.get(o)??0;return r.length>c&&(s.push({kind:"text",text:r.slice(c)}),t.emittedTextLengths.set(o,r.length)),s}if(a==="error"){let r=typeof n.message=="string"?n.message:"item error";return s.push({kind:"error",message:r}),s}if(e==="item.started"){let r=ga(n);r&&s.push({kind:"tool",toolName:r.tool,command:r.command})}return s}return s}var vs={id:"codex",label:"Codex",cliPath(i){return i.codexCliPath},buildExec(i){let{args:t,stdinPayload:e}=Ql(i);return Promise.resolve({cliPath:i.settings.codexCliPath,args:t,stdinPayload:e})},parseExecOutput(i,t,e){let s=[],n=[],a=[],r=0,o,c;for(let d of ye(i)){let u=sr(d);if(!u)continue;let p=typeof u.type=="string"?u.type:"";if(p==="thread.started"&&typeof u.thread_id=="string")o=u.thread_id;else if(p==="turn.completed")r+=nr(u),c=u;else if(p==="turn.failed"){let m=u.error;a.push(m&&typeof m.message=="string"?m.message:"turn failed")}else if(p==="error")a.push(typeof u.message=="string"?u.message:"stream error");else if(p==="item.completed"){let m=an(u);if(!m)continue;let f=typeof m.type=="string"?m.type:"";if(f==="agent_message"&&typeof m.text=="string"&&m.text.trim())s.push(m.text);else if(f==="error"&&typeof m.message=="string")a.push(m.message);else{let g=ga(m);g&&n.push(g)}}}let l=s.join(`
11784
+ `,dropped:e,emitted:t.length}}function Hl(){let r=process.env.CODEX_HOME?.trim();return r||(0,Xe.join)((0,ys.homedir)(),".codex")}var Qr=(0,Xe.join)((0,ys.tmpdir)(),"agent-fleet-codex");function ql(r){let t=r.name.replace(/[^A-Za-z0-9_-]+/g,"-").replace(/^-+|-+$/g,"")||"agent",e=(0,fa.createHash)("sha1").update(r.filePath).digest("hex").slice(0,8);return(0,Xe.join)(Qr,`${t}-${e}`)}function zl(r,t){let e=`${r}.${process.pid}.${Date.now()}.tmp`;(0,le.writeFileSync)(e,t,"utf-8"),(0,le.renameSync)(e,r)}function Gl(r,t){try{let e=(0,le.lstatSync)(t);if(e.isSymbolicLink()){try{if((0,le.readlinkSync)(t)===r)return}catch{}(0,le.unlinkSync)(t)}else e.isDirectory()?(0,le.rmSync)(t,{recursive:!0,force:!0}):(0,le.unlinkSync)(t)}catch{}(0,le.symlinkSync)(r,t)}function Vl(r,t,e=Hl()){if(!(0,le.existsSync)(e))return null;let s=ql(r);(0,le.mkdirSync)(s,{recursive:!0});for(let o of(0,le.readdirSync)(e))o!=="rules"&&Gl((0,Xe.join)(e,o),(0,Xe.join)(s,o));let n=(0,Xe.join)(s,"rules");(0,le.mkdirSync)(n,{recursive:!0});let a=(0,Xe.join)(e,"rules");if((0,le.existsSync)(a)){for(let o of(0,le.readdirSync)(a))if(o.endsWith(".rules"))try{(0,le.copyFileSync)((0,Xe.join)(a,o),(0,Xe.join)(n,`user-${o}`))}catch{}}let i=(0,Xe.join)(n,"agent-fleet.rules");return zl(i,t),{codexHome:s,rulesPath:i}}function Zr(){try{(0,le.rmSync)(Qr,{recursive:!0,force:!0})}catch{}}var pa=new Map,ma=new Map;function ei(){pa.clear(),ma.clear()}function ti(r,t,e){return new Promise(s=>{let n=!1,a=i=>{n||(n=!0,s(i))};try{let i=dt(r,["execpolicy","check","--rules",t,...e]),o=window.setTimeout(()=>{try{i.kill()}catch{}a(null)},8e3);i.on("error",()=>{window.clearTimeout(o),a(null)}),i.on("close",c=>{window.clearTimeout(o),a(c)})}catch{a(null)}})}function Yl(r){let t=pa.get(r);if(t)return t;let e=(async()=>{let s=null;try{s=(0,le.mkdtempSync)((0,Xe.join)((0,ys.tmpdir)(),"af-codex-probe-"));let n=(0,Xe.join)(s,"probe.rules");return(0,le.writeFileSync)(n,`prefix_rule(pattern=["ls"], decision="allow")
11785
+ `,"utf-8"),await ti(r,n,["ls"])===0}catch{return!1}finally{if(s)try{(0,le.rmSync)(s,{recursive:!0,force:!0})}catch{}}})();return pa.set(r,e),e}async function Kl(r,t,e){let s=`${r}:${(0,fa.createHash)("sha1").update(e).digest("hex")}`,n=ma.get(s);if(n!==void 0)return n;let i=await ti(r,t,["true"])===0;return ma.set(s,i),i}var Xr=new Set;function gs(r,t){Xr.has(r)||(Xr.add(r),console.warn(`Agent Fleet: ${t}`))}async function si(r,t){if(!(r.permissionRules.allow.length>0||r.permissionRules.deny.length>0))return null;let{rulesText:s,dropped:n}=Wl(r);if(n.length>0&&gs(`dropped:${r.name}:${n.map(l=>l.rule).join("|")}`,`agent "${r.name}": ${n.length} permission rule(s) can't be enforced by Codex and were ignored \u2014 ${n.map(l=>`"${l.rule}" (${l.reason})`).join("; ")}. File/network access is governed by the sandbox (Permission Mode).`),!s)return null;let a=t.codexCliPath;if(!await Yl(a))return gs(`unsupported:${a}`,`this Codex build doesn't support execpolicy rules; agent "${r.name}" runs with sandbox-only enforcement. Update Codex to enforce command allow/deny rules.`),null;let o;try{o=Vl(r,s)}catch(l){return gs(`overlay-fail:${r.name}`,`couldn't build the Codex permission overlay for agent "${r.name}" (${l instanceof Error?l.message:String(l)}); falling back to sandbox-only (symlinks may be unavailable on this platform).`),null}return o?await Kl(a,o.rulesPath,s)?{restore:()=>{},env:{CODEX_HOME:o.codexHome}}:(gs(`invalid-rules:${r.name}`,`generated Codex rules for agent "${r.name}" failed validation; falling back to sandbox-only.`),null):(gs("no-codex-home",`Codex home (~/.codex) not found; agent "${r.name}" runs with sandbox-only enforcement. Run \`codex login\` first.`),null)}function Jl(r){let t=r.trim();return/^(opus|sonnet|haiku|opusplan)$/i.test(t)||/claude|anthropic/i.test(t)}function Xl(r){let t=r.trim().toLowerCase();return t?t==="max"?"xhigh":["low","medium","high","xhigh"].includes(t)?t:"":""}function Ql(r){switch((r??"").trim()){case"plan":case"read-only":return["--sandbox","read-only"];case"acceptEdits":case"default":case"workspace-write":return["--sandbox","workspace-write"];case"danger-full-access":return["--sandbox","danger-full-access"];default:return["--dangerously-bypass-approvals-and-sandbox"]}}function Zl(r){let t=["exec","--json","--skip-git-repo-check"],e=r.modelSource==="settings"&&Jl(r.model);r.model&&!e&&t.push("-m",r.model);let s=Xl(r.effort);return s&&t.push("-c",`model_reasoning_effort="${s}"`),t.push(...Ql(r.agent.permissionMode)),r.resumeSessionId&&t.push("resume",r.resumeSessionId),t.push("-"),{args:t,stdinPayload:r.prompt}}function ni(r){let t=r.trim();if(!t)return null;try{let e=JSON.parse(t);return e&&typeof e=="object"&&!Array.isArray(e)?e:null}catch{return null}}function an(r){let t=r.item;return t&&typeof t=="object"?t:null}function ga(r){let t=typeof r.type=="string"?r.type:"";if(t==="command_execution")return{tool:"shell",command:typeof r.command=="string"?r.command:void 0};if(t==="mcp_tool_call"){let e=typeof r.server=="string"?r.server:"mcp",s=typeof r.tool=="string"?r.tool:"tool";return{tool:`mcp__${e}__${s}`}}return t==="web_search"?{tool:"web_search",command:typeof r.query=="string"?r.query:void 0}:t==="file_change"?{tool:"file_change",command:(Array.isArray(r.changes)?r.changes:[]).map(n=>`${typeof n.kind=="string"?n.kind:"update"} ${typeof n.path=="string"?n.path:""}`.trim()).filter(Boolean).join(", ")||void 0}:null}function ai(r){let t=r.usage;if(!t||typeof t!="object")return 0;let e=typeof t.input_tokens=="number"?t.input_tokens:0,s=typeof t.output_tokens=="number"?t.output_tokens:0;return e+s}function ec(r){let t=r.usage;return!t||typeof t!="object"?0:typeof t.input_tokens=="number"?t.input_tokens:0}function ya(){return{emittedTextLengths:new Map}}function ri(r,t){let e=typeof r.type=="string"?r.type:"",s=[];if(e==="thread.started"&&typeof r.thread_id=="string"&&r.thread_id)return s.push({kind:"session",sessionId:r.thread_id}),s;if(e==="turn.completed")return s.push({kind:"usage",contextTokens:ec(r),totalTokens:ai(r)}),s;if(e==="turn.failed"){let n=r.error,a=n&&typeof n.message=="string"?n.message:"turn failed";return s.push({kind:"turn-failed",message:a}),s}if(e==="error"){let n=typeof r.message=="string"?r.message:"stream error";return s.push({kind:"error",message:n}),s}if(e==="item.started"||e==="item.updated"||e==="item.completed"){let n=an(r);if(!n)return s;let a=typeof n.type=="string"?n.type:"";if(a==="agent_message"){let i=typeof n.text=="string"?n.text:"",o=typeof n.id=="string"?n.id:"__single__",c=t.emittedTextLengths.get(o)??0;return i.length>c&&(s.push({kind:"text",text:i.slice(c)}),t.emittedTextLengths.set(o,i.length)),s}if(a==="error"){let i=typeof n.message=="string"?n.message:"item error";return s.push({kind:"error",message:i}),s}if(e==="item.started"){let i=ga(n);i&&s.push({kind:"tool",toolName:i.tool,command:i.command})}return s}return s}var vs={id:"codex",label:"Codex",cliPath(r){return r.codexCliPath},buildExec(r){let{args:t,stdinPayload:e}=Zl(r);return Promise.resolve({cliPath:r.settings.codexCliPath,args:t,stdinPayload:e})},parseExecOutput(r,t,e){let s=[],n=[],a=[],i=0,o,c;for(let d of ye(r)){let u=ni(d);if(!u)continue;let p=typeof u.type=="string"?u.type:"";if(p==="thread.started"&&typeof u.thread_id=="string")o=u.thread_id;else if(p==="turn.completed")i+=ai(u),c=u;else if(p==="turn.failed"){let m=u.error;a.push(m&&typeof m.message=="string"?m.message:"turn failed")}else if(p==="error")a.push(typeof u.message=="string"?u.message:"stream error");else if(p==="item.completed"){let m=an(u);if(!m)continue;let f=typeof m.type=="string"?m.type:"";if(f==="agent_message"&&typeof m.text=="string"&&m.text.trim())s.push(m.text);else if(f==="error"&&typeof m.message=="string")a.push(m.message);else{let g=ga(m);g&&n.push(g)}}}let l=s.join(`
11875
11786
 
11876
11787
  `).trim();l||(l=a.join(`
11877
- `).trim()),l||(l=t.trim()||"(no output)");let h=s[s.length-1];return{outputText:l,finalResult:h?.trim()?h:void 0,tokensUsed:r>0?r:void 0,costUsd:void 0,toolsUsed:n,concreteModel:void 0,rawJson:c,sessionId:o}},extractStreamChunk(i){let t=sr(i);if(!t)return null;let e=typeof t.type=="string"?t.type:"";if(e==="item.completed"){let s=an(t);return s&&s.type==="agent_message"&&typeof s.text=="string"&&s.text.trim()?s.text:null}if(e==="item.started"){let s=an(t);if(!s)return null;let n=ga(s);if(n){let a=n.command?`: ${n.command.slice(0,200)}`:"";return`
11788
+ `).trim()),l||(l=t.trim()||"(no output)");let h=s[s.length-1];return{outputText:l,finalResult:h?.trim()?h:void 0,tokensUsed:i>0?i:void 0,costUsd:void 0,toolsUsed:n,concreteModel:void 0,rawJson:c,sessionId:o}},extractStreamChunk(r){let t=ni(r);if(!t)return null;let e=typeof t.type=="string"?t.type:"";if(e==="item.completed"){let s=an(t);return s&&s.type==="agent_message"&&typeof s.text=="string"&&s.text.trim()?s.text:null}if(e==="item.started"){let s=an(t);if(!s)return null;let n=ga(s);if(n){let a=n.command?`: ${n.command.slice(0,200)}`:"";return`
11878
11789
  \u25B8 ${n.tool}${a}
11879
11790
  `}return null}if(e==="turn.failed"){let s=t.error;return`
11880
11791
  \u2716 ${s&&typeof s.message=="string"?s.message:"turn failed"}
11881
- `}return null},setupPermissions(i,t,e,s){return tr(t,e)}};function kt(i){let t=(i??"").trim().toLowerCase();return t==="codex"||t==="openai-codex"?"codex":"claude-code"}function ir(i){return kt(i)==="codex"?vs:Ki}var rr=new Set(["","default","subscription"]);function Dt(i,t,e){let s=va(i?.model);if(s)return{value:wa(s),source:"task"};let n=va(t.model);if(n)return{value:wa(n),source:"agent"};let a=va(e.defaultModel);return a?{value:wa(a),source:"settings"}:{value:"",source:"cli-default"}}function ws(i){return!rr.has(i.trim())}function va(i){if(!i)return"";let t=i.trim();return t||""}function wa(i){return rr.has(i)?"":i}function rn(i,t){let e=i.wikiReferences;if(!e||e.length===0)return null;let s=[];for(let a of e){let r=t.getAgentByName(a.agent);if(!r||!r.wikiKeeper)continue;let o=r.wikiKeeper,c=o.scopeRoot?`${o.scopeRoot.replace(/\/+$/,"")}/`:"";s.push({agentName:r.name,scopeRoot:o.scopeRoot||"(whole vault)",topicsPath:`${c}${o.topicsRoot}`,inboxPath:`${c}${o.inboxPath}`,indexPath:`${c}${o.indexPath}`})}if(s.length===0)return null;let n=["## Wiki Access"];n.push("You have read access to the following wikis maintained by other agents. Use the `wiki-query` skill in consumer mode when the user asks something a wiki may already cover."),n.push("");for(let a of s)n.push(`### Wiki: \`${a.agentName}\``),n.push(`- scope root: \`${a.scopeRoot}\``),n.push(`- topics: \`${a.topicsPath}/\``),n.push(`- index: \`${a.indexPath}\``),n.push(`- inbox: \`${a.inboxPath}/\``),n.push("");return n.push("### Rules"),n.push("- **Cite every factual claim** from a wiki using `[[<topics-path>/<page>]]`. If a claim has no page, say the wiki doesn't cover it \u2014 do not fabricate."),n.push("- **When the user shares something durable** that isn't yet in a wiki (a decision, a new entity mention, a competitor change, a meeting outcome), write a short markdown file to the relevant wiki's inbox at `<inbox>/YYYY-MM-DD-<slug>.md` with a one-line note + the source. The wiki keeper files it canonically on its next ingest."),n.push("- **Do NOT write to `<topics-path>/` directly.** The wiki keeper is the canonical curator of topic pages. Use the inbox as your deposit point."),n.push("- **When the question spans multiple wikis**, be explicit about which scope each cited page belongs to."),n.join(`
11882
- `)}var ht=require("fs"),ln=require("path");var on=require("fs"),ec=require("path");var ba="remember",fu=`mcp__${ba}`,or=String.raw`
11792
+ `}return null},setupPermissions(r,t,e,s){return si(t,e)}};function kt(r){let t=(r??"").trim().toLowerCase();return t==="codex"||t==="openai-codex"?"codex":"claude-code"}function ii(r){return kt(r)==="codex"?vs:Jr}var oi=new Set(["","default","subscription"]);function It(r,t,e){let s=va(r?.model);if(s)return{value:wa(s),source:"task"};let n=va(t.model);if(n)return{value:wa(n),source:"agent"};let a=va(e.defaultModel);return a?{value:wa(a),source:"settings"}:{value:"",source:"cli-default"}}function ws(r){return!oi.has(r.trim())}function va(r){if(!r)return"";let t=r.trim();return t||""}function wa(r){return oi.has(r)?"":r}function rn(r,t){let e=r.wikiReferences;if(!e||e.length===0)return null;let s=[];for(let a of e){let i=t.getAgentByName(a.agent);if(!i||!i.wikiKeeper)continue;let o=i.wikiKeeper,c=o.scopeRoot?`${o.scopeRoot.replace(/\/+$/,"")}/`:"";s.push({agentName:i.name,scopeRoot:o.scopeRoot||"(whole vault)",topicsPath:`${c}${o.topicsRoot}`,inboxPath:`${c}${o.inboxPath}`,indexPath:`${c}${o.indexPath}`})}if(s.length===0)return null;let n=["## Wiki Access"];n.push("You have read access to the following wikis maintained by other agents. Use the `wiki-query` skill in consumer mode when the user asks something a wiki may already cover."),n.push("");for(let a of s)n.push(`### Wiki: \`${a.agentName}\``),n.push(`- scope root: \`${a.scopeRoot}\``),n.push(`- topics: \`${a.topicsPath}/\``),n.push(`- index: \`${a.indexPath}\``),n.push(`- inbox: \`${a.inboxPath}/\``),n.push("");return n.push("### Rules"),n.push("- **Cite every factual claim** from a wiki using `[[<topics-path>/<page>]]`. If a claim has no page, say the wiki doesn't cover it \u2014 do not fabricate."),n.push("- **When the user shares something durable** that isn't yet in a wiki (a decision, a new entity mention, a competitor change, a meeting outcome), write a short markdown file to the relevant wiki's inbox at `<inbox>/YYYY-MM-DD-<slug>.md` with a one-line note + the source. The wiki keeper files it canonically on its next ingest."),n.push("- **Do NOT write to `<topics-path>/` directly.** The wiki keeper is the canonical curator of topic pages. Use the inbox as your deposit point."),n.push("- **When the question spans multiple wikis**, be explicit about which scope each cited page belongs to."),n.join(`
11793
+ `)}var ht=require("fs"),ln=require("path");var on=require("fs"),tc=require("path");var ba="remember",vu=`mcp__${ba}`,li=String.raw`
11883
11794
  const fs = require("fs");
11884
11795
  const path = require("path");
11885
11796
  const PENDING_DIR = process.env.AF_PENDING_DIR;
@@ -11971,7 +11882,7 @@ function handle(req) {
11971
11882
  send({ jsonrpc: "2.0", id, error: { code: -32601, message: "method not found: " + method } });
11972
11883
  }
11973
11884
  }
11974
- `;function lr(i){let t=[];for(let e of i){let s=e.trim();if(s)try{let n=JSON.parse(s),a=typeof n.text=="string"?n.text.trim():"";if(!a)continue;let r=n.section==="Preferences"||n.section==="Procedures"||n.section==="Observations"?n.section:void 0;t.push({text:a,pinned:n.pinned===!0,section:r,source:typeof n.source=="string"?n.source:"mcp"})}catch{}}return t}var tc=0;function sc(i,t){return{def:{name:ba,type:"stdio",enabled:!0,command:"node",env:{AF_PENDING_DIR:i,AF_SOURCE:t},status:"connected",scope:"user",tools:[],toolDetails:[]},inlineScript:or}}function cn(i){let t=i.agentGrants.map(n=>n.trim().toLowerCase()).filter(Boolean),e=t.length>0?new Set(t):null,s=[];for(let n of i.registry){if(!n.enabled||e&&!e.has(n.name.trim().toLowerCase())||n.type==="unknown")continue;let a={};if(n.type==="stdio"){if(n.envSecretKeys&&n.envSecretKeys.length>0){let r={};for(let o of n.envSecretKeys){let c=process.env[o];c&&(r[o]=c)}Object.keys(r).length>0&&(a.env=r)}}else if(n.auth!=="none"){let r=i.getBearerToken(n.name);r&&(a.bearerToken=r)}s.push({def:n,secrets:a})}return i.remember&&s.push(sc(i.remember.pendingDir,i.remember.source)),s}function nc(i){return`AF_MCP_${i.toUpperCase().replace(/[^A-Z0-9]+/g,"_").replace(/^_+|_+$/g,"")||"SERVER"}_TOKEN`}function cr(i){return/^[A-Za-z0-9_-]+$/.test(i)?i:`"${i.replace(/"/g,'\\"')}"`}function ac(i,t){let{def:e,secrets:s}=i;if(e.type==="unknown")throw new Error(`server ${e.name} has unknown transport`);let n=e.type;if(n==="stdio"){let a=e.command??"node",r=t?[t]:e.args??[],o={...e.env??{},...s?.env??{}};return{name:e.name,type:n,command:a,args:r,env:o}}if(!e.url)throw new Error(`server ${e.name} (${n}) has no url`);return{name:e.name,type:n,url:e.url,headers:e.headers,bearerToken:s?.bearerToken,oauthResource:e.oauth?.resource,oauthClientId:e.oauth?.clientId}}function ic(i){if(i.type==="stdio"){let s={command:i.command,args:i.args??[]};return i.env&&Object.keys(i.env).length>0&&(s.env=i.env),s}let t={...i.headers??{}};i.bearerToken&&(t.Authorization=`Bearer ${i.bearerToken}`);let e={type:i.type,url:i.url};return Object.keys(t).length>0&&(e.headers=t),e}function rc(i){let t=cr(i.name),e=[],s={};if(i.type==="stdio"){e.push("-c",`mcp_servers.${t}.command=${JSON.stringify(i.command??"node")}`),i.args&&i.args.length>0&&e.push("-c",`mcp_servers.${t}.args=${JSON.stringify(i.args)}`);for(let[n,a]of Object.entries(i.env??{}))e.push("-c",`mcp_servers.${t}.env.${cr(n)}=${JSON.stringify(a)}`)}else{if(e.push("-c",`mcp_servers.${t}.url=${JSON.stringify(i.url)}`),i.bearerToken){let n=nc(i.name);s[n]=i.bearerToken,e.push("-c",`mcp_servers.${t}.bearer_token_env_var=${JSON.stringify(n)}`)}i.oauthResource&&e.push("-c",`mcp_servers.${t}.oauth_resource=${JSON.stringify(i.oauthResource)}`),i.oauthClientId&&e.push("-c",`mcp_servers.${t}.oauth.client_id=${JSON.stringify(i.oauthClientId)}`)}return e.push("-c",`mcp_servers.${t}.enabled=true`),{args:e,env:s}}function dn(i,t,e){if(e.length===0)return null;let s=kt(t)==="codex",n=[],a=(0,ln.join)(i,".claude");try{(0,ht.existsSync)(a)||(0,ht.mkdirSync)(a,{recursive:!0});let r=`${process.pid}-${Date.now()}-${tc++}`,o=[],c=0;for(let d of e){try{let u=null;d.inlineScript&&(u=(0,ln.join)(a,`af-mcp-${oc(d.def.name)}.${r}-${c}.cjs`),(0,ht.writeFileSync)(u,d.inlineScript,"utf-8"),n.push(u)),o.push(ac(d,u))}catch(u){console.warn(`Agent Fleet: skipping MCP server "${d.def.name}" in projection:`,u)}c++}if(o.length===0)return ka(n),null;if(s){let d=[],u={};for(let p of o){let m=rc(p);d.push(...m.args),Object.assign(u,m.env)}return{args:d,env:u,tempFiles:n}}let l={};for(let d of o)l[d.name]=ic(d);let h=(0,ln.join)(a,`af-mcp.${r}.json`);return(0,ht.writeFileSync)(h,JSON.stringify({mcpServers:l},null,2),"utf-8"),n.push(h),{args:["--mcp-config",h],env:{},tempFiles:n}}catch(r){return console.warn("Agent Fleet: couldn't install MCP projection; run proceeds without fleet MCP.",r),ka(n),null}}function xt(i){i&&ka(i.tempFiles)}function ka(i){for(let t of i)try{(0,ht.existsSync)(t)&&(0,ht.unlinkSync)(t)}catch{}}function oc(i){return i.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"")||"server"}var hn=class{constructor(t,e,s){this.settings=t;this.repository=e;this.mcpAuth=s}runningProcesses=new Map;abortAgent(t){let e=this.runningProcesses.get(t);return e?(e.kill(),this.runningProcesses.delete(t),!0):!1}async buildPrompt(t,e,s,n=t.memory){let a=[t.body.trim()];for(let o of t.skills){let c=this.repository.getSkillByName(o);if(c){let l=[c.body.trim()];c.toolsBody.trim()&&l.push(`### Tools
11885
+ `;function ci(r){let t=[];for(let e of r){let s=e.trim();if(s)try{let n=JSON.parse(s),a=typeof n.text=="string"?n.text.trim():"";if(!a)continue;let i=n.section==="Preferences"||n.section==="Procedures"||n.section==="Observations"?n.section:void 0;t.push({text:a,pinned:n.pinned===!0,section:i,source:typeof n.source=="string"?n.source:"mcp"})}catch{}}return t}var sc=0;function nc(r,t){return{def:{name:ba,type:"stdio",enabled:!0,command:"node",env:{AF_PENDING_DIR:r,AF_SOURCE:t},status:"connected",scope:"user",tools:[],toolDetails:[]},inlineScript:li}}function cn(r){let t=r.agentGrants.map(n=>n.trim().toLowerCase()).filter(Boolean),e=t.length>0?new Set(t):null,s=[];for(let n of r.registry){if(!n.enabled||e&&!e.has(n.name.trim().toLowerCase())||n.type==="unknown")continue;let a={};if(n.type==="stdio"){if(n.envSecretKeys&&n.envSecretKeys.length>0){let i={};for(let o of n.envSecretKeys){let c=process.env[o];c&&(i[o]=c)}Object.keys(i).length>0&&(a.env=i)}}else if(n.auth!=="none"){let i=r.getBearerToken(n.name);i&&(a.bearerToken=i)}s.push({def:n,secrets:a})}return r.remember&&s.push(nc(r.remember.pendingDir,r.remember.source)),s}function ac(r){return`AF_MCP_${r.toUpperCase().replace(/[^A-Z0-9]+/g,"_").replace(/^_+|_+$/g,"")||"SERVER"}_TOKEN`}function di(r){return/^[A-Za-z0-9_-]+$/.test(r)?r:`"${r.replace(/"/g,'\\"')}"`}function rc(r,t){let{def:e,secrets:s}=r;if(e.type==="unknown")throw new Error(`server ${e.name} has unknown transport`);let n=e.type;if(n==="stdio"){let a=e.command??"node",i=t?[t]:e.args??[],o={...e.env??{},...s?.env??{}};return{name:e.name,type:n,command:a,args:i,env:o}}if(!e.url)throw new Error(`server ${e.name} (${n}) has no url`);return{name:e.name,type:n,url:e.url,headers:e.headers,bearerToken:s?.bearerToken,oauthResource:e.oauth?.resource,oauthClientId:e.oauth?.clientId}}function ic(r){if(r.type==="stdio"){let s={command:r.command,args:r.args??[]};return r.env&&Object.keys(r.env).length>0&&(s.env=r.env),s}let t={...r.headers??{}};r.bearerToken&&(t.Authorization=`Bearer ${r.bearerToken}`);let e={type:r.type,url:r.url};return Object.keys(t).length>0&&(e.headers=t),e}function oc(r){let t=di(r.name),e=[],s={};if(r.type==="stdio"){e.push("-c",`mcp_servers.${t}.command=${JSON.stringify(r.command??"node")}`),r.args&&r.args.length>0&&e.push("-c",`mcp_servers.${t}.args=${JSON.stringify(r.args)}`);for(let[n,a]of Object.entries(r.env??{}))e.push("-c",`mcp_servers.${t}.env.${di(n)}=${JSON.stringify(a)}`)}else{if(e.push("-c",`mcp_servers.${t}.url=${JSON.stringify(r.url)}`),r.bearerToken){let n=ac(r.name);s[n]=r.bearerToken,e.push("-c",`mcp_servers.${t}.bearer_token_env_var=${JSON.stringify(n)}`)}r.oauthResource&&e.push("-c",`mcp_servers.${t}.oauth_resource=${JSON.stringify(r.oauthResource)}`),r.oauthClientId&&e.push("-c",`mcp_servers.${t}.oauth.client_id=${JSON.stringify(r.oauthClientId)}`)}return e.push("-c",`mcp_servers.${t}.enabled=true`),{args:e,env:s}}function dn(r,t,e){if(e.length===0)return null;let s=kt(t)==="codex",n=[],a=(0,ln.join)(r,".claude");try{(0,ht.existsSync)(a)||(0,ht.mkdirSync)(a,{recursive:!0});let i=`${process.pid}-${Date.now()}-${sc++}`,o=[],c=0;for(let d of e){try{let u=null;d.inlineScript&&(u=(0,ln.join)(a,`af-mcp-${lc(d.def.name)}.${i}-${c}.cjs`),(0,ht.writeFileSync)(u,d.inlineScript,"utf-8"),n.push(u)),o.push(rc(d,u))}catch(u){console.warn(`Agent Fleet: skipping MCP server "${d.def.name}" in projection:`,u)}c++}if(o.length===0)return ka(n),null;if(s){let d=[],u={};for(let p of o){let m=oc(p);d.push(...m.args),Object.assign(u,m.env)}return{args:d,env:u,tempFiles:n}}let l={};for(let d of o)l[d.name]=ic(d);let h=(0,ln.join)(a,`af-mcp.${i}.json`);return(0,ht.writeFileSync)(h,JSON.stringify({mcpServers:l},null,2),"utf-8"),n.push(h),{args:["--mcp-config",h],env:{},tempFiles:n}}catch(i){return console.warn("Agent Fleet: couldn't install MCP projection; run proceeds without fleet MCP.",i),ka(n),null}}function xt(r){r&&ka(r.tempFiles)}function ka(r){for(let t of r)try{(0,ht.existsSync)(t)&&(0,ht.unlinkSync)(t)}catch{}}function lc(r){return r.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,"")||"server"}var hn=class{constructor(t,e,s){this.settings=t;this.repository=e;this.mcpAuth=s}runningProcesses=new Map;abortAgent(t){let e=this.runningProcesses.get(t);return e?(e.kill(),this.runningProcesses.delete(t),!0):!1}async buildPrompt(t,e,s,n=t.memory){let a=[t.body.trim()];for(let o of t.skills){let c=this.repository.getSkillByName(o);if(c){let l=[c.body.trim()];c.toolsBody.trim()&&l.push(`### Tools
11975
11886
  ${c.toolsBody.trim()}`),c.referencesBody.trim()&&l.push(`### References
11976
11887
  ${c.referencesBody.trim()}`),c.examplesBody.trim()&&l.push(`### Examples
11977
11888
  ${c.examplesBody.trim()}`),a.push(`## Skill: ${c.name}
@@ -11979,25 +11890,25 @@ ${l.join(`
11979
11890
 
11980
11891
  `)}`)}}if(t.skillsBody.trim()&&a.push(`## Agent Skills
11981
11892
  ${t.skillsBody.trim()}`),t.contextBody.trim()&&a.push(`## Agent Context
11982
- ${t.contextBody.trim()}`),n){let o=await this.repository.readWorkingMemory(t.name),c=Gs(t,o);c&&a.push(c)}let r=rn(t,this.repository);return r&&a.push(r),a.push(`## Task
11893
+ ${t.contextBody.trim()}`),n){let o=await this.repository.readWorkingMemory(t.name),c=Gs(t,o);c&&a.push(c)}let i=rn(t,this.repository);return i&&a.push(i),a.push(`## Task
11983
11894
  ${(s??e.body).trim()}`),a.filter(Boolean).join(`
11984
11895
 
11985
- `)}async execute(t,e,s,n,a){let r=t.memory&&!a?.suppressMemoryCapture,o=await this.buildPrompt(t,e,s,r),c=(0,dr.randomUUID)(),l=Dt(e,t,this.settings),h=n!=null,d=ir(t.adapter),u=await d.buildExec({prompt:o,model:ws(l.value)?l.value:"",modelSource:l.source,effort:e.effort||t.effort||"",agent:t,settings:this.settings,streaming:h}),p=t.cwd?.trim()?t.cwd:this.repository.getVaultBasePath()??".",m=r?this.repository.getPendingDirAbsolutePath(t.name):null,f=cn({registry:this.repository.getMcpServers(),agentGrants:t.mcpServers??[],getBearerToken:k=>this.mcpAuth?.getToken(k),remember:m?{pendingDir:m,source:"mcp"}:null}),g=dn(p,t.adapter,f);g&&u.args.push(...g.args);let y=await d.setupPermissions(p,t,this.settings,{mcpAllowServers:f.map(k=>k.def.name)}),v=Date.now();try{return await new Promise((k,w)=>{let S=dt(u.cliPath,u.args,{cwd:p,env:{...process.env,AWS_REGION:this.settings.awsRegion,...g?.env??{},...y?.env??{}}});if(u.stdinPayload!==void 0)try{S.stdin?.write(u.stdinPayload),S.stdin?.end()}catch(C){S.kill(),w(C instanceof Error?C:new Error(String(C)));return}this.runningProcesses.set(t.name,S);let T="",_="",D=!1,O=window.setTimeout(()=>{D=!0,S.kill()},t.timeout*1e3);S.stdout.on("data",C=>{let E=C.toString();if(T+=E,h&&n)for(let P of ye(E)){let N=d.extractStreamChunk(P);N&&n(N)}}),S.stderr.on("data",C=>{_+=C.toString()}),S.on("error",C=>{window.clearTimeout(O),w(C)}),S.on("close",C=>{window.clearTimeout(O),this.runningProcesses.delete(t.name);let E=d.parseExecOutput(T,_,h);k({runId:c,prompt:o,exitCode:C,durationSeconds:Math.max(1,Math.round((Date.now()-v)/1e3)),stdout:T,stderr:_,outputText:E.outputText,rawJson:E.rawJson,tokensUsed:E.tokensUsed,costUsd:E.costUsd,toolsUsed:E.toolsUsed,timedOut:D,resolvedModel:l.value,modelSource:l.source,concreteModel:E.concreteModel,finalResult:E.finalResult})})})}finally{y?.restore(),xt(g)}}};function hr(i){return oe(i).slice(0,60)||"candidate"}function pr(i){let t=Math.max(400,i.tokenBudget*4);return[`You are running a nightly memory **reflection** for the agent "${i.agentName}".`,"Review the raw memory log and your current working memory below, then produce an","updated, consolidated working memory.","","Rules:","- Remove duplicates; resolve contradictions in favor of the NEWEST fact.","- File each fact under exactly one of: Preferences, Procedures, Observations.","- Keep durable user/process preferences, marking them `[pin]` \u2014 never drop or"," summarize pinned Preferences.","- Summarize the Observations section FROM THE RAW LOG (not from prior summaries)",` if needed so the whole memory fits within roughly ${i.tokenBudget} tokens`,` (~${t} characters).`,"- Note any friction that recurred across multiple runs as a skill candidate.","","Reply with EXACTLY these two fenced blocks and nothing else:","","```memory","## Preferences","- [pin] <durable preference> <!-- src:reflection -->","## Procedures","- <how-to learning>","## Observations","- <current fact>","```","","```candidates",'[{"key":"short-stable-key","pattern":"what recurred","evidence":["runs/..."],"suggestedSkill":"optional-name"}]',"```","","If there is nothing worth remembering, still emit a `memory` block preserving the","existing pinned Preferences. Use an empty array `[]` for candidates when none.","","\u2500\u2500 CURRENT WORKING MEMORY \u2500\u2500",i.workingMemoryBody.trim()||"(empty)","","\u2500\u2500 RAW MEMORY LOG (ground truth) \u2500\u2500",i.recentRaw.trim()||"(empty)",i.recentRunsSummary?`
11896
+ `)}async execute(t,e,s,n,a){let i=t.memory&&!a?.suppressMemoryCapture,o=await this.buildPrompt(t,e,s,i),c=(0,hi.randomUUID)(),l=It(e,t,this.settings),h=n!=null,d=ii(t.adapter),u=await d.buildExec({prompt:o,model:ws(l.value)?l.value:"",modelSource:l.source,effort:e.effort||t.effort||"",agent:t,settings:this.settings,streaming:h}),p=t.cwd?.trim()?t.cwd:this.repository.getVaultBasePath()??".",m=i?this.repository.getPendingDirAbsolutePath(t.name):null,f=cn({registry:this.repository.getMcpServers(),agentGrants:t.mcpServers??[],getBearerToken:k=>this.mcpAuth?.getToken(k),remember:m?{pendingDir:m,source:"mcp"}:null}),g=dn(p,t.adapter,f);g&&u.args.push(...g.args);let y=await d.setupPermissions(p,t,this.settings,{mcpAllowServers:f.map(k=>k.def.name)}),v=Date.now();try{return await new Promise((k,w)=>{let S=dt(u.cliPath,u.args,{cwd:p,env:{...process.env,AWS_REGION:this.settings.awsRegion,...g?.env??{},...y?.env??{}}});if(u.stdinPayload!==void 0)try{S.stdin?.write(u.stdinPayload),S.stdin?.end()}catch(C){S.kill(),w(C instanceof Error?C:new Error(String(C)));return}this.runningProcesses.set(t.name,S);let T="",_="",D=!1,O=window.setTimeout(()=>{D=!0,S.kill()},t.timeout*1e3);S.stdout.on("data",C=>{let E=C.toString();if(T+=E,h&&n)for(let P of ye(E)){let N=d.extractStreamChunk(P);N&&n(N)}}),S.stderr.on("data",C=>{_+=C.toString()}),S.on("error",C=>{window.clearTimeout(O),w(C)}),S.on("close",C=>{window.clearTimeout(O),this.runningProcesses.delete(t.name);let E=d.parseExecOutput(T,_,h);k({runId:c,prompt:o,exitCode:C,durationSeconds:Math.max(1,Math.round((Date.now()-v)/1e3)),stdout:T,stderr:_,outputText:E.outputText,rawJson:E.rawJson,tokensUsed:E.tokensUsed,costUsd:E.costUsd,toolsUsed:E.toolsUsed,timedOut:D,resolvedModel:l.value,modelSource:l.source,concreteModel:E.concreteModel,finalResult:E.finalResult})})})}finally{y?.restore(),xt(g)}}};function ui(r){return oe(r).slice(0,60)||"candidate"}function mi(r){let t=Math.max(400,r.tokenBudget*4);return[`You are running a nightly memory **reflection** for the agent "${r.agentName}".`,"Review the raw memory log and your current working memory below, then produce an","updated, consolidated working memory.","","Rules:","- Remove duplicates; resolve contradictions in favor of the NEWEST fact.","- File each fact under exactly one of: Preferences, Procedures, Observations.","- Keep durable user/process preferences, marking them `[pin]` \u2014 never drop or"," summarize pinned Preferences.","- Summarize the Observations section FROM THE RAW LOG (not from prior summaries)",` if needed so the whole memory fits within roughly ${r.tokenBudget} tokens`,` (~${t} characters).`,"- Note any friction that recurred across multiple runs as a skill candidate.","","Reply with EXACTLY these two fenced blocks and nothing else:","","```memory","## Preferences","- [pin] <durable preference> <!-- src:reflection -->","## Procedures","- <how-to learning>","## Observations","- <current fact>","```","","```candidates",'[{"key":"short-stable-key","pattern":"what recurred","evidence":["runs/..."],"suggestedSkill":"optional-name"}]',"```","","If there is nothing worth remembering, still emit a `memory` block preserving the","existing pinned Preferences. Use an empty array `[]` for candidates when none.","","\u2500\u2500 CURRENT WORKING MEMORY \u2500\u2500",r.workingMemoryBody.trim()||"(empty)","","\u2500\u2500 RAW MEMORY LOG (ground truth) \u2500\u2500",r.recentRaw.trim()||"(empty)",r.recentRunsSummary?`
11986
11897
  \u2500\u2500 RECENT ACTIVITY \u2500\u2500
11987
- ${i.recentRunsSummary.trim()}`:""].join(`
11988
- `)}function ur(i,t){let e=new RegExp("```"+t+"(?![A-Za-z0-9-])[^\\n]*\\n([\\s\\S]*?)```","i"),s=i.match(e);return s?s[1]??"":null}function mr(i){let t=ur(i,"memory"),e=t!==null?zs(t):null,s=[],n=ur(i,"candidates");if(n!==null&&n.trim())try{let a=JSON.parse(n.trim());Array.isArray(a)&&(s=a.map(r=>lc(r)).filter(r=>r!==null))}catch{}return{sections:e,candidates:s}}function lc(i){if(typeof i!="object"||i===null)return null;let t=i,e=typeof t.pattern=="string"?t.pattern.trim():"";return e?{key:typeof t.key=="string"&&t.key.trim()?hr(t.key):hr(e),pattern:e,evidence:Array.isArray(t.evidence)?t.evidence.filter(n=>typeof n=="string"):[],suggestedSkill:typeof t.suggestedSkill=="string"?t.suggestedSkill:void 0}:null}function fr(i,t,e){let s=new Map;for(let a of i)s.set(a.key,{...a,evidence:[...a.evidence]});let n=new Map;for(let a of t){let r=n.get(a.key);r?(r.evidence=Array.from(new Set([...r.evidence??[],...a.evidence??[]])),a.suggestedSkill&&(r.suggestedSkill=a.suggestedSkill)):n.set(a.key,{...a,evidence:[...a.evidence??[]]})}for(let a of n.values()){let r=s.get(a.key);r?(r.occurrences+=1,r.lastSeen=e,r.evidence=Array.from(new Set([...r.evidence,...a.evidence??[]])),a.suggestedSkill&&(r.suggestedSkill=a.suggestedSkill)):s.set(a.key,{key:a.key,pattern:a.pattern,occurrences:1,firstSeen:e,lastSeen:e,evidence:[...a.evidence??[]],proposed:!1,suggestedSkill:a.suggestedSkill})}return[...s.values()]}var cc=1.5,zt=class i{constructor(t){this.store=t}static locks=new Map;static __resetLocksForTest(){i.locks.clear()}async capture(t,e,s,n){!t.memory||e.length===0||await this.withLock(t.name,()=>this.applyCaptures(t,e,s,n))}async drainPending(t,e){t.memory&&await this.withLock(t.name,async()=>{let s=await this.store.readAndClearPending(t.name),n=lr(s);if(n.length===0)return;let a=[];for(let r of n){let o=a[a.length-1],c={text:r.text,pinned:r.pinned,section:r.section};o&&o.source===r.source?o.entries.push(c):a.push({source:r.source,entries:[c]})}for(let r of a)await this.applyCaptures(t,r.entries,r.source,e)})}async reflect(t,e,s){return!t.memory||e===null||!e.some(n=>n.entries.length>0)?!1:(await this.withLock(t.name,async()=>{let n=this.store.getWorkingMemoryPath(t.name),a=await this.store.readWorkingMemory(t.name),r=bi(a,e),o={filePath:a?.filePath??n,agent:t.name,schema:a?.schema??2,lastUpdated:s,lastReflection:s,tokenEstimate:0,sections:r};await this.store.writeWorkingMemory(t.name,o)}),!0)}async applyCaptures(t,e,s,n){let a=e.map(p=>({...p,text:pi(p.text)})).filter(p=>p.text);if(a.length===0)return;await this.store.migrateLegacyMemory?.(t.name);let r=this.store.getWorkingMemoryPath(t.name),o=await this.store.readWorkingMemory(t.name)??mi(r,t.name),c=n.slice(0,10),l=a.map(p=>`- ${n} [${s}]${p.pinned?" [pin]":""} ${p.text}`);await this.store.appendRawMemory(t.name,l,n);let h=new Set(o.sections.flatMap(p=>p.entries).map(p=>p.text.trim().toLowerCase())),d=o;for(let p of a){let m=p.text.trim().toLowerCase();if(h.has(m))continue;h.add(m);let f={text:p.text,source:s,date:c,pinned:p.pinned??!1},g=p.section??(p.pinned?"Preferences":hi);d=vi(d,[f],g,n)}let u=Math.ceil((t.memoryTokenBudget||0)*cc);if(u>0){let{wm:p,spilled:m}=wi(d,u);m.length>0&&console.info(`Agent Fleet: working memory for "${t.name}" exceeded hard cap; spilled ${m.length} entr${m.length===1?"y":"ies"} to raw-only (still in the archive). A reflection pass will consolidate.`),d=p}await this.store.writeWorkingMemory(t.name,d)}withLock(t,e){let n=(i.locks.get(t)??Promise.resolve()).then(e,e);return i.locks.set(t,n.then(()=>{},()=>{})),n}};var bs=Ye(require("crypto")),gr=Ye(require("https")),un=Ye(require("http")),yr=Ye(require("fs")),xa=Ye(require("path"));var pn=class{constructor(t){this.settings=t;this.settings}authManager;setAuthManager(t){this.authManager=t}async probeServer(t){try{if(t.type==="stdio"&&t.command)return(await this.probeStdioServer(t.command,t.args)).tools;if((t.type==="http"||t.type==="sse")&&t.url)return await this.probeHttpServer(t)}catch(e){console.warn(`McpManager: probe failed for ${t.name}:`,e)}return[]}async authenticateServer(t,e,s="http"){let n=await this.discoverOAuthMetadata(e);if(!n)throw new Error("Server does not support OAuth \u2014 no authorization metadata found.");let{metadata:a,resourceScopes:r,resource:o}=n;if(!a.registration_endpoint)throw new Error("Server does not support Dynamic Client Registration.");let c=await this.startOAuthCallbackServer(),l=`http://localhost:${c.port}/callback`;try{let h=await this.registerOAuthClient(a.registration_endpoint,l),d=bs.randomBytes(32).toString("base64url"),u=bs.createHash("sha256").update(d).digest("base64url"),p=bs.randomBytes(16).toString("hex"),m=new URLSearchParams({response_type:"code",client_id:h,code_challenge:u,code_challenge_method:"S256",redirect_uri:l,state:p,resource:o}),f=r??a.scopes_supported;f?.length&&m.set("scope",f.join(" "));let g=new URL(a.authorization_endpoint);for(let[w,S]of m)g.searchParams.set(w,S);let y=g.toString();Ci(y);let v=await c.waitForCode(p,18e4),k=await this.exchangeOAuthCode(a.token_endpoint,v,l,h,d);this.authManager?.storeOAuthToken(t,{accessToken:k.access_token,refreshToken:k.refresh_token,expiresAt:k.expires_in?Date.now()+k.expires_in*1e3:void 0,tokenEndpoint:a.token_endpoint,clientId:h,resource:o})}finally{c.close()}}async refreshProbeTokens(){if(!this.authManager)return;let t=this.authManager.getExpiringTokens();for(let[e,s]of t)if(s.refreshToken)try{let n=new URLSearchParams({grant_type:"refresh_token",refresh_token:s.refreshToken,client_id:s.clientId}).toString(),a=await this.oauthFetch(s.tokenEndpoint,"POST",n,"application/x-www-form-urlencoded");if(a.status===200){let r=JSON.parse(a.body),o=r.access_token;o&&this.authManager.storeOAuthToken(e,{accessToken:o,refreshToken:r.refresh_token??s.refreshToken,expiresAt:typeof r.expires_in=="number"?Date.now()+r.expires_in*1e3:void 0,tokenEndpoint:s.tokenEndpoint,clientId:s.clientId,resource:s.resource})}}catch(n){console.warn(`McpManager: failed to refresh token for ${e}:`,n)}}async discoverOAuthMetadata(t){let e=t.endsWith("/sse")?t.replace(/\/sse$/,"/mcp"):t,s=new URL(e),n;try{let d=await this.oauthFetch(e,"POST","{}","application/json");d.status===401&&(n=d.headers?.["www-authenticate"]?.match(/resource_metadata="([^"]+)"/)?.[1])}catch{}n||(n=`${s.origin}/.well-known/oauth-protected-resource${s.pathname}`);let a=s.origin,r,o=e;try{let d=await this.oauthFetch(n,"GET");if(d.status===200){let u=JSON.parse(d.body),p=u.authorization_servers;p?.[0]&&(a=p[0]),Array.isArray(u.scopes_supported)&&(r=u.scopes_supported),typeof u.resource=="string"&&(o=u.resource)}}catch{}let c=new URL(a),l=c.pathname==="/"?"":c.pathname,h=`${c.origin}/.well-known/oauth-authorization-server${l}`;try{let d=await this.oauthFetch(h,"GET");if(d.status===200)return{metadata:JSON.parse(d.body),resourceScopes:r,resource:o}}catch{}if(l){let d=`${c.origin}/.well-known/oauth-authorization-server`;try{let u=await this.oauthFetch(d,"GET");if(u.status===200)return{metadata:JSON.parse(u.body),resourceScopes:r,resource:o}}catch{}}return null}async registerOAuthClient(t,e){let s=JSON.stringify({client_name:"Agent Fleet Obsidian Plugin",redirect_uris:[e],grant_types:["authorization_code","refresh_token"],response_types:["code"],token_endpoint_auth_method:"none"}),n=await this.oauthFetch(t,"POST",s,"application/json");if(n.status!==200&&n.status!==201)throw new Error(`Client registration failed (HTTP ${n.status})`);let a=JSON.parse(n.body);if(!a.client_id)throw new Error("Client registration response missing client_id");return a.client_id}async startOAuthCallbackServer(){let t=null,e=null,s=un.createServer((a,r)=>{let o=new URL(a.url??"/","http://localhost");if(o.pathname!=="/callback"){r.writeHead(404),r.end();return}let c=o.searchParams.get("error");if(c){let d=o.searchParams.get("error_description")??c;r.writeHead(200,{"Content-Type":"text/html"}),r.end("<html><body style='font-family:system-ui;text-align:center;padding:60px'><h2>Authorization Failed</h2><p>"+d+"</p><p style='color:#888'>You can close this tab.</p></body></html>"),e?.(new Error(`OAuth denied: ${d}`));return}let l=o.searchParams.get("code"),h=o.searchParams.get("state");if(!l||!h){r.writeHead(400),r.end("Missing code or state");return}r.writeHead(200,{"Content-Type":"text/html"}),r.end("<html><body style='font-family:system-ui;text-align:center;padding:60px'><h2 style='color:#22c55e'>Authenticated!</h2><p>You can close this tab and return to Obsidian.</p><script>window.setTimeout(()=>window.close(),2000)</script></body></html>"),t?.(l,h)});return{port:await new Promise((a,r)=>{s.listen(0,"127.0.0.1",()=>{let o=s.address();if(!o||typeof o=="string"){r(new Error("Failed to bind callback server"));return}a(o.port)}),s.on("error",r)}),waitForCode:(a,r)=>new Promise((o,c)=>{let l=window.setTimeout(()=>{c(new Error("Authentication timed out \u2014 complete authorization in your browser and try again."))},r);t=(h,d)=>{window.clearTimeout(l),d!==a?c(new Error("OAuth state mismatch \u2014 possible CSRF attack")):o(h)},e=h=>{window.clearTimeout(l),c(h)}}),close:()=>{try{s.close()}catch{}}}}async exchangeOAuthCode(t,e,s,n,a){let r=new URLSearchParams({grant_type:"authorization_code",code:e,redirect_uri:s,client_id:n,code_verifier:a}).toString(),o=await this.oauthFetch(t,"POST",r,"application/x-www-form-urlencoded");if(o.status!==200)throw new Error(`Token exchange failed (HTTP ${o.status}): ${o.body}`);let c=JSON.parse(o.body);if(!c.access_token)throw new Error("Token response missing access_token");return c}oauthFetch(t,e,s,n){return new Promise((a,r)=>{let o=new URL(t),c=o.protocol==="https:",l={Accept:"application/json"};s&&(l["Content-Type"]=n??"application/json",l["Content-Length"]=String(Buffer.byteLength(s)));let h={hostname:o.hostname,port:o.port||(c?443:80),path:o.pathname+o.search,method:e,headers:l},u=(c?gr:un).request(h,m=>{let f="";m.on("data",g=>{f+=g.toString()}),m.on("end",()=>{let g={};for(let[y,v]of Object.entries(m.headers))typeof v=="string"?g[y]=v:Array.isArray(v)&&(g[y]=v.join(", "));a({status:m.statusCode??0,body:f,headers:g})})});u.on("error",r);let p=window.setTimeout(()=>{u.destroy(),r(new Error("OAuth request timed out"))},15e3);u.on("close",()=>window.clearTimeout(p)),s&&u.write(s),u.end()})}probeStdioServer(t,e){return new Promise(s=>{let n=e&&e.length>0?`${t} ${e.join(" ")}`:t,a=Si(n,{env:{...process.env}}),r="",o,c=[],l=!1,h=!1,d=!1,u=()=>{l||(l=!0,a.kill(),s({description:o,tools:c}))},p=window.setTimeout(u,1e4);a.stdout.on("data",m=>{r+=m.toString();let f=ye(r);r=f.pop()??"";for(let g of f){let y=g.trim();if(y){try{let v=JSON.parse(y);if(v.id===1&&v.result){o=v.result.instructions??v.result.serverInfo?.description,h=!0;try{a.stdin.write(JSON.stringify({jsonrpc:"2.0",method:"notifications/initialized"})+`
11898
+ ${r.recentRunsSummary.trim()}`:""].join(`
11899
+ `)}function pi(r,t){let e=new RegExp("```"+t+"(?![A-Za-z0-9-])[^\\n]*\\n([\\s\\S]*?)```","i"),s=r.match(e);return s?s[1]??"":null}function fi(r){let t=pi(r,"memory"),e=t!==null?zs(t):null,s=[],n=pi(r,"candidates");if(n!==null&&n.trim())try{let a=JSON.parse(n.trim());Array.isArray(a)&&(s=a.map(i=>cc(i)).filter(i=>i!==null))}catch{}return{sections:e,candidates:s}}function cc(r){if(typeof r!="object"||r===null)return null;let t=r,e=typeof t.pattern=="string"?t.pattern.trim():"";return e?{key:typeof t.key=="string"&&t.key.trim()?ui(t.key):ui(e),pattern:e,evidence:Array.isArray(t.evidence)?t.evidence.filter(n=>typeof n=="string"):[],suggestedSkill:typeof t.suggestedSkill=="string"?t.suggestedSkill:void 0}:null}function gi(r,t,e){let s=new Map;for(let a of r)s.set(a.key,{...a,evidence:[...a.evidence]});let n=new Map;for(let a of t){let i=n.get(a.key);i?(i.evidence=Array.from(new Set([...i.evidence??[],...a.evidence??[]])),a.suggestedSkill&&(i.suggestedSkill=a.suggestedSkill)):n.set(a.key,{...a,evidence:[...a.evidence??[]]})}for(let a of n.values()){let i=s.get(a.key);i?(i.occurrences+=1,i.lastSeen=e,i.evidence=Array.from(new Set([...i.evidence,...a.evidence??[]])),a.suggestedSkill&&(i.suggestedSkill=a.suggestedSkill)):s.set(a.key,{key:a.key,pattern:a.pattern,occurrences:1,firstSeen:e,lastSeen:e,evidence:[...a.evidence??[]],proposed:!1,suggestedSkill:a.suggestedSkill})}return[...s.values()]}var dc=1.5,Gt=class r{constructor(t){this.store=t}static locks=new Map;static __resetLocksForTest(){r.locks.clear()}async capture(t,e,s,n){!t.memory||e.length===0||await this.withLock(t.name,()=>this.applyCaptures(t,e,s,n))}async drainPending(t,e){t.memory&&await this.withLock(t.name,async()=>{let s=await this.store.readAndClearPending(t.name),n=ci(s);if(n.length===0)return;let a=[];for(let i of n){let o=a[a.length-1],c={text:i.text,pinned:i.pinned,section:i.section};o&&o.source===i.source?o.entries.push(c):a.push({source:i.source,entries:[c]})}for(let i of a)await this.applyCaptures(t,i.entries,i.source,e)})}async reflect(t,e,s){return!t.memory||e===null||!e.some(n=>n.entries.length>0)?!1:(await this.withLock(t.name,async()=>{let n=this.store.getWorkingMemoryPath(t.name),a=await this.store.readWorkingMemory(t.name),i=br(a,e),o={filePath:a?.filePath??n,agent:t.name,schema:a?.schema??2,lastUpdated:s,lastReflection:s,tokenEstimate:0,sections:i};await this.store.writeWorkingMemory(t.name,o)}),!0)}async applyCaptures(t,e,s,n){let a=e.map(p=>({...p,text:pr(p.text)})).filter(p=>p.text);if(a.length===0)return;await this.store.migrateLegacyMemory?.(t.name);let i=this.store.getWorkingMemoryPath(t.name),o=await this.store.readWorkingMemory(t.name)??mr(i,t.name),c=n.slice(0,10),l=a.map(p=>`- ${n} [${s}]${p.pinned?" [pin]":""} ${p.text}`);await this.store.appendRawMemory(t.name,l,n);let h=new Set(o.sections.flatMap(p=>p.entries).map(p=>p.text.trim().toLowerCase())),d=o;for(let p of a){let m=p.text.trim().toLowerCase();if(h.has(m))continue;h.add(m);let f={text:p.text,source:s,date:c,pinned:p.pinned??!1},g=p.section??(p.pinned?"Preferences":hr);d=vr(d,[f],g,n)}let u=Math.ceil((t.memoryTokenBudget||0)*dc);if(u>0){let{wm:p,spilled:m}=wr(d,u);m.length>0&&console.info(`Agent Fleet: working memory for "${t.name}" exceeded hard cap; spilled ${m.length} entr${m.length===1?"y":"ies"} to raw-only (still in the archive). A reflection pass will consolidate.`),d=p}await this.store.writeWorkingMemory(t.name,d)}withLock(t,e){let n=(r.locks.get(t)??Promise.resolve()).then(e,e);return r.locks.set(t,n.then(()=>{},()=>{})),n}};var bs=Ye(require("crypto")),yi=Ye(require("https")),un=Ye(require("http")),vi=Ye(require("fs")),xa=Ye(require("path"));var pn=class{constructor(t){this.settings=t;this.settings}authManager;setAuthManager(t){this.authManager=t}async probeServer(t){try{if(t.type==="stdio"&&t.command)return(await this.probeStdioServer(t.command,t.args)).tools;if((t.type==="http"||t.type==="sse")&&t.url)return await this.probeHttpServer(t)}catch(e){console.warn(`McpManager: probe failed for ${t.name}:`,e)}return[]}async authenticateServer(t,e,s="http"){let n=await this.discoverOAuthMetadata(e);if(!n)throw new Error("Server does not support OAuth \u2014 no authorization metadata found.");let{metadata:a,resourceScopes:i,resource:o}=n;if(!a.registration_endpoint)throw new Error("Server does not support Dynamic Client Registration.");let c=await this.startOAuthCallbackServer(),l=`http://localhost:${c.port}/callback`;try{let h=await this.registerOAuthClient(a.registration_endpoint,l),d=bs.randomBytes(32).toString("base64url"),u=bs.createHash("sha256").update(d).digest("base64url"),p=bs.randomBytes(16).toString("hex"),m=new URLSearchParams({response_type:"code",client_id:h,code_challenge:u,code_challenge_method:"S256",redirect_uri:l,state:p,resource:o}),f=i??a.scopes_supported;f?.length&&m.set("scope",f.join(" "));let g=new URL(a.authorization_endpoint);for(let[w,S]of m)g.searchParams.set(w,S);let y=g.toString();Cr(y);let v=await c.waitForCode(p,18e4),k=await this.exchangeOAuthCode(a.token_endpoint,v,l,h,d);this.authManager?.storeOAuthToken(t,{accessToken:k.access_token,refreshToken:k.refresh_token,expiresAt:k.expires_in?Date.now()+k.expires_in*1e3:void 0,tokenEndpoint:a.token_endpoint,clientId:h,resource:o})}finally{c.close()}}async refreshProbeTokens(){if(!this.authManager)return;let t=this.authManager.getExpiringTokens();for(let[e,s]of t)if(s.refreshToken)try{let n=new URLSearchParams({grant_type:"refresh_token",refresh_token:s.refreshToken,client_id:s.clientId}).toString(),a=await this.oauthFetch(s.tokenEndpoint,"POST",n,"application/x-www-form-urlencoded");if(a.status===200){let i=JSON.parse(a.body),o=i.access_token;o&&this.authManager.storeOAuthToken(e,{accessToken:o,refreshToken:i.refresh_token??s.refreshToken,expiresAt:typeof i.expires_in=="number"?Date.now()+i.expires_in*1e3:void 0,tokenEndpoint:s.tokenEndpoint,clientId:s.clientId,resource:s.resource})}}catch(n){console.warn(`McpManager: failed to refresh token for ${e}:`,n)}}async discoverOAuthMetadata(t){let e=t.endsWith("/sse")?t.replace(/\/sse$/,"/mcp"):t,s=new URL(e),n;try{let d=await this.oauthFetch(e,"POST","{}","application/json");d.status===401&&(n=d.headers?.["www-authenticate"]?.match(/resource_metadata="([^"]+)"/)?.[1])}catch{}n||(n=`${s.origin}/.well-known/oauth-protected-resource${s.pathname}`);let a=s.origin,i,o=e;try{let d=await this.oauthFetch(n,"GET");if(d.status===200){let u=JSON.parse(d.body),p=u.authorization_servers;p?.[0]&&(a=p[0]),Array.isArray(u.scopes_supported)&&(i=u.scopes_supported),typeof u.resource=="string"&&(o=u.resource)}}catch{}let c=new URL(a),l=c.pathname==="/"?"":c.pathname,h=`${c.origin}/.well-known/oauth-authorization-server${l}`;try{let d=await this.oauthFetch(h,"GET");if(d.status===200)return{metadata:JSON.parse(d.body),resourceScopes:i,resource:o}}catch{}if(l){let d=`${c.origin}/.well-known/oauth-authorization-server`;try{let u=await this.oauthFetch(d,"GET");if(u.status===200)return{metadata:JSON.parse(u.body),resourceScopes:i,resource:o}}catch{}}return null}async registerOAuthClient(t,e){let s=JSON.stringify({client_name:"Agent Fleet Obsidian Plugin",redirect_uris:[e],grant_types:["authorization_code","refresh_token"],response_types:["code"],token_endpoint_auth_method:"none"}),n=await this.oauthFetch(t,"POST",s,"application/json");if(n.status!==200&&n.status!==201)throw new Error(`Client registration failed (HTTP ${n.status})`);let a=JSON.parse(n.body);if(!a.client_id)throw new Error("Client registration response missing client_id");return a.client_id}async startOAuthCallbackServer(){let t=null,e=null,s=un.createServer((a,i)=>{let o=new URL(a.url??"/","http://localhost");if(o.pathname!=="/callback"){i.writeHead(404),i.end();return}let c=o.searchParams.get("error");if(c){let d=o.searchParams.get("error_description")??c;i.writeHead(200,{"Content-Type":"text/html"}),i.end("<html><body style='font-family:system-ui;text-align:center;padding:60px'><h2>Authorization Failed</h2><p>"+d+"</p><p style='color:#888'>You can close this tab.</p></body></html>"),e?.(new Error(`OAuth denied: ${d}`));return}let l=o.searchParams.get("code"),h=o.searchParams.get("state");if(!l||!h){i.writeHead(400),i.end("Missing code or state");return}i.writeHead(200,{"Content-Type":"text/html"}),i.end("<html><body style='font-family:system-ui;text-align:center;padding:60px'><h2 style='color:#22c55e'>Authenticated!</h2><p>You can close this tab and return to Obsidian.</p><script>window.setTimeout(()=>window.close(),2000)</script></body></html>"),t?.(l,h)});return{port:await new Promise((a,i)=>{s.listen(0,"127.0.0.1",()=>{let o=s.address();if(!o||typeof o=="string"){i(new Error("Failed to bind callback server"));return}a(o.port)}),s.on("error",i)}),waitForCode:(a,i)=>new Promise((o,c)=>{let l=window.setTimeout(()=>{c(new Error("Authentication timed out \u2014 complete authorization in your browser and try again."))},i);t=(h,d)=>{window.clearTimeout(l),d!==a?c(new Error("OAuth state mismatch \u2014 possible CSRF attack")):o(h)},e=h=>{window.clearTimeout(l),c(h)}}),close:()=>{try{s.close()}catch{}}}}async exchangeOAuthCode(t,e,s,n,a){let i=new URLSearchParams({grant_type:"authorization_code",code:e,redirect_uri:s,client_id:n,code_verifier:a}).toString(),o=await this.oauthFetch(t,"POST",i,"application/x-www-form-urlencoded");if(o.status!==200)throw new Error(`Token exchange failed (HTTP ${o.status}): ${o.body}`);let c=JSON.parse(o.body);if(!c.access_token)throw new Error("Token response missing access_token");return c}oauthFetch(t,e,s,n){return new Promise((a,i)=>{let o=new URL(t),c=o.protocol==="https:",l={Accept:"application/json"};s&&(l["Content-Type"]=n??"application/json",l["Content-Length"]=String(Buffer.byteLength(s)));let h={hostname:o.hostname,port:o.port||(c?443:80),path:o.pathname+o.search,method:e,headers:l},u=(c?yi:un).request(h,m=>{let f="";m.on("data",g=>{f+=g.toString()}),m.on("end",()=>{let g={};for(let[y,v]of Object.entries(m.headers))typeof v=="string"?g[y]=v:Array.isArray(v)&&(g[y]=v.join(", "));a({status:m.statusCode??0,body:f,headers:g})})});u.on("error",i);let p=window.setTimeout(()=>{u.destroy(),i(new Error("OAuth request timed out"))},15e3);u.on("close",()=>window.clearTimeout(p)),s&&u.write(s),u.end()})}probeStdioServer(t,e){return new Promise(s=>{let n=e&&e.length>0?`${t} ${e.join(" ")}`:t,a=Sr(n,{env:{...process.env}}),i="",o,c=[],l=!1,h=!1,d=!1,u=()=>{l||(l=!0,a.kill(),s({description:o,tools:c}))},p=window.setTimeout(u,1e4);a.stdout.on("data",m=>{i+=m.toString();let f=ye(i);i=f.pop()??"";for(let g of f){let y=g.trim();if(y){try{let v=JSON.parse(y);if(v.id===1&&v.result){o=v.result.instructions??v.result.serverInfo?.description,h=!0;try{a.stdin.write(JSON.stringify({jsonrpc:"2.0",method:"notifications/initialized"})+`
11989
11900
  `),a.stdin.write(JSON.stringify({jsonrpc:"2.0",id:2,method:"tools/list"})+`
11990
11901
  `)}catch{window.clearTimeout(p),u();return}}else if(v.id===2&&v.result){for(let k of v.result.tools??[])c.push({name:k.name,description:k.description,inputSchema:k.inputSchema});d=!0}}catch{}h&&d&&(window.clearTimeout(p),u())}}}),a.on("error",()=>{window.clearTimeout(p),u()}),a.on("close",()=>{window.clearTimeout(p),u()});try{a.stdin.write(JSON.stringify({jsonrpc:"2.0",id:1,method:"initialize",params:{protocolVersion:"2024-11-05",capabilities:{},clientInfo:{name:"agent-fleet",version:"1.0.0"}}})+`
11991
- `)}catch{window.clearTimeout(p),u()}})}async probeHttpServer(t){let e=await this.findServerToken(t);if(!e)return[];let s=t.url.endsWith("/sse")?t.url.replace(/\/sse$/,"/mcp"):t.url;try{let a=(await this.httpRequest(s,e,{jsonrpc:"2.0",id:1,method:"initialize",params:{protocolVersion:"2025-03-26",capabilities:{},clientInfo:{name:"agent-fleet",version:"1.0.0"}}}))?._sessionId;await this.httpRequest(s,e,{jsonrpc:"2.0",method:"notifications/initialized"},a);let r=await this.httpRequest(s,e,{jsonrpc:"2.0",id:2,method:"tools/list"},a),o=[],h=r?.result?.tools??[];for(let d of h)o.push({name:d.name,description:d.description,inputSchema:d.inputSchema});return o}catch(n){return console.warn(`McpManager: HTTP probe failed for ${t.name}:`,n),[]}}async findServerToken(t){if(this.authManager){let a=this.authManager.getToken(t.name);if(a)return a}let e=t.name.replace(/^claude\.ai\s+/i,"").replace(/\s+/g,"_").toUpperCase(),s=[`${e}_API_KEY`,`${e}_API_KEY_MILO`,`${e}_TOKEN`];for(let a of s){let r=process.env[a];if(r)return r}let n=[xa.join(ea(),".openclaw","workspace",".env"),xa.join(ea(),".env")];for(let a of n)try{let r=yr.readFileSync(a,"utf8");for(let o of ye(r)){let c=o.trim().match(/^(?:export\s+)?([A-Za-z_]\w*)=(.*)$/);if(c){let l=c[1],h=c[2].replace(/^["']|["']$/g,"");if(s.includes(l))return h}}}catch{}}httpRequest(t,e,s,n){return new Promise((a,r)=>{let o=JSON.stringify(s),c=new URL(t),l=c.protocol==="https:",h={"Content-Type":"application/json",Accept:"application/json, text/event-stream",Authorization:`Bearer ${e}`,"Content-Length":String(Buffer.byteLength(o))};n&&(h["mcp-session-id"]=n);let d={hostname:c.hostname,port:c.port||(l?443:80),path:c.pathname+c.search,method:"POST",headers:h},p=(l?gr:un).request(d,f=>{let g="";f.on("data",y=>{g+=y.toString()}),f.on("end",()=>{let y=f.headers["mcp-session-id"];if((f.headers["content-type"]??"").includes("text/event-stream")){for(let k of ye(g))if(k.startsWith("data: "))try{let w=JSON.parse(k.slice(6));y&&(w._sessionId=y),a(w);return}catch{}a(null)}else try{let k=JSON.parse(g);y&&(k._sessionId=y),a(k)}catch{a(null)}})});p.on("error",r);let m=window.setTimeout(()=>{p.destroy(),a(null)},15e3);p.on("close",()=>window.clearTimeout(m)),p.write(o),p.end()})}};var dc={"every 5m":"*/5 * * * *","every 10m":"*/10 * * * *","every 15m":"*/15 * * * *","every 30m":"*/30 * * * *","every 1h":"0 * * * *","every 2h":"0 */2 * * *",hourly:"0 * * * *","daily at 9am":"0 9 * * *","daily at 6pm":"0 18 * * *","weekdays at 9am":"0 9 * * 1-5","weekly on monday":"0 9 * * 1","monthly on 1st":"0 9 1 * *"},mn=class{constructor(t,e){this.maxConcurrentRuns=t;this.callbacks=e}jobs=new Map;activeRuns=0;queue=[];paused=!1;setMaxConcurrentRuns(t){this.maxConcurrentRuns=t}parseSchedule(t){return dc[t.toLowerCase()]??t}toLocalISOString(t){let e=s=>String(s).padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}T${e(t.getHours())}:${e(t.getMinutes())}:${e(t.getSeconds())}`}async registerTask(t){if(this.unregisterTask(t.taskId),!(!t.enabled||t.type==="immediate"))try{if(t.type==="once"&&t.runAt){let e=new pe(new Date(t.runAt),{name:t.taskId,catch:!0},()=>{this.enqueue({task:t,reason:"scheduled"})});this.jobs.set(t.taskId,e),await this.callbacks.onTaskScheduled(t,t.runAt);return}if(t.type==="recurring"&&t.schedule){let e=this.parseSchedule(t.schedule),s=new pe(e,{name:t.taskId,catch:!0,protect:!0,timezone:Intl.DateTimeFormat().resolvedOptions().timeZone},()=>{this.enqueue({task:t,reason:"scheduled"})});this.jobs.set(t.taskId,s);let n=s.nextRun();await this.callbacks.onTaskScheduled(t,n?this.toLocalISOString(n):void 0)}}catch(e){console.error(`Agent Fleet: Failed to register task "${t.taskId}":`,e)}}unregisterTask(t){let e=this.jobs.get(t);e&&(e.stop(),this.jobs.delete(t))}async loadTasks(t){for(let e of t)await this.registerTask(e)}async handleStartupCatchUp(t){let e=Date.now();for(let s of t)!s.enabled||!s.catchUp||!s.nextRun||new Date(s.nextRun).getTime()<e&&await this.enqueue({task:s,reason:"catch-up"})}async enqueue(t){this.queue.push(t),await this.processQueue()}pauseAll(){this.paused=!0,this.jobs.forEach(t=>t.pause())}resumeAll(){this.paused=!1,this.jobs.forEach(t=>t.resume()),this.processQueue()}shutdown(){this.paused=!0,this.jobs.forEach(t=>t.stop()),this.jobs.clear(),this.queue.length=0}getQueueSize(){return this.queue.length}async processQueue(){if(!this.paused)for(;this.activeRuns<this.maxConcurrentRuns&&this.queue.length>0;){let t=this.queue.shift();if(!t)return;this.activeRuns+=1,this.callbacks.onTaskTriggered(t).finally(async()=>{this.activeRuns-=1,await this.processQueue()})}}};var ks=class i{constructor(t,e,s){this.repository=t;this.settings=e;this.executor=new hn(e,t,s),this.mcpManager=new pn(e),s&&this.mcpManager.setAuthManager(s),this.memoryWriter=new zt(t),this.scheduler=new mn(e.maxConcurrentRuns,{onTaskTriggered:n=>this.runPendingTask(n),onTaskScheduled:(n,a)=>this.repository.updateTaskRunMetadata(n,{nextRun:a})})}scheduler;executor;mcpManager;snapshot={agents:[],skills:[],tasks:[],channels:[],mcpServers:[],validationIssues:[]};runtimeState=new Map;recentRuns=[];chartRuns=[];static CHART_WINDOW_DAYS=14;recentUsage=[];statusChangeListeners=new Set;runOutputListeners=new Map;runOutputBuffers=new Map;heartbeatJobs=new Map;reflectionJobs=new Map;reflectionsInFlight=new Set;reflectionRunning=0;reflectionWaiters=[];heartbeatRegisteredAt=0;heartbeatsInFlight=new Set;channelResultHandler;memoryWriter;async initialize(){this.snapshot=await this.repository.loadAll(),await this.repository.migrateAllLegacyMemory(),await this.refreshRunCaches();let t=this.snapshot.tasks.filter(e=>this.repository.getAgentByName(e.agent)?.enabled!==!1);await this.scheduler.loadTasks(t),await this.scheduler.handleStartupCatchUp(t),this.registerHeartbeats(),this.registerReflections(),this.emitStatusChange()}onChannelResult(t){this.channelResultHandler=t}async refreshFromVault(){this.snapshot=await this.repository.loadAll(),await this.rebuildSchedules(),await this.refreshRunCaches(),this.emitStatusChange()}getSnapshot(){return this.snapshot}getRecentRuns(){return this.recentRuns}getChartRuns(){return this.chartRuns}async refreshRunCaches(){this.recentRuns=await this.repository.listRecentRuns();let t=new Date;t.setDate(t.getDate()-(i.CHART_WINDOW_DAYS-1)),this.chartRuns=await this.repository.listRunsSince(t),this.recentUsage=await this.repository.readUsageSince(t)}getUsageRecords(){return this.recentUsage}recordUsage(t){this.recentUsage.push(t),this.repository.appendUsage(t).catch(e=>{console.warn("Agent Fleet: failed to append usage record",e)}),this.emitStatusChange()}getAgentState(t){let e=this.runtimeState.get(t),s=this.snapshot.agents.find(n=>n.name===t);return s&&!s.enabled?{status:"disabled",lastRun:e?.lastRun,currentRunId:e?.currentRunId}:e??{status:"idle"}}getFleetStatus(){let t=new Date,e=o=>String(o).padStart(2,"0"),s=`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}`,n=this.recentRuns.filter(o=>{let c=new Date(o.started);return`${c.getFullYear()}-${e(c.getMonth()+1)}-${e(c.getDate())}`===s}).length,a=Array.from(this.runtimeState.values()).filter(o=>o.status==="running").length,r=this.recentRuns.flatMap(o=>o.approvals??[]).filter(o=>o.status==="pending").length;return{running:a,pending:r,completedToday:n}}subscribe(t){return this.statusChangeListeners.add(t),()=>this.statusChangeListeners.delete(t)}onRunOutput(t,e){let s=this.runOutputListeners.get(t);s||(s=new Set,this.runOutputListeners.set(t,s)),s.add(e);let n=this.runOutputBuffers.get(t);return n&&e(n),()=>{this.runOutputListeners.get(t)?.delete(e)}}getRunOutputBuffer(t){return this.runOutputBuffers.get(t)??""}async handleVaultChange(t){await this.repository.loadFile(t),this.snapshot=this.repository.getSnapshot(),await this.rebuildSchedules(),await this.refreshRunCaches(),this.emitStatusChange()}async handleVaultDelete(t){this.repository.removeFile(t),this.snapshot=this.repository.getSnapshot(),await this.rebuildSchedules(),await this.refreshRunCaches(),this.emitStatusChange()}abortedAgents=new Set;abortAgentRun(t){let e=this.executor.abortAgent(t);return e&&(this.abortedAgents.add(t),this.runtimeState.set(t,{status:"idle"}),this.emitStatusChange()),e}wasAborted(t){return this.abortedAgents.has(t)}consumeAborted(t){let e=this.abortedAgents.has(t);return this.abortedAgents.delete(t),e}async runTaskNow(t,e){await this.scheduler.enqueue({task:{...t,type:"immediate"},reason:"manual",promptOverride:e})}async runAgentNow(t,e){let s=t.heartbeatBody.trim()&&e==="Run now and summarize the current state."?t.heartbeatBody.trim():e,n=s===t.heartbeatBody.trim()&&t.heartbeatBody.trim().length>0,a={filePath:"",taskId:n?`heartbeat-${Date.now()}`:`manual-${Date.now()}`,agent:t.name,type:"immediate",priority:"medium",enabled:!0,created:new Date().toISOString(),runCount:0,catchUp:!1,tags:n?[...t.tags,"heartbeat"]:t.tags,body:s};await this.scheduler.enqueue({task:a,reason:n?"heartbeat":"manual",promptOverride:s})}async resolveApproval(t,e,s){t.filePath&&(await this.repository.setApprovalDecision(t.filePath,e,s),await this.refreshRunCaches(),this.emitStatusChange())}async pruneOldRuns(){let t=Date.now()-this.settings.runLogRetentionDays*24*60*60*1e3,e=await this.repository.listRecentRuns(500);for(let s of e)s.filePath&&new Date(s.started).getTime()<t&&await this.repository.trashFile(s.filePath)}async rebuildSchedules(){this.scheduler.pauseAll();for(let t of this.snapshot.tasks)this.scheduler.unregisterTask(t.taskId);this.scheduler.setMaxConcurrentRuns(this.settings.maxConcurrentRuns),await this.scheduler.loadTasks(this.snapshot.tasks.filter(t=>this.repository.getAgentByName(t.agent)?.enabled!==!1)),this.scheduler.resumeAll(),this.registerHeartbeats(),this.registerReflections()}shutdown(){for(let[,t]of this.heartbeatJobs)t.stop();this.heartbeatJobs.clear(),this.heartbeatsInFlight.clear();for(let[,t]of this.reflectionJobs)t.stop();this.reflectionJobs.clear(),this.reflectionsInFlight.clear(),this.scheduler.shutdown()}registerHeartbeats(){for(let[,t]of this.heartbeatJobs)t.stop();this.heartbeatJobs.clear(),this.heartbeatRegisteredAt=Date.now();for(let t of this.snapshot.agents)if(!(!t.enabled||!t.heartbeatEnabled||!t.heartbeatSchedule.trim()||!t.heartbeatBody.trim()))try{let e=new pe(t.heartbeatSchedule,{name:`heartbeat:${t.name}`,catch:!0,protect:!0,timezone:Intl.DateTimeFormat().resolvedOptions().timeZone},()=>{this.runHeartbeat(t.name)});this.heartbeatJobs.set(t.name,e)}catch(e){console.error(`Agent Fleet: failed to register heartbeat for "${t.name}":`,e)}}async runHeartbeat(t){if(!(Date.now()-this.heartbeatRegisteredAt<1e4)&&!this.heartbeatsInFlight.has(t)){this.heartbeatsInFlight.add(t);try{let e=this.repository.getAgentByName(t);if(!e||!e.enabled||!e.heartbeatBody.trim())return;let s={filePath:"",taskId:`heartbeat-${Date.now()}`,agent:e.name,type:"immediate",priority:"medium",enabled:!0,created:new Date().toISOString(),runCount:0,catchUp:!1,tags:[...e.tags,"heartbeat"],body:e.heartbeatBody.trim()};await this.scheduler.enqueue({task:s,reason:"heartbeat",promptOverride:e.heartbeatBody.trim()})}finally{this.heartbeatsInFlight.delete(t)}}}getNextHeartbeat(t){let e=this.heartbeatJobs.get(t);return e?e.nextRun()??null:null}registerReflections(){for(let[,t]of this.reflectionJobs)t.stop();this.reflectionJobs.clear();for(let t of this.snapshot.agents){if(!t.enabled||!t.memory||!t.reflection.enabled)continue;let e=t.reflection.schedule.trim();if(e)try{let s=new pe(e,{name:`reflection:${t.name}`,catch:!0,protect:!0,timezone:Intl.DateTimeFormat().resolvedOptions().timeZone},()=>{this.runReflection(t.name)});this.reflectionJobs.set(t.name,s)}catch(s){console.error(`Agent Fleet: failed to register reflection for "${t.name}":`,s)}}}getNextReflection(t){return this.reflectionJobs.get(t)?.nextRun()??null}async runReflectionNow(t){return this.runReflection(t)}async listPendingProposals(){return(await this.repository.listProposals()).filter(e=>e.status==="pending")}async acceptProposal(t){let e=await this.repository.readProposal(t);if(!e)return{ok:!1,message:"Proposal not found."};try{let s=await this.repository.applyProposal(e);return s?(await this.repository.setProposalStatus(t,"accepted"),await this.refreshFromVault(),{ok:!0,message:`Applied \u2192 ${s}`}):{ok:!1,message:"Couldn't apply \u2014 target skill not found; proposal left pending."}}catch(s){return{ok:!1,message:`Failed to apply: ${s instanceof Error?s.message:String(s)}`}}}async rejectProposal(t){await this.repository.setProposalStatus(t,"rejected"),this.emitStatusChange()}async withReflectionSlot(t){let e=Math.max(1,this.settings.maxConcurrentRuns);this.reflectionRunning>=e?await new Promise(s=>this.reflectionWaiters.push(s)):this.reflectionRunning++;try{return await t()}finally{let s=this.reflectionWaiters.shift();s?s():this.reflectionRunning--}}async runReflection(t){let e=this.repository.getAgentByName(t);if(!e||!e.enabled||!e.memory)return{ok:!1,message:"Agent not found, disabled, or memory off."};if(this.reflectionsInFlight.has(t))return{ok:!1,message:"A reflection is already in progress."};this.reflectionsInFlight.add(t);let s=new Date().toISOString(),n=s,a=`reflection-${Date.now()}`;this.runtimeState.set(t,{status:"running",currentTaskId:a,runStarted:s}),this.runOutputBuffers.set(t,""),this.emitStatusChange();try{await this.repository.migrateLegacyMemory(t);let r=await this.repository.readWorkingMemory(t),o=r?Ke(r.sections):"",c=await this.repository.readRecentRaw(t,2),l=pr({agentName:t,workingMemoryBody:o,recentRaw:c,tokenBudget:e.memoryTokenBudget}),h={filePath:"",taskId:a,agent:e.name,type:"immediate",priority:"low",enabled:!0,created:n,runCount:0,catchUp:!1,tags:[...e.tags,"reflection"],body:l,model:e.reflection.model||void 0},d=await this.withReflectionSlot(()=>this.executor.execute(e,h,l,v=>{let k=this.runOutputBuffers.get(t)??"";this.runOutputBuffers.set(t,k+v);let w=this.runOutputListeners.get(t);if(w)for(let S of w)S(v)},{suppressMemoryCapture:!0})),u=mr(d.outputText),p=await this.memoryWriter.reflect(e,u.sections,n);if(u.candidates.length>0){let v=await this.repository.readCandidates(t),k=fr(v,u.candidates,n);await this.repository.writeCandidates(t,k),await this.generateProposals(e,k,n)}let m=p?"Reflection complete \u2014 working memory consolidated.":"Reflection produced no memory block; working memory left unchanged.",f=d.exitCode===0||d.exitCode===null?"success":"failure",g={runId:d.runId,agent:e.name,task:a,status:f,started:s,completed:new Date().toISOString(),durationSeconds:d.durationSeconds,tokensUsed:d.tokensUsed,costUsd:d.costUsd,model:d.resolvedModel||e.reflection.model||e.model,modelSource:d.modelSource,concreteModel:d.concreteModel,exitCode:d.exitCode,tags:Array.from(new Set([...e.tags,"reflection"])),prompt:d.prompt,output:d.outputText,toolsUsed:d.toolsUsed.map(v=>`${v.tool}${v.command?`: ${v.command}`:""}`),finalResult:m,stderr:d.stderr},y=await this.repository.writeRunLog(g);return await this.refreshRunCaches(),this.runtimeState.set(t,{status:f==="success"?"idle":"error",currentRunId:d.runId,lastRun:{...g,filePath:y}}),{ok:p,message:m}}catch(r){let o=r instanceof Error?r.message:String(r);console.warn(`Agent Fleet: reflection failed for "${t}":`,r);let c={runId:(0,Sa.randomUUID)(),agent:t,task:a,status:"failure",started:s,completed:new Date().toISOString(),durationSeconds:Math.round((Date.now()-new Date(s).getTime())/1e3),model:e.reflection.model||e.model,exitCode:1,tags:Array.from(new Set([...e.tags,"reflection"])),prompt:"",output:`Reflection failed: ${o}`,toolsUsed:[]},l=c;try{let h=await this.repository.writeRunLog(c);await this.refreshRunCaches(),l={...c,filePath:h}}catch(h){console.warn(`Agent Fleet: failed to write reflection run log for "${t}"`,h)}return this.runtimeState.set(t,{status:"error",lastRun:l}),{ok:!1,message:`Reflection failed: ${o}`}}finally{this.reflectionsInFlight.delete(t),this.runOutputBuffers.delete(t),this.runOutputListeners.delete(t),this.emitStatusChange()}}async generateProposals(t,e,s){if(!t.reflection.proposeSkills)return;let n=t.reflection.recurrenceThreshold||3;for(let a of e){if(a.proposed||a.occurrences<n)continue;let o={id:`prop-${s.slice(0,10)}-${a.key}`.slice(0,80),type:"skill_create",agent:t.name,status:"pending",created:s,targetSkill:a.suggestedSkill||a.key,candidate:a.key,evidence:a.evidence,rationale:a.pattern,body:`This skill was proposed by reflection because the following friction recurred ${a.occurrences} times:
11902
+ `)}catch{window.clearTimeout(p),u()}})}async probeHttpServer(t){let e=await this.findServerToken(t);if(!e)return[];let s=t.url.endsWith("/sse")?t.url.replace(/\/sse$/,"/mcp"):t.url;try{let a=(await this.httpRequest(s,e,{jsonrpc:"2.0",id:1,method:"initialize",params:{protocolVersion:"2025-03-26",capabilities:{},clientInfo:{name:"agent-fleet",version:"1.0.0"}}}))?._sessionId;await this.httpRequest(s,e,{jsonrpc:"2.0",method:"notifications/initialized"},a);let i=await this.httpRequest(s,e,{jsonrpc:"2.0",id:2,method:"tools/list"},a),o=[],h=i?.result?.tools??[];for(let d of h)o.push({name:d.name,description:d.description,inputSchema:d.inputSchema});return o}catch(n){return console.warn(`McpManager: HTTP probe failed for ${t.name}:`,n),[]}}async findServerToken(t){if(this.authManager){let a=this.authManager.getToken(t.name);if(a)return a}let e=t.name.replace(/^claude\.ai\s+/i,"").replace(/\s+/g,"_").toUpperCase(),s=[`${e}_API_KEY`,`${e}_API_KEY_MILO`,`${e}_TOKEN`];for(let a of s){let i=process.env[a];if(i)return i}let n=[xa.join(ea(),".openclaw","workspace",".env"),xa.join(ea(),".env")];for(let a of n)try{let i=vi.readFileSync(a,"utf8");for(let o of ye(i)){let c=o.trim().match(/^(?:export\s+)?([A-Za-z_]\w*)=(.*)$/);if(c){let l=c[1],h=c[2].replace(/^["']|["']$/g,"");if(s.includes(l))return h}}}catch{}}httpRequest(t,e,s,n){return new Promise((a,i)=>{let o=JSON.stringify(s),c=new URL(t),l=c.protocol==="https:",h={"Content-Type":"application/json",Accept:"application/json, text/event-stream",Authorization:`Bearer ${e}`,"Content-Length":String(Buffer.byteLength(o))};n&&(h["mcp-session-id"]=n);let d={hostname:c.hostname,port:c.port||(l?443:80),path:c.pathname+c.search,method:"POST",headers:h},p=(l?yi:un).request(d,f=>{let g="";f.on("data",y=>{g+=y.toString()}),f.on("end",()=>{let y=f.headers["mcp-session-id"];if((f.headers["content-type"]??"").includes("text/event-stream")){for(let k of ye(g))if(k.startsWith("data: "))try{let w=JSON.parse(k.slice(6));y&&(w._sessionId=y),a(w);return}catch{}a(null)}else try{let k=JSON.parse(g);y&&(k._sessionId=y),a(k)}catch{a(null)}})});p.on("error",i);let m=window.setTimeout(()=>{p.destroy(),a(null)},15e3);p.on("close",()=>window.clearTimeout(m)),p.write(o),p.end()})}};var hc={"every 5m":"*/5 * * * *","every 10m":"*/10 * * * *","every 15m":"*/15 * * * *","every 30m":"*/30 * * * *","every 1h":"0 * * * *","every 2h":"0 */2 * * *",hourly:"0 * * * *","daily at 9am":"0 9 * * *","daily at 6pm":"0 18 * * *","weekdays at 9am":"0 9 * * 1-5","weekly on monday":"0 9 * * 1","monthly on 1st":"0 9 1 * *"},mn=class{constructor(t,e){this.maxConcurrentRuns=t;this.callbacks=e}jobs=new Map;activeRuns=0;queue=[];paused=!1;setMaxConcurrentRuns(t){this.maxConcurrentRuns=t}parseSchedule(t){return hc[t.toLowerCase()]??t}toLocalISOString(t){let e=s=>String(s).padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}T${e(t.getHours())}:${e(t.getMinutes())}:${e(t.getSeconds())}`}async registerTask(t){if(this.unregisterTask(t.taskId),!(!t.enabled||t.type==="immediate"))try{if(t.type==="once"&&t.runAt){let e=new pe(new Date(t.runAt),{name:t.taskId,catch:!0},()=>{this.enqueue({task:t,reason:"scheduled"})});this.jobs.set(t.taskId,e),await this.callbacks.onTaskScheduled(t,t.runAt);return}if(t.type==="recurring"&&t.schedule){let e=this.parseSchedule(t.schedule),s=new pe(e,{name:t.taskId,catch:!0,protect:!0,timezone:Intl.DateTimeFormat().resolvedOptions().timeZone},()=>{this.enqueue({task:t,reason:"scheduled"})});this.jobs.set(t.taskId,s);let n=s.nextRun();await this.callbacks.onTaskScheduled(t,n?this.toLocalISOString(n):void 0)}}catch(e){console.error(`Agent Fleet: Failed to register task "${t.taskId}":`,e)}}unregisterTask(t){let e=this.jobs.get(t);e&&(e.stop(),this.jobs.delete(t))}async loadTasks(t){for(let e of t)await this.registerTask(e)}async handleStartupCatchUp(t){let e=Date.now();for(let s of t)!s.enabled||!s.catchUp||!s.nextRun||new Date(s.nextRun).getTime()<e&&await this.enqueue({task:s,reason:"catch-up"})}async enqueue(t){this.queue.push(t),await this.processQueue()}pauseAll(){this.paused=!0,this.jobs.forEach(t=>t.pause())}resumeAll(){this.paused=!1,this.jobs.forEach(t=>t.resume()),this.processQueue()}shutdown(){this.paused=!0,this.jobs.forEach(t=>t.stop()),this.jobs.clear(),this.queue.length=0}getQueueSize(){return this.queue.length}async processQueue(){if(!this.paused)for(;this.activeRuns<this.maxConcurrentRuns&&this.queue.length>0;){let t=this.queue.shift();if(!t)return;this.activeRuns+=1,this.callbacks.onTaskTriggered(t).finally(async()=>{this.activeRuns-=1,await this.processQueue()})}}};var ks=class r{constructor(t,e,s){this.repository=t;this.settings=e;this.executor=new hn(e,t,s),this.mcpManager=new pn(e),s&&this.mcpManager.setAuthManager(s),this.memoryWriter=new Gt(t),this.scheduler=new mn(e.maxConcurrentRuns,{onTaskTriggered:n=>this.runPendingTask(n),onTaskScheduled:(n,a)=>this.repository.updateTaskRunMetadata(n,{nextRun:a})})}scheduler;executor;mcpManager;snapshot={agents:[],skills:[],tasks:[],channels:[],mcpServers:[],validationIssues:[]};runtimeState=new Map;recentRuns=[];chartRuns=[];static CHART_WINDOW_DAYS=14;recentUsage=[];statusChangeListeners=new Set;runOutputListeners=new Map;runOutputBuffers=new Map;heartbeatJobs=new Map;reflectionJobs=new Map;reflectionsInFlight=new Set;reflectionRunning=0;reflectionWaiters=[];heartbeatRegisteredAt=0;heartbeatsInFlight=new Set;channelResultHandler;memoryWriter;async initialize(){this.snapshot=await this.repository.loadAll(),await this.repository.migrateAllLegacyMemory(),await this.refreshRunCaches();let t=this.snapshot.tasks.filter(e=>this.repository.getAgentByName(e.agent)?.enabled!==!1);await this.scheduler.loadTasks(t),await this.scheduler.handleStartupCatchUp(t),this.registerHeartbeats(),this.registerReflections(),this.emitStatusChange()}onChannelResult(t){this.channelResultHandler=t}async refreshFromVault(){this.snapshot=await this.repository.loadAll(),await this.rebuildSchedules(),await this.refreshRunCaches(),this.emitStatusChange()}getSnapshot(){return this.snapshot}getRecentRuns(){return this.recentRuns}getChartRuns(){return this.chartRuns}async refreshRunCaches(){this.recentRuns=await this.repository.listRecentRuns();let t=new Date;t.setDate(t.getDate()-(r.CHART_WINDOW_DAYS-1)),this.chartRuns=await this.repository.listRunsSince(t),this.recentUsage=await this.repository.readUsageSince(t)}getUsageRecords(){return this.recentUsage}recordUsage(t){this.recentUsage.push(t),this.repository.appendUsage(t).catch(e=>{console.warn("Agent Fleet: failed to append usage record",e)}),this.emitStatusChange()}getAgentState(t){let e=this.runtimeState.get(t),s=this.snapshot.agents.find(n=>n.name===t);return s&&!s.enabled?{status:"disabled",lastRun:e?.lastRun,currentRunId:e?.currentRunId}:e??{status:"idle"}}getFleetStatus(){let t=new Date,e=o=>String(o).padStart(2,"0"),s=`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}`,n=this.recentRuns.filter(o=>{let c=new Date(o.started);return`${c.getFullYear()}-${e(c.getMonth()+1)}-${e(c.getDate())}`===s}).length,a=Array.from(this.runtimeState.values()).filter(o=>o.status==="running").length,i=this.recentRuns.flatMap(o=>o.approvals??[]).filter(o=>o.status==="pending").length;return{running:a,pending:i,completedToday:n}}subscribe(t){return this.statusChangeListeners.add(t),()=>this.statusChangeListeners.delete(t)}onRunOutput(t,e){let s=this.runOutputListeners.get(t);s||(s=new Set,this.runOutputListeners.set(t,s)),s.add(e);let n=this.runOutputBuffers.get(t);return n&&e(n),()=>{this.runOutputListeners.get(t)?.delete(e)}}getRunOutputBuffer(t){return this.runOutputBuffers.get(t)??""}async handleVaultChange(t){await this.repository.loadFile(t),this.snapshot=this.repository.getSnapshot(),await this.rebuildSchedules(),await this.refreshRunCaches(),this.emitStatusChange()}async handleVaultDelete(t){this.repository.removeFile(t),this.snapshot=this.repository.getSnapshot(),await this.rebuildSchedules(),await this.refreshRunCaches(),this.emitStatusChange()}abortedAgents=new Set;abortAgentRun(t){let e=this.executor.abortAgent(t);return e&&(this.abortedAgents.add(t),this.runtimeState.set(t,{status:"idle"}),this.emitStatusChange()),e}wasAborted(t){return this.abortedAgents.has(t)}consumeAborted(t){let e=this.abortedAgents.has(t);return this.abortedAgents.delete(t),e}async runTaskNow(t,e){await this.scheduler.enqueue({task:{...t,type:"immediate"},reason:"manual",promptOverride:e})}async runAgentNow(t,e){let s=t.heartbeatBody.trim()&&e==="Run now and summarize the current state."?t.heartbeatBody.trim():e,n=s===t.heartbeatBody.trim()&&t.heartbeatBody.trim().length>0,a={filePath:"",taskId:n?`heartbeat-${Date.now()}`:`manual-${Date.now()}`,agent:t.name,type:"immediate",priority:"medium",enabled:!0,created:new Date().toISOString(),runCount:0,catchUp:!1,tags:n?[...t.tags,"heartbeat"]:t.tags,body:s};await this.scheduler.enqueue({task:a,reason:n?"heartbeat":"manual",promptOverride:s})}async resolveApproval(t,e,s){t.filePath&&(await this.repository.setApprovalDecision(t.filePath,e,s),await this.refreshRunCaches(),this.emitStatusChange())}async pruneOldRuns(){let t=Date.now()-this.settings.runLogRetentionDays*24*60*60*1e3,e=await this.repository.listRecentRuns(500);for(let s of e)s.filePath&&new Date(s.started).getTime()<t&&await this.repository.trashFile(s.filePath)}async rebuildSchedules(){this.scheduler.pauseAll();for(let t of this.snapshot.tasks)this.scheduler.unregisterTask(t.taskId);this.scheduler.setMaxConcurrentRuns(this.settings.maxConcurrentRuns),await this.scheduler.loadTasks(this.snapshot.tasks.filter(t=>this.repository.getAgentByName(t.agent)?.enabled!==!1)),this.scheduler.resumeAll(),this.registerHeartbeats(),this.registerReflections()}shutdown(){for(let[,t]of this.heartbeatJobs)t.stop();this.heartbeatJobs.clear(),this.heartbeatsInFlight.clear();for(let[,t]of this.reflectionJobs)t.stop();this.reflectionJobs.clear(),this.reflectionsInFlight.clear(),this.scheduler.shutdown()}registerHeartbeats(){for(let[,t]of this.heartbeatJobs)t.stop();this.heartbeatJobs.clear(),this.heartbeatRegisteredAt=Date.now();for(let t of this.snapshot.agents)if(!(!t.enabled||!t.heartbeatEnabled||!t.heartbeatSchedule.trim()||!t.heartbeatBody.trim()))try{let e=new pe(t.heartbeatSchedule,{name:`heartbeat:${t.name}`,catch:!0,protect:!0,timezone:Intl.DateTimeFormat().resolvedOptions().timeZone},()=>{this.runHeartbeat(t.name)});this.heartbeatJobs.set(t.name,e)}catch(e){console.error(`Agent Fleet: failed to register heartbeat for "${t.name}":`,e)}}async runHeartbeat(t){if(!(Date.now()-this.heartbeatRegisteredAt<1e4)&&!this.heartbeatsInFlight.has(t)){this.heartbeatsInFlight.add(t);try{let e=this.repository.getAgentByName(t);if(!e||!e.enabled||!e.heartbeatBody.trim())return;let s={filePath:"",taskId:`heartbeat-${Date.now()}`,agent:e.name,type:"immediate",priority:"medium",enabled:!0,created:new Date().toISOString(),runCount:0,catchUp:!1,tags:[...e.tags,"heartbeat"],body:e.heartbeatBody.trim()};await this.scheduler.enqueue({task:s,reason:"heartbeat",promptOverride:e.heartbeatBody.trim()})}finally{this.heartbeatsInFlight.delete(t)}}}getNextHeartbeat(t){let e=this.heartbeatJobs.get(t);return e?e.nextRun()??null:null}registerReflections(){for(let[,t]of this.reflectionJobs)t.stop();this.reflectionJobs.clear();for(let t of this.snapshot.agents){if(!t.enabled||!t.memory||!t.reflection.enabled)continue;let e=t.reflection.schedule.trim();if(e)try{let s=new pe(e,{name:`reflection:${t.name}`,catch:!0,protect:!0,timezone:Intl.DateTimeFormat().resolvedOptions().timeZone},()=>{this.runReflection(t.name)});this.reflectionJobs.set(t.name,s)}catch(s){console.error(`Agent Fleet: failed to register reflection for "${t.name}":`,s)}}}getNextReflection(t){return this.reflectionJobs.get(t)?.nextRun()??null}async runReflectionNow(t){return this.runReflection(t)}async listPendingProposals(){return(await this.repository.listProposals()).filter(e=>e.status==="pending")}async acceptProposal(t){let e=await this.repository.readProposal(t);if(!e)return{ok:!1,message:"Proposal not found."};try{let s=await this.repository.applyProposal(e);return s?(await this.repository.setProposalStatus(t,"accepted"),await this.refreshFromVault(),{ok:!0,message:`Applied \u2192 ${s}`}):{ok:!1,message:"Couldn't apply \u2014 target skill not found; proposal left pending."}}catch(s){return{ok:!1,message:`Failed to apply: ${s instanceof Error?s.message:String(s)}`}}}async rejectProposal(t){await this.repository.setProposalStatus(t,"rejected"),this.emitStatusChange()}async withReflectionSlot(t){let e=Math.max(1,this.settings.maxConcurrentRuns);this.reflectionRunning>=e?await new Promise(s=>this.reflectionWaiters.push(s)):this.reflectionRunning++;try{return await t()}finally{let s=this.reflectionWaiters.shift();s?s():this.reflectionRunning--}}async runReflection(t){let e=this.repository.getAgentByName(t);if(!e||!e.enabled||!e.memory)return{ok:!1,message:"Agent not found, disabled, or memory off."};if(this.reflectionsInFlight.has(t))return{ok:!1,message:"A reflection is already in progress."};this.reflectionsInFlight.add(t);let s=new Date().toISOString(),n=s,a=`reflection-${Date.now()}`;this.runtimeState.set(t,{status:"running",currentTaskId:a,runStarted:s}),this.runOutputBuffers.set(t,""),this.emitStatusChange();try{await this.repository.migrateLegacyMemory(t);let i=await this.repository.readWorkingMemory(t),o=i?Ke(i.sections):"",c=await this.repository.readRecentRaw(t,2),l=mi({agentName:t,workingMemoryBody:o,recentRaw:c,tokenBudget:e.memoryTokenBudget}),h={filePath:"",taskId:a,agent:e.name,type:"immediate",priority:"low",enabled:!0,created:n,runCount:0,catchUp:!1,tags:[...e.tags,"reflection"],body:l,model:e.reflection.model||void 0},d=await this.withReflectionSlot(()=>this.executor.execute(e,h,l,v=>{let k=this.runOutputBuffers.get(t)??"";this.runOutputBuffers.set(t,k+v);let w=this.runOutputListeners.get(t);if(w)for(let S of w)S(v)},{suppressMemoryCapture:!0})),u=fi(d.outputText),p=await this.memoryWriter.reflect(e,u.sections,n);if(u.candidates.length>0){let v=await this.repository.readCandidates(t),k=gi(v,u.candidates,n);await this.repository.writeCandidates(t,k),await this.generateProposals(e,k,n)}let m=p?"Reflection complete \u2014 working memory consolidated.":"Reflection produced no memory block; working memory left unchanged.",f=d.exitCode===0||d.exitCode===null?"success":"failure",g={runId:d.runId,agent:e.name,task:a,status:f,started:s,completed:new Date().toISOString(),durationSeconds:d.durationSeconds,tokensUsed:d.tokensUsed,costUsd:d.costUsd,model:d.resolvedModel||e.reflection.model||e.model,modelSource:d.modelSource,concreteModel:d.concreteModel,exitCode:d.exitCode,tags:Array.from(new Set([...e.tags,"reflection"])),prompt:d.prompt,output:d.outputText,toolsUsed:d.toolsUsed.map(v=>`${v.tool}${v.command?`: ${v.command}`:""}`),finalResult:m,stderr:d.stderr},y=await this.repository.writeRunLog(g);return await this.refreshRunCaches(),this.runtimeState.set(t,{status:f==="success"?"idle":"error",currentRunId:d.runId,lastRun:{...g,filePath:y}}),{ok:p,message:m}}catch(i){let o=i instanceof Error?i.message:String(i);console.warn(`Agent Fleet: reflection failed for "${t}":`,i);let c={runId:(0,Sa.randomUUID)(),agent:t,task:a,status:"failure",started:s,completed:new Date().toISOString(),durationSeconds:Math.round((Date.now()-new Date(s).getTime())/1e3),model:e.reflection.model||e.model,exitCode:1,tags:Array.from(new Set([...e.tags,"reflection"])),prompt:"",output:`Reflection failed: ${o}`,toolsUsed:[]},l=c;try{let h=await this.repository.writeRunLog(c);await this.refreshRunCaches(),l={...c,filePath:h}}catch(h){console.warn(`Agent Fleet: failed to write reflection run log for "${t}"`,h)}return this.runtimeState.set(t,{status:"error",lastRun:l}),{ok:!1,message:`Reflection failed: ${o}`}}finally{this.reflectionsInFlight.delete(t),this.runOutputBuffers.delete(t),this.runOutputListeners.delete(t),this.emitStatusChange()}}async generateProposals(t,e,s){if(!t.reflection.proposeSkills)return;let n=t.reflection.recurrenceThreshold||3;for(let a of e){if(a.proposed||a.occurrences<n)continue;let o={id:`prop-${s.slice(0,10)}-${a.key}`.slice(0,80),type:"skill_create",agent:t.name,status:"pending",created:s,targetSkill:a.suggestedSkill||a.key,candidate:a.key,evidence:a.evidence,rationale:a.pattern,body:`This skill was proposed by reflection because the following friction recurred ${a.occurrences} times:
11992
11903
 
11993
11904
  > ${a.pattern}
11994
11905
 
11995
- Document the reliable steps to handle this so future runs don't rediscover them.`};try{await this.repository.writeProposal(o),a.proposed=!0,await this.repository.writeCandidates(t.name,e)}catch(c){console.warn(`Agent Fleet: failed to write proposal for "${t.name}"`,c)}}}async runPendingTask({task:t,promptOverride:e}){let s=this.repository.getAgentByName(t.agent);if(!s||!s.enabled)return;let n=new Date().toISOString();this.runtimeState.set(s.name,{status:"running",currentTaskId:t.taskId,runStarted:n}),this.runOutputBuffers.set(s.name,""),this.emitStatusChange();try{let a=await this.executor.execute(s,t,e,m=>{let f=this.runOutputBuffers.get(s.name)??"";this.runOutputBuffers.set(s.name,f+m);let g=this.runOutputListeners.get(s.name);if(g)for(let y of g)y(m)}),r=this.consumeAborted(s.name),o=r?[]:this.buildApprovals(s,a.toolsUsed),c=r?"cancelled":this.resolveRunStatus(a,o),l={runId:a.runId,agent:s.name,task:t.taskId,status:c,started:n,completed:new Date().toISOString(),durationSeconds:a.durationSeconds,tokensUsed:a.tokensUsed,costUsd:a.costUsd,model:a.resolvedModel||s.model,modelSource:a.modelSource,concreteModel:a.concreteModel,exitCode:a.exitCode,tags:Array.from(new Set([...s.tags,...t.tags])),prompt:a.prompt,output:a.outputText,toolsUsed:a.toolsUsed.map(m=>`${m.tool}${m.command?`: ${m.command}`:""}`),finalResult:a.finalResult,stderr:a.stderr,approvals:o},h=await this.repository.writeRunLog(l);if(await this.repository.updateTaskRunMetadata(t,{lastRun:n,runCount:t.runCount+1}),s.memory){let f=t.tags.includes("heartbeat")?"heartbeat":`task:${t.taskId}`,g=qs(a.outputText);try{await this.memoryWriter.capture(s,g,f,new Date().toISOString())}catch(y){console.warn(`Agent Fleet: failed to append memory for "${s.name}"`,y)}}let d=t.tags.includes("heartbeat"),u=d?s.heartbeatChannel:t.channel??"",p=d?s.heartbeatChannelTarget??"":t.channelTarget??"";if(u&&!r&&l.output.trim())try{this.channelResultHandler?.(s.name,u,l.output,d?"heartbeat":t.taskId,p)}catch(m){console.warn(`Agent Fleet: channel delivery failed for ${s.name}`,m)}r&&(l.output="Task was manually stopped."),await this.refreshRunCaches(),this.runtimeState.set(s.name,{status:r||c==="success"?"idle":c==="pending_approval"?"pending":"error",currentRunId:a.runId,lastRun:{...l,filePath:h}}),r||this.notify(l)}catch(a){let r=this.consumeAborted(s.name),o=r?"cancelled":"failure",c=Dt(t,s,this.settings),l={runId:(0,Sa.randomUUID)(),agent:s.name,task:t.taskId,status:o,started:n,completed:new Date().toISOString(),durationSeconds:Math.round((Date.now()-new Date(n).getTime())/1e3),model:c.value||s.model,modelSource:c.source,exitCode:r?-1:1,tags:Array.from(new Set([...s.tags,...t.tags])),prompt:e??t.body,output:r?"Task was manually stopped.":a instanceof Error?a.message:String(a),toolsUsed:[]},h=await this.repository.writeRunLog(l);await this.refreshRunCaches(),this.runtimeState.set(s.name,{status:r?"idle":"error",lastRun:{...l,filePath:h}}),r||this.notify(l)}finally{if(s.memory)try{await this.memoryWriter.drainPending(s,new Date().toISOString())}catch(a){console.warn(`Agent Fleet: failed to drain pending memory for "${s.name}"`,a)}this.runOutputBuffers.delete(s.name),this.runOutputListeners.delete(s.name),this.emitStatusChange()}}buildApprovals(t,e){let s=e.filter(n=>t.approvalRequired.includes(n.tool)).map(n=>({tool:n.tool,command:n.command,reason:n.reason,status:"pending"}));return s.length>0?s:void 0}resolveRunStatus(t,e){return e?.length?"pending_approval":t.timedOut?"timeout":t.exitCode===0?"success":"failure"}notify(t){if(this.settings.notificationLevel==="none"||this.settings.notificationLevel==="failures-only"&&t.status==="success")return;let s=(ye(t.output).map(a=>a.trim()).find(a=>a&&!a.startsWith("{")&&!a.startsWith("["))??"").slice(0,120)||t.status,n=t.status==="success"?`\u2705 ${t.agent}: ${s}`:t.status==="pending_approval"?`\u{1F535} ${t.agent} needs approval: ${(t.approvals??[])[0]?.tool??"tool action"}`:`\u274C ${t.agent}: ${s}`;new vr.Notice(n,t.status==="success"?5e3:0)}emitStatusChange(){for(let t of this.statusChangeListeners)t()}};function hc(i){return i.toLowerCase().replace(/[^a-z0-9-]/g,"-").replace(/-{2,}/g,"-").replace(/^-|-$/g,"")}function xs(i,t){return`${i}-${hc(t)}`}var Gt="af-channel-cred",Ss="af-mcp-secret",fn=class{constructor(t){this.storage=t;t||console.warn("Agent Fleet: SecretStorage unavailable (Obsidian < 1.11.4). Secrets will use plaintext fallback.")}get available(){return!!this.storage}setJson(t,e,s){if(!this.storage)return;let n=xs(t,e);this.storage.setSecret(n,JSON.stringify(s))}getJson(t,e){if(!this.storage)return null;let s=xs(t,e),n=this.storage.getSecret(s);if(!n)return null;try{return JSON.parse(n)}catch{return null}}setString(t,e,s){if(!this.storage)return;let n=xs(t,e);this.storage.setSecret(n,s)}getString(t,e){if(!this.storage)return null;let s=xs(t,e);return this.storage.getSecret(s)||null}delete(t,e){if(!this.storage)return;let s=xs(t,e);this.storage.setSecret(s,"")}listByPrefix(t){return this.storage?this.storage.listSecrets().filter(e=>e.startsWith(t+"-")):[]}};var gn=class{tokens=new Map;oauthTokens=new Map;secretStore;setSecretStore(t){this.secretStore=t}storeProbeToken(t,e){this.tokens.set(t,e)}storeStaticToken(t,e){this.tokens.set(t,e),this.secretStore?.setJson(Ss,t,{accessToken:e})}storeOAuthToken(t,e){this.oauthTokens.set(t,e),this.tokens.set(t,e.accessToken),this.secretStore?.setJson(Ss,t,{accessToken:e.accessToken,refreshToken:e.refreshToken,expiresAt:e.expiresAt,tokenEndpoint:e.tokenEndpoint,clientId:e.clientId,resource:e.resource})}hydrate(t){if(!this.secretStore)return null;let e=this.secretStore.getJson(Ss,t);return e?.accessToken?(this.tokens.set(t,e.accessToken),e.tokenEndpoint&&e.clientId&&e.resource&&this.oauthTokens.set(t,{accessToken:e.accessToken,refreshToken:e.refreshToken,expiresAt:e.expiresAt,tokenEndpoint:e.tokenEndpoint,clientId:e.clientId,resource:e.resource}),e):null}getToken(t){return this.tokens.get(t)??this.hydrate(t)?.accessToken??void 0}getOAuthToken(t){return this.oauthTokens.has(t)?this.oauthTokens.get(t):(this.hydrate(t),this.oauthTokens.get(t))}hasToken(t){return this.tokens.has(t)||!!this.hydrate(t)}removeToken(t){this.tokens.delete(t),this.oauthTokens.delete(t),this.secretStore?.delete(Ss,t)}getExpiringTokens(t=5*6e4){let e=new Map,s=Date.now();for(let[n,a]of this.oauthTokens)a.refreshToken&&a.expiresAt&&a.expiresAt-s<t&&e.set(n,a);return e}};var ut=require("obsidian");var yn=require("crypto"),ot=require("obsidian");function Cs(){return(0,yn.randomUUID)()}function wr(i){let t=`${i.timestamp}|${i.content.slice(0,80)}`;return(0,yn.createHash)("sha1").update(t).digest("hex").slice(0,16)}var Vt=class i{constructor(t,e,s,n,a){this.agent=t;this.settings=e;this.repository=s;this.vault=n,this.channelName=a?.channelName,this.conversationId=a?.conversationId,this.channelContext=a?.channelContext,this.inAppConversationId=a?.inAppConversationId,this.threadAnchorId=a?.threadAnchorId,this.parentSession=a?.parentSession,this.mcpAuth=a?.mcpAuth,this.memoryWriter=new zt(s)}messages=[];isStreaming=!1;isProcessAlive=!1;settingsState=null;mcpProjection=null;get pendingTurnCount(){return this.pendingTurns}lastActiveAt=Date.now();process=null;claudeSessionId=null;claudeResumeAttempted=!1;vault;stdoutBuffer="";processListeners=null;basePromptSent=!1;codexTurnState=null;codexQueue=[];codexResumeAttempted=!1;codexTurnErrors=[];codexStderr="";codexPermState=null;get isCodex(){return kt(this.agent.adapter)==="codex"}channelName;conversationId;channelContext;inAppConversationId;conversationName="";threadAnchorId;parentSession;threadAnchorIndex;threads=new Map;threadIndex={};activeOnEvent=null;turnResponseText="";displayedLen=0;turnToolCalls=[];pendingTurns=0;turnResolve=null;turnReject=null;needsCompactBeforeNextTurn=!1;lastCompactTriggerAt=0;static WATCHDOG_FALLBACK_MINUTES=10;watchdogTimer=null;getWatchdogMs(){let t=this.settings.chatWatchdogMinutes;return(typeof t=="number"&&t>0?t:i.WATCHDOG_FALLBACK_MINUTES)*60*1e3}armWatchdog(){this.clearWatchdog();let t=this.getWatchdogMs();this.watchdogTimer=window.setTimeout(()=>{if(this.watchdogTimer=null,!this.isStreaming)return;this.activeOnEvent?.({type:"error",content:"",errorMessage:`no response from the CLI for ${Math.round(t/6e4)} minutes \u2014 giving up`});let e=new Error("Watchdog timeout");this.handleProcessError(e);try{this.process?.kill()}catch{}},t)}clearWatchdog(){this.watchdogTimer&&(window.clearTimeout(this.watchdogTimer),this.watchdogTimer=null)}currentToolName;activityListeners=new Set;onActivityChange(t){return this.activityListeners.add(t),t(),()=>{this.activityListeners.delete(t)}}emitActivity(){for(let t of this.activityListeners)t()}setStreaming(t){this.isStreaming!==t&&(this.isStreaming=t,t||(this.currentToolName=void 0),this.emitActivity())}setCurrentTool(t){this.currentToolName!==t&&(this.currentToolName=t,this.emitActivity())}stats={costTotalUsd:0,turnCount:0};usageRecorder;setUsageRecorder(t){this.usageRecorder=t}statsListeners=new Set;onStatsChange(t){return this.statsListeners.add(t),t({...this.stats}),()=>{this.statsListeners.delete(t)}}getStats(){return{...this.stats}}emitStats(){let t={...this.stats};for(let e of this.statsListeners)e(t)}mcpAuth;buildMcpProjection(t){let e=this.agent.memory?this.repository.getPendingDirAbsolutePath(this.agent.name):null,s=cn({registry:this.repository.getMcpServers(),agentGrants:this.agent.mcpServers??[],getBearerToken:n=>this.mcpAuth?.getToken(n),remember:e?{pendingDir:e,source:`mcp:${this.captureSource()}`}:null});return this.mcpProjection=dn(t,this.agent.adapter,s),{args:this.mcpProjection?.args??[],env:this.mcpProjection?.env??{},allowServers:s.map(n=>n.def.name)}}memoryWriter;captureSource(){return`chat:${this.inAppConversationId||this.conversationId||"default"}`}getConversationName(){return this.conversationName}async setConversationName(t){this.conversationName=t,await this.persist()}refreshAgent(){let t=this.repository.getAgentByName(this.agent.name);t&&(this.agent=t)}get isThread(){return!!this.threadAnchorId}async loadPersistedState(){let t=this.getChatFilePath(),e=this.vault.getAbstractFileByPath(t);if(!(e instanceof ot.TFile))return!1;try{let s=await this.vault.cachedRead(e);if(this.isThread){let a=JSON.parse(s);return a.messages?.length>0||a.sessionId?(this.messages=(a.messages??[]).map(r=>r.id?r:{...r,id:wr(r)}),this.claudeSessionId=a.sessionId??null,this.threadAnchorIndex=a.anchorIndex,this.claudeSessionId&&(this.basePromptSent=!0),!0):!1}let n=JSON.parse(s);if(n.name&&(this.conversationName=n.name),n.messages?.length>0)return this.messages=n.messages.map(a=>a.id?a:{...a,id:wr(a)}),this.claudeSessionId=n.sessionId??null,this.threadIndex=n.threads??{},this.claudeSessionId&&(this.basePromptSent=!0),!0}catch{}return!1}getThreadIndex(){return{...this.threadIndex}}async persist(){let t=new Date().toISOString(),e=this.getChatFilePath(),s;if(this.isThread){let a={anchorMessageId:this.threadAnchorId,anchorIndex:this.threadAnchorIndex??0,sessionId:this.claudeSessionId,messages:this.messages,createdAt:this.parentSession?.threadIndex[this.threadAnchorId]?.createdAt??t,lastActive:t};s=JSON.stringify(a,null,2)}else{let a={sessionId:this.claudeSessionId,messages:this.messages,lastActive:t,name:this.conversationName||void 0,threads:Object.keys(this.threadIndex).length>0?this.threadIndex:void 0};s=JSON.stringify(a,null,2)}let n=this.vault.getAbstractFileByPath(e);n instanceof ot.TFile?await this.vault.modify(n,s):(await this.ensureParentFolders(e),await this.vault.create(e,s)),this.isThread&&this.parentSession&&this.threadAnchorId&&await this.parentSession.upsertThreadIndex(this.threadAnchorId,{path:e,createdAt:this.parentSession.threadIndex[this.threadAnchorId]?.createdAt??t,messageCount:this.messages.length,lastActive:t})}async upsertThreadIndex(t,e){this.threadIndex[t]=e,await this.persist()}async ensureParentFolders(t){let e=t.lastIndexOf("/");if(e<=0)return;let n=t.slice(0,e).split("/"),a="";for(let r of n)if(a=a?`${a}/${r}`:r,!this.vault.getAbstractFileByPath(a))try{await this.vault.createFolder(a)}catch(o){if(!(o instanceof Error?o.message:String(o)).includes("already exists"))throw o}}async clearPersistedState(){let t=this.getChatFilePath();await this.repository.trashFile(t),this.messages=[],this.claudeSessionId=null,this.basePromptSent=!1}getChatFilePath(){if(this.threadAnchorId&&this.parentSession)return this.parentSession.getThreadFilePath(this.threadAnchorId);if(this.channelName&&this.conversationId){let s=this.settings.fleetFolder,n=oe(this.conversationId)||"conversation";return(0,ot.normalizePath)(`${s}/channels/${this.channelName}/sessions/${n}.json`)}if(!this.inAppConversationId)throw new Error("ChatSession requires inAppConversationId (or channel options) to resolve a file path");let t=oe(this.inAppConversationId)||"conversation";if(this.agent.isFolder){let s=this.agent.filePath.replace(/\/agent\.md$/,"");return(0,ot.normalizePath)(`${s}/conversations/${t}.json`)}let e=this.repository.getMemoryPath(this.agent.name).replace(/\/[^/]+$/,"");return(0,ot.normalizePath)(`${e}/${this.agent.name}-conversations/${t}.json`)}getThreadFilePath(t){let e=this.getParentChatFilePath(),s=e.replace(/\/[^/]+$/,""),n=e.slice(s.length+1).replace(/\.json$/,"");return(0,ot.normalizePath)(`${s}/${n}.threads/${t}.json`)}getParentChatFilePath(){if(this.channelName&&this.conversationId){let t=this.settings.fleetFolder,e=oe(this.conversationId)||"conversation";return(0,ot.normalizePath)(`${t}/channels/${this.channelName}/sessions/${e}.json`)}if(this.inAppConversationId){let t=oe(this.inAppConversationId)||"conversation";if(this.agent.isFolder){let s=this.agent.filePath.replace(/\/agent\.md$/,"");return(0,ot.normalizePath)(`${s}/conversations/${t}.json`)}let e=this.repository.getMemoryPath(this.agent.name).replace(/\/[^/]+$/,"");return(0,ot.normalizePath)(`${e}/${this.agent.name}-conversations/${t}.json`)}throw new Error("ChatSession requires inAppConversationId (or channel options) to resolve a parent file path")}async ensureProcess(){if(this.process&&this.isProcessAlive)return;this.refreshAgent();let t=["--input-format","stream-json","--output-format","stream-json","--verbose"];this.claudeSessionId?(t.push("--resume",this.claudeSessionId),this.basePromptSent=!0,this.claudeResumeAttempted=!0):this.claudeResumeAttempted=!1;let e=Dt(null,this.agent,this.settings);ws(e.value)&&t.push("--model",e.value);let s=this.agent.permissionMode?.trim();s&&s!=="default"?t.push("--permission-mode",s):t.push("--permission-mode","bypassPermissions"),this.agent.effort&&t.push("--effort",this.agent.effort);let n=this.agent.cwd?.trim()?this.agent.cwd:this.repository.getVaultBasePath()??".",a=this.buildMcpProjection(n);t.push(...a.args),this.settingsState=sn(n,this.agent,{mcpAllowServers:a.allowServers});let r=dt(this.settings.claudeCliPath,t,{cwd:n,env:{...process.env,AWS_REGION:this.settings.awsRegion,...a.env}});this.process=r,this.isProcessAlive=!0,this.stdoutBuffer="",this.processListeners={onStdout:o=>this.handleStdout(o),onStderr:()=>{},onError:o=>this.handleProcessError(o),onClose:()=>this.handleProcessClose()},r.stdout.on("data",this.processListeners.onStdout),r.stderr.on("data",this.processListeners.onStderr),r.on("error",this.processListeners.onError),r.on("close",this.processListeners.onClose)}detachProcessListeners(){this.process&&this.processListeners&&(this.process.stdout?.removeListener("data",this.processListeners.onStdout),this.process.stderr?.removeListener("data",this.processListeners.onStderr),this.process.removeListener("error",this.processListeners.onError),this.process.removeListener("close",this.processListeners.onClose)),this.processListeners=null}handleStdout(t){this.isStreaming&&this.armWatchdog(),this.stdoutBuffer+=t.toString();let e=ye(this.stdoutBuffer);this.stdoutBuffer=e.pop()??"";for(let s of e){let n=s.trim();if(n)try{let a=JSON.parse(n);this.handleEvent(a)}catch{}}}handleEvent(t){if(this.isCodex){this.handleCodexEvent(t);return}if(typeof t.session_id=="string"&&(this.claudeSessionId=t.session_id),this.updateStatsFromEvent(t),t.type==="system"&&t.subtype==="compact_boundary"){let s=t.compact_metadata,n=s&&typeof s.pre_tokens=="number"?s.pre_tokens:0,a=s&&typeof s.post_tokens=="number"?s.post_tokens:0;a>0&&(this.stats.contextTokensUsed=a),this.stats.lastCompact={preTokens:n,postTokens:a},this.emitStats(),this.needsCompactBeforeNextTurn=!1,this.activeOnEvent?.({type:"compacted",content:"",compact:{preTokens:n,postTokens:a}});return}if(t.type==="result"){let s=typeof t.api_error_status=="string"&&!!t.api_error_status;if(t.is_error===!0||s){let n=this.describeResultError(t);this.activeOnEvent?.({type:"error",content:"",errorMessage:n}),this.claudeResumeAttempted&&t.is_error===!0&&!s&&this.clearSessionId()}this.handleTurnEnd();return}let e=this.parseStreamEvent(t);e&&this.dispatchStreamEvent(e)}dispatchStreamEvent(t){let e=t;if(t.type==="text"){let s=this.turnResponseText.length===0;if(s&&(this.displayedLen=0),this.turnResponseText+=t.content,this.setCurrentTool(void 0),s&&this.turnResponseText.length>0&&this.emitActivity(),this.agent.memory){let n=ui(this.turnResponseText),a=n.length>this.displayedLen?n.slice(this.displayedLen):"";this.displayedLen=n.length,e={...t,content:a}}}else t.type==="tool_use"&&t.toolName&&(this.turnToolCalls.push({name:t.toolName,command:t.content||void 0}),this.setCurrentTool(t.toolName));this.activeOnEvent?.(e)}handleCodexEvent(t){this.codexTurnState||(this.codexTurnState=ya());let e=ar(t,this.codexTurnState);for(let s of e)switch(s.kind){case"session":this.claudeSessionId=s.sessionId;break;case"text":this.dispatchStreamEvent({type:"text",content:s.text});break;case"tool":this.dispatchStreamEvent({type:"tool_use",content:s.command?s.command.slice(0,150):"",toolName:s.toolName});break;case"usage":{s.contextTokens>0&&s.contextTokens!==this.stats.contextTokensUsed&&(this.stats.contextTokensUsed=s.contextTokens),this.stats.turnCount+=1,this.emitStats();break}case"turn-failed":case"error":this.codexTurnErrors.push(s.message),this.activeOnEvent?.({type:"error",content:"",errorMessage:s.message});break}}get hasCurrentTurnText(){return this.turnResponseText.length>0}describeResultError(t){let e=[],s=typeof t.api_error_status=="string"?t.api_error_status:"",n=typeof t.subtype=="string"?t.subtype:"",a=typeof t.result=="string"?t.result:"";return s?e.push(`API ${s}`):n?e.push(n.replace(/_/g," ")):e.push("unknown error"),a&&e.push(`\u2014 ${a}`),e.join(" ")}updateStatsFromEvent(t){let e=!1,s=typeof t.model=="string"?t.model:void 0,n=t.message,a=n&&typeof n.model=="string"?n.model:void 0,r=s||a;if(r&&r!==this.stats.concreteModel&&(this.stats.concreteModel=r,e=!0),t.type==="rate_limit_event"){let o=t.rate_limit_info;o&&(this.stats.rateLimit={type:typeof o.rateLimitType=="string"?o.rateLimitType:"unknown",resetsAt:typeof o.resetsAt=="number"?o.resetsAt:void 0,status:typeof o.status=="string"?o.status:void 0,isUsingOverage:typeof o.isUsingOverage=="boolean"?o.isUsingOverage:void 0},e=!0)}if(t.type==="assistant"&&n){let o=n.usage;if(o){let c=typeof o.input_tokens=="number"?o.input_tokens:0,l=typeof o.cache_read_input_tokens=="number"?o.cache_read_input_tokens:0,h=typeof o.cache_creation_input_tokens=="number"?o.cache_creation_input_tokens:0,d=c+l+h;d>0&&d!==this.stats.contextTokensUsed&&(this.stats.contextTokensUsed=d,e=!0)}}if(t.type==="result"){let o=typeof t.total_cost_usd=="number"?t.total_cost_usd:0;o>0&&(this.stats.costTotalUsd+=o,e=!0);let c=t.modelUsage;if(c)for(let l of Object.values(c)){let h=l;typeof h.contextWindow=="number"&&h.contextWindow!==this.stats.contextWindow&&(this.stats.contextWindow=h.contextWindow,e=!0)}this.stats.turnCount+=1,e=!0,this.recordTurnUsage(t,o)}t.type==="result"&&this.evaluateAutoCompact(),e&&this.emitStats()}recordTurnUsage(t,e){if(!this.usageRecorder)return;let s=t.usage,n=h=>typeof h=="number"?h:0,a=n(s?.input_tokens),r=n(s?.output_tokens),o=n(s?.cache_read_input_tokens),c=n(s?.cache_creation_input_tokens),l=a+r+o+c;l===0&&e===0||this.usageRecorder({ts:new Date().toISOString(),agent:this.agent.name,source:this.channelName?"channel":"chat",model:this.stats.concreteModel??this.agent.model,inputTokens:a,outputTokens:r,cacheReadTokens:o,cacheCreateTokens:c,totalTokens:l,...e>0?{costUsd:e}:{}})}evaluateAutoCompact(){if(this.isCodex)return;let t=this.agent.autoCompactThreshold??0;if(t<=0||t>=100)return;let e=this.stats.contextWindow,s=this.stats.contextTokensUsed;!e||!s||s/e*100<t||Date.now()-this.lastCompactTriggerAt<3e4||(this.needsCompactBeforeNextTurn=!0)}handleTurnEnd(){this.lastActiveAt=Date.now();let t=this.turnResponseText,e=t;if(this.agent.memory){let n=qs(t);e=hs(t),n.length>0&&this.memoryWriter.capture(this.agent,n,this.captureSource(),new Date().toISOString()).catch(a=>console.warn(`Agent Fleet: chat memory capture failed for "${this.agent.name}"`,a)),this.memoryWriter.drainPending(this.agent,new Date().toISOString()).catch(a=>console.warn(`Agent Fleet: chat pending drain failed for "${this.agent.name}"`,a))}e.trim()&&this.messages.push({id:Cs(),role:"assistant",content:e,timestamp:new Date().toISOString(),toolCalls:this.turnToolCalls.length>0?[...this.turnToolCalls]:void 0});let s={text:this.turnResponseText,toolCalls:[...this.turnToolCalls]};if(this.activeOnEvent?.({type:"result",content:"",toolCalls:[...this.turnToolCalls]}),this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns--,this.isCodex){let n=this.codexQueue.shift();if(n!==void 0){this.armWatchdog(),this.startCodexTurn(n).catch(a=>{this.handleProcessError(a instanceof Error?a:new Error(String(a)))});return}this.pendingTurns=0}if(this.pendingTurns<=0){this.pendingTurns=0,this.clearWatchdog(),this.setStreaming(!1),this.persist();let n=this.turnResolve;this.turnResolve=null,this.turnReject=null,n?.(s)}}handleProcessError(t){this.isProcessAlive=!1,this.process=null,this.pendingTurns=0,this.turnResponseText="",this.turnToolCalls=[],this.codexQueue=[],this.clearWatchdog(),this.setStreaming(!1),Rt(this.settingsState),this.settingsState=null,this.codexPermState?.restore(),this.codexPermState=null,xt(this.mcpProjection),this.mcpProjection=null;let e=this.turnReject;this.turnResolve=null,this.turnReject=null,e?.(t)}handleProcessClose(){if(this.isProcessAlive=!1,this.process=null,Rt(this.settingsState),this.settingsState=null,xt(this.mcpProjection),this.mcpProjection=null,this.turnResolve){this.claudeResumeAttempted&&!this.turnResponseText.trim()&&this.clearSessionId();let t={text:this.turnResponseText,toolCalls:[...this.turnToolCalls]};this.turnResponseText.trim()&&this.messages.push({id:Cs(),role:"assistant",content:this.turnResponseText,timestamp:new Date().toISOString(),toolCalls:this.turnToolCalls.length>0?[...this.turnToolCalls]:void 0}),this.pendingTurns=0,this.turnResponseText="",this.turnToolCalls=[],this.clearWatchdog(),this.setStreaming(!1),this.persist();let e=this.turnResolve;this.turnResolve=null,this.turnReject=null,e?.(t)}}async sendMessage(t,e,s,n){this.lastActiveAt=Date.now(),this.messages.push({id:Cs(),role:"user",content:t,timestamp:new Date().toISOString(),attachments:n&&n.length>0?n:void 0});let a=s??t;if(this.basePromptSent||(a=`${await this.buildBasePrompt()}
11906
+ Document the reliable steps to handle this so future runs don't rediscover them.`};try{await this.repository.writeProposal(o),a.proposed=!0,await this.repository.writeCandidates(t.name,e)}catch(c){console.warn(`Agent Fleet: failed to write proposal for "${t.name}"`,c)}}}async runPendingTask({task:t,promptOverride:e}){let s=this.repository.getAgentByName(t.agent);if(!s||!s.enabled)return;let n=new Date().toISOString();this.runtimeState.set(s.name,{status:"running",currentTaskId:t.taskId,runStarted:n}),this.runOutputBuffers.set(s.name,""),this.emitStatusChange();try{let a=await this.executor.execute(s,t,e,m=>{let f=this.runOutputBuffers.get(s.name)??"";this.runOutputBuffers.set(s.name,f+m);let g=this.runOutputListeners.get(s.name);if(g)for(let y of g)y(m)}),i=this.consumeAborted(s.name),o=i?[]:this.buildApprovals(s,a.toolsUsed),c=i?"cancelled":this.resolveRunStatus(a,o),l={runId:a.runId,agent:s.name,task:t.taskId,status:c,started:n,completed:new Date().toISOString(),durationSeconds:a.durationSeconds,tokensUsed:a.tokensUsed,costUsd:a.costUsd,model:a.resolvedModel||s.model,modelSource:a.modelSource,concreteModel:a.concreteModel,exitCode:a.exitCode,tags:Array.from(new Set([...s.tags,...t.tags])),prompt:a.prompt,output:a.outputText,toolsUsed:a.toolsUsed.map(m=>`${m.tool}${m.command?`: ${m.command}`:""}`),finalResult:a.finalResult,stderr:a.stderr,approvals:o},h=await this.repository.writeRunLog(l);if(await this.repository.updateTaskRunMetadata(t,{lastRun:n,runCount:t.runCount+1}),s.memory){let f=t.tags.includes("heartbeat")?"heartbeat":`task:${t.taskId}`,g=qs(a.outputText);try{await this.memoryWriter.capture(s,g,f,new Date().toISOString())}catch(y){console.warn(`Agent Fleet: failed to append memory for "${s.name}"`,y)}}let d=t.tags.includes("heartbeat"),u=d?s.heartbeatChannel:t.channel??"",p=d?s.heartbeatChannelTarget??"":t.channelTarget??"";if(u&&!i&&l.output.trim())try{this.channelResultHandler?.(s.name,u,l.output,d?"heartbeat":t.taskId,p)}catch(m){console.warn(`Agent Fleet: channel delivery failed for ${s.name}`,m)}i&&(l.output="Task was manually stopped."),await this.refreshRunCaches(),this.runtimeState.set(s.name,{status:i||c==="success"?"idle":c==="pending_approval"?"pending":"error",currentRunId:a.runId,lastRun:{...l,filePath:h}}),i||this.notify(l)}catch(a){let i=this.consumeAborted(s.name),o=i?"cancelled":"failure",c=It(t,s,this.settings),l={runId:(0,Sa.randomUUID)(),agent:s.name,task:t.taskId,status:o,started:n,completed:new Date().toISOString(),durationSeconds:Math.round((Date.now()-new Date(n).getTime())/1e3),model:c.value||s.model,modelSource:c.source,exitCode:i?-1:1,tags:Array.from(new Set([...s.tags,...t.tags])),prompt:e??t.body,output:i?"Task was manually stopped.":a instanceof Error?a.message:String(a),toolsUsed:[]},h=await this.repository.writeRunLog(l);await this.refreshRunCaches(),this.runtimeState.set(s.name,{status:i?"idle":"error",lastRun:{...l,filePath:h}}),i||this.notify(l)}finally{if(s.memory)try{await this.memoryWriter.drainPending(s,new Date().toISOString())}catch(a){console.warn(`Agent Fleet: failed to drain pending memory for "${s.name}"`,a)}this.runOutputBuffers.delete(s.name),this.runOutputListeners.delete(s.name),this.emitStatusChange()}}buildApprovals(t,e){let s=e.filter(n=>t.approvalRequired.includes(n.tool)).map(n=>({tool:n.tool,command:n.command,reason:n.reason,status:"pending"}));return s.length>0?s:void 0}resolveRunStatus(t,e){return e?.length?"pending_approval":t.timedOut?"timeout":t.exitCode===0?"success":"failure"}notify(t){if(this.settings.notificationLevel==="none"||this.settings.notificationLevel==="failures-only"&&t.status==="success")return;let s=(ye(t.output).map(a=>a.trim()).find(a=>a&&!a.startsWith("{")&&!a.startsWith("["))??"").slice(0,120)||t.status,n=t.status==="success"?`\u2705 ${t.agent}: ${s}`:t.status==="pending_approval"?`\u{1F535} ${t.agent} needs approval: ${(t.approvals??[])[0]?.tool??"tool action"}`:`\u274C ${t.agent}: ${s}`;new wi.Notice(n,t.status==="success"?5e3:0)}emitStatusChange(){for(let t of this.statusChangeListeners)t()}};function uc(r){return r.toLowerCase().replace(/[^a-z0-9-]/g,"-").replace(/-{2,}/g,"-").replace(/^-|-$/g,"")}function xs(r,t){return`${r}-${uc(t)}`}var Vt="af-channel-cred",Ss="af-mcp-secret",fn=class{constructor(t){this.storage=t;t||console.warn("Agent Fleet: SecretStorage unavailable (Obsidian < 1.11.4). Secrets will use plaintext fallback.")}get available(){return!!this.storage}setJson(t,e,s){if(!this.storage)return;let n=xs(t,e);this.storage.setSecret(n,JSON.stringify(s))}getJson(t,e){if(!this.storage)return null;let s=xs(t,e),n=this.storage.getSecret(s);if(!n)return null;try{return JSON.parse(n)}catch{return null}}setString(t,e,s){if(!this.storage)return;let n=xs(t,e);this.storage.setSecret(n,s)}getString(t,e){if(!this.storage)return null;let s=xs(t,e);return this.storage.getSecret(s)||null}delete(t,e){if(!this.storage)return;let s=xs(t,e);this.storage.setSecret(s,"")}listByPrefix(t){return this.storage?this.storage.listSecrets().filter(e=>e.startsWith(t+"-")):[]}};var gn=class{tokens=new Map;oauthTokens=new Map;secretStore;setSecretStore(t){this.secretStore=t}storeProbeToken(t,e){this.tokens.set(t,e)}storeStaticToken(t,e){this.tokens.set(t,e),this.secretStore?.setJson(Ss,t,{accessToken:e})}storeOAuthToken(t,e){this.oauthTokens.set(t,e),this.tokens.set(t,e.accessToken),this.secretStore?.setJson(Ss,t,{accessToken:e.accessToken,refreshToken:e.refreshToken,expiresAt:e.expiresAt,tokenEndpoint:e.tokenEndpoint,clientId:e.clientId,resource:e.resource})}hydrate(t){if(!this.secretStore)return null;let e=this.secretStore.getJson(Ss,t);return e?.accessToken?(this.tokens.set(t,e.accessToken),e.tokenEndpoint&&e.clientId&&e.resource&&this.oauthTokens.set(t,{accessToken:e.accessToken,refreshToken:e.refreshToken,expiresAt:e.expiresAt,tokenEndpoint:e.tokenEndpoint,clientId:e.clientId,resource:e.resource}),e):null}getToken(t){return this.tokens.get(t)??this.hydrate(t)?.accessToken??void 0}getOAuthToken(t){return this.oauthTokens.has(t)?this.oauthTokens.get(t):(this.hydrate(t),this.oauthTokens.get(t))}hasToken(t){return this.tokens.has(t)||!!this.hydrate(t)}removeToken(t){this.tokens.delete(t),this.oauthTokens.delete(t),this.secretStore?.delete(Ss,t)}getExpiringTokens(t=5*6e4){let e=new Map,s=Date.now();for(let[n,a]of this.oauthTokens)a.refreshToken&&a.expiresAt&&a.expiresAt-s<t&&e.set(n,a);return e}};var ut=require("obsidian");var yn=require("crypto"),ot=require("obsidian");function Cs(){return(0,yn.randomUUID)()}function bi(r){let t=`${r.timestamp}|${r.content.slice(0,80)}`;return(0,yn.createHash)("sha1").update(t).digest("hex").slice(0,16)}var Yt=class r{constructor(t,e,s,n,a){this.agent=t;this.settings=e;this.repository=s;this.vault=n,this.channelName=a?.channelName,this.conversationId=a?.conversationId,this.channelContext=a?.channelContext,this.inAppConversationId=a?.inAppConversationId,this.threadAnchorId=a?.threadAnchorId,this.parentSession=a?.parentSession,this.mcpAuth=a?.mcpAuth,this.memoryWriter=new Gt(s)}messages=[];isStreaming=!1;isProcessAlive=!1;settingsState=null;mcpProjection=null;get pendingTurnCount(){return this.pendingTurns}lastActiveAt=Date.now();process=null;claudeSessionId=null;claudeResumeAttempted=!1;vault;stdoutBuffer="";processListeners=null;basePromptSent=!1;codexTurnState=null;codexQueue=[];codexResumeAttempted=!1;codexTurnErrors=[];codexStderr="";codexPermState=null;get isCodex(){return kt(this.agent.adapter)==="codex"}channelName;conversationId;channelContext;inAppConversationId;conversationName="";threadAnchorId;parentSession;threadAnchorIndex;threads=new Map;threadIndex={};activeOnEvent=null;turnResponseText="";displayedLen=0;turnToolCalls=[];pendingTurns=0;turnResolve=null;turnReject=null;needsCompactBeforeNextTurn=!1;lastCompactTriggerAt=0;static WATCHDOG_FALLBACK_MINUTES=10;watchdogTimer=null;getWatchdogMs(){let t=this.settings.chatWatchdogMinutes;return(typeof t=="number"&&t>0?t:r.WATCHDOG_FALLBACK_MINUTES)*60*1e3}armWatchdog(){this.clearWatchdog();let t=this.getWatchdogMs();this.watchdogTimer=window.setTimeout(()=>{if(this.watchdogTimer=null,!this.isStreaming)return;this.activeOnEvent?.({type:"error",content:"",errorMessage:`no response from the CLI for ${Math.round(t/6e4)} minutes \u2014 giving up`});let e=new Error("Watchdog timeout");this.handleProcessError(e);try{this.process?.kill()}catch{}},t)}clearWatchdog(){this.watchdogTimer&&(window.clearTimeout(this.watchdogTimer),this.watchdogTimer=null)}currentToolName;activityListeners=new Set;onActivityChange(t){return this.activityListeners.add(t),t(),()=>{this.activityListeners.delete(t)}}emitActivity(){for(let t of this.activityListeners)t()}setStreaming(t){this.isStreaming!==t&&(this.isStreaming=t,t||(this.currentToolName=void 0),this.emitActivity())}setCurrentTool(t){this.currentToolName!==t&&(this.currentToolName=t,this.emitActivity())}stats={costTotalUsd:0,turnCount:0};usageRecorder;lastResultCostUsd=0;setUsageRecorder(t){this.usageRecorder=t}statsListeners=new Set;onStatsChange(t){return this.statsListeners.add(t),t({...this.stats}),()=>{this.statsListeners.delete(t)}}getStats(){return{...this.stats}}emitStats(){let t={...this.stats};for(let e of this.statsListeners)e(t)}mcpAuth;buildMcpProjection(t){let e=this.agent.memory?this.repository.getPendingDirAbsolutePath(this.agent.name):null,s=cn({registry:this.repository.getMcpServers(),agentGrants:this.agent.mcpServers??[],getBearerToken:n=>this.mcpAuth?.getToken(n),remember:e?{pendingDir:e,source:`mcp:${this.captureSource()}`}:null});return this.mcpProjection=dn(t,this.agent.adapter,s),{args:this.mcpProjection?.args??[],env:this.mcpProjection?.env??{},allowServers:s.map(n=>n.def.name)}}memoryWriter;captureSource(){return`chat:${this.inAppConversationId||this.conversationId||"default"}`}getConversationName(){return this.conversationName}async setConversationName(t){this.conversationName=t,await this.persist()}refreshAgent(){let t=this.repository.getAgentByName(this.agent.name);t&&(this.agent=t)}get isThread(){return!!this.threadAnchorId}async loadPersistedState(){let t=this.getChatFilePath(),e=this.vault.getAbstractFileByPath(t);if(!(e instanceof ot.TFile))return!1;try{let s=await this.vault.cachedRead(e);if(this.isThread){let a=JSON.parse(s);return a.messages?.length>0||a.sessionId?(this.messages=(a.messages??[]).map(i=>i.id?i:{...i,id:bi(i)}),this.claudeSessionId=a.sessionId??null,this.threadAnchorIndex=a.anchorIndex,this.claudeSessionId&&(this.basePromptSent=!0),!0):!1}let n=JSON.parse(s);if(n.name&&(this.conversationName=n.name),n.messages?.length>0)return this.messages=n.messages.map(a=>a.id?a:{...a,id:bi(a)}),this.claudeSessionId=n.sessionId??null,this.threadIndex=n.threads??{},this.claudeSessionId&&(this.basePromptSent=!0),!0}catch{}return!1}getThreadIndex(){return{...this.threadIndex}}async persist(){let t=new Date().toISOString(),e=this.getChatFilePath(),s;if(this.isThread){let a={anchorMessageId:this.threadAnchorId,anchorIndex:this.threadAnchorIndex??0,sessionId:this.claudeSessionId,messages:this.messages,createdAt:this.parentSession?.threadIndex[this.threadAnchorId]?.createdAt??t,lastActive:t};s=JSON.stringify(a,null,2)}else{let a={sessionId:this.claudeSessionId,messages:this.messages,lastActive:t,name:this.conversationName||void 0,threads:Object.keys(this.threadIndex).length>0?this.threadIndex:void 0};s=JSON.stringify(a,null,2)}let n=this.vault.getAbstractFileByPath(e);n instanceof ot.TFile?await this.vault.modify(n,s):(await this.ensureParentFolders(e),await this.vault.create(e,s)),this.isThread&&this.parentSession&&this.threadAnchorId&&await this.parentSession.upsertThreadIndex(this.threadAnchorId,{path:e,createdAt:this.parentSession.threadIndex[this.threadAnchorId]?.createdAt??t,messageCount:this.messages.length,lastActive:t})}async upsertThreadIndex(t,e){this.threadIndex[t]=e,await this.persist()}async ensureParentFolders(t){let e=t.lastIndexOf("/");if(e<=0)return;let n=t.slice(0,e).split("/"),a="";for(let i of n)if(a=a?`${a}/${i}`:i,!this.vault.getAbstractFileByPath(a))try{await this.vault.createFolder(a)}catch(o){if(!(o instanceof Error?o.message:String(o)).includes("already exists"))throw o}}async clearPersistedState(){let t=this.getChatFilePath();await this.repository.trashFile(t),this.messages=[],this.claudeSessionId=null,this.basePromptSent=!1}getChatFilePath(){if(this.threadAnchorId&&this.parentSession)return this.parentSession.getThreadFilePath(this.threadAnchorId);if(this.channelName&&this.conversationId){let s=this.settings.fleetFolder,n=oe(this.conversationId)||"conversation";return(0,ot.normalizePath)(`${s}/channels/${this.channelName}/sessions/${n}.json`)}if(!this.inAppConversationId)throw new Error("ChatSession requires inAppConversationId (or channel options) to resolve a file path");let t=oe(this.inAppConversationId)||"conversation";if(this.agent.isFolder){let s=this.agent.filePath.replace(/\/agent\.md$/,"");return(0,ot.normalizePath)(`${s}/conversations/${t}.json`)}let e=this.repository.getMemoryPath(this.agent.name).replace(/\/[^/]+$/,"");return(0,ot.normalizePath)(`${e}/${this.agent.name}-conversations/${t}.json`)}getThreadFilePath(t){let e=this.getParentChatFilePath(),s=e.replace(/\/[^/]+$/,""),n=e.slice(s.length+1).replace(/\.json$/,"");return(0,ot.normalizePath)(`${s}/${n}.threads/${t}.json`)}getParentChatFilePath(){if(this.channelName&&this.conversationId){let t=this.settings.fleetFolder,e=oe(this.conversationId)||"conversation";return(0,ot.normalizePath)(`${t}/channels/${this.channelName}/sessions/${e}.json`)}if(this.inAppConversationId){let t=oe(this.inAppConversationId)||"conversation";if(this.agent.isFolder){let s=this.agent.filePath.replace(/\/agent\.md$/,"");return(0,ot.normalizePath)(`${s}/conversations/${t}.json`)}let e=this.repository.getMemoryPath(this.agent.name).replace(/\/[^/]+$/,"");return(0,ot.normalizePath)(`${e}/${this.agent.name}-conversations/${t}.json`)}throw new Error("ChatSession requires inAppConversationId (or channel options) to resolve a parent file path")}async ensureProcess(){if(this.process&&this.isProcessAlive)return;this.refreshAgent();let t=["--input-format","stream-json","--output-format","stream-json","--verbose"];this.claudeSessionId?(t.push("--resume",this.claudeSessionId),this.basePromptSent=!0,this.claudeResumeAttempted=!0):this.claudeResumeAttempted=!1;let e=It(null,this.agent,this.settings);ws(e.value)&&t.push("--model",e.value);let s=this.agent.permissionMode?.trim();s&&s!=="default"?t.push("--permission-mode",s):t.push("--permission-mode","bypassPermissions"),this.agent.effort&&t.push("--effort",this.agent.effort);let n=this.agent.cwd?.trim()?this.agent.cwd:this.repository.getVaultBasePath()??".",a=this.buildMcpProjection(n);t.push(...a.args),this.settingsState=sn(n,this.agent,{mcpAllowServers:a.allowServers});let i=dt(this.settings.claudeCliPath,t,{cwd:n,env:{...process.env,AWS_REGION:this.settings.awsRegion,...a.env}});this.process=i,this.isProcessAlive=!0,this.stdoutBuffer="",this.lastResultCostUsd=0,this.processListeners={onStdout:o=>this.handleStdout(o),onStderr:()=>{},onError:o=>this.handleProcessError(o),onClose:()=>this.handleProcessClose()},i.stdout.on("data",this.processListeners.onStdout),i.stderr.on("data",this.processListeners.onStderr),i.on("error",this.processListeners.onError),i.on("close",this.processListeners.onClose)}detachProcessListeners(){this.process&&this.processListeners&&(this.process.stdout?.removeListener("data",this.processListeners.onStdout),this.process.stderr?.removeListener("data",this.processListeners.onStderr),this.process.removeListener("error",this.processListeners.onError),this.process.removeListener("close",this.processListeners.onClose)),this.processListeners=null}handleStdout(t){this.isStreaming&&this.armWatchdog(),this.stdoutBuffer+=t.toString();let e=ye(this.stdoutBuffer);this.stdoutBuffer=e.pop()??"";for(let s of e){let n=s.trim();if(n)try{let a=JSON.parse(n);this.handleEvent(a)}catch{}}}handleEvent(t){if(this.isCodex){this.handleCodexEvent(t);return}if(typeof t.session_id=="string"&&(this.claudeSessionId=t.session_id),this.updateStatsFromEvent(t),t.type==="system"&&t.subtype==="compact_boundary"){let s=t.compact_metadata,n=s&&typeof s.pre_tokens=="number"?s.pre_tokens:0,a=s&&typeof s.post_tokens=="number"?s.post_tokens:0;a>0&&(this.stats.contextTokensUsed=a),this.stats.lastCompact={preTokens:n,postTokens:a},this.emitStats(),this.needsCompactBeforeNextTurn=!1,this.activeOnEvent?.({type:"compacted",content:"",compact:{preTokens:n,postTokens:a}});return}if(t.type==="result"){let s=typeof t.api_error_status=="string"&&!!t.api_error_status;if(t.is_error===!0||s){let n=this.describeResultError(t);this.activeOnEvent?.({type:"error",content:"",errorMessage:n}),this.claudeResumeAttempted&&t.is_error===!0&&!s&&this.clearSessionId()}this.handleTurnEnd();return}let e=this.parseStreamEvent(t);e&&this.dispatchStreamEvent(e)}dispatchStreamEvent(t){let e=t;if(t.type==="text"){let s=this.turnResponseText.length===0;if(s&&(this.displayedLen=0),this.turnResponseText+=t.content,this.setCurrentTool(void 0),s&&this.turnResponseText.length>0&&this.emitActivity(),this.agent.memory){let n=ur(this.turnResponseText),a=n.length>this.displayedLen?n.slice(this.displayedLen):"";this.displayedLen=n.length,e={...t,content:a}}}else t.type==="tool_use"&&t.toolName&&(this.turnToolCalls.push({name:t.toolName,command:t.content||void 0}),this.setCurrentTool(t.toolName));this.activeOnEvent?.(e)}handleCodexEvent(t){this.codexTurnState||(this.codexTurnState=ya());let e=ri(t,this.codexTurnState);for(let s of e)switch(s.kind){case"session":this.claudeSessionId=s.sessionId;break;case"text":this.dispatchStreamEvent({type:"text",content:s.text});break;case"tool":this.dispatchStreamEvent({type:"tool_use",content:s.command?s.command.slice(0,150):"",toolName:s.toolName});break;case"usage":{s.contextTokens>0&&s.contextTokens!==this.stats.contextTokensUsed&&(this.stats.contextTokensUsed=s.contextTokens),this.stats.turnCount+=1,this.emitStats();break}case"turn-failed":case"error":this.codexTurnErrors.push(s.message),this.activeOnEvent?.({type:"error",content:"",errorMessage:s.message});break}}get hasCurrentTurnText(){return this.turnResponseText.length>0}describeResultError(t){let e=[],s=typeof t.api_error_status=="string"?t.api_error_status:"",n=typeof t.subtype=="string"?t.subtype:"",a=typeof t.result=="string"?t.result:"";return s?e.push(`API ${s}`):n?e.push(n.replace(/_/g," ")):e.push("unknown error"),a&&e.push(`\u2014 ${a}`),e.join(" ")}updateStatsFromEvent(t){let e=!1,s=typeof t.model=="string"?t.model:void 0,n=t.message,a=n&&typeof n.model=="string"?n.model:void 0,i=s||a;if(i&&i!==this.stats.concreteModel&&(this.stats.concreteModel=i,e=!0),t.type==="rate_limit_event"){let o=t.rate_limit_info;o&&(this.stats.rateLimit={type:typeof o.rateLimitType=="string"?o.rateLimitType:"unknown",resetsAt:typeof o.resetsAt=="number"?o.resetsAt:void 0,status:typeof o.status=="string"?o.status:void 0,isUsingOverage:typeof o.isUsingOverage=="boolean"?o.isUsingOverage:void 0},e=!0)}if(t.type==="assistant"&&n){let o=n.usage;if(o){let c=typeof o.input_tokens=="number"?o.input_tokens:0,l=typeof o.cache_read_input_tokens=="number"?o.cache_read_input_tokens:0,h=typeof o.cache_creation_input_tokens=="number"?o.cache_creation_input_tokens:0,d=c+l+h;d>0&&d!==this.stats.contextTokensUsed&&(this.stats.contextTokensUsed=d,e=!0)}}if(t.type==="result"){let o=typeof t.total_cost_usd=="number"?t.total_cost_usd:0,c=Math.max(0,o-this.lastResultCostUsd);this.lastResultCostUsd=o,c>0&&(this.stats.costTotalUsd+=c,e=!0);let l=t.modelUsage;if(l)for(let h of Object.values(l)){let d=h;typeof d.contextWindow=="number"&&d.contextWindow!==this.stats.contextWindow&&(this.stats.contextWindow=d.contextWindow,e=!0)}this.stats.turnCount+=1,e=!0,this.recordTurnUsage(t,c)}t.type==="result"&&this.evaluateAutoCompact(),e&&this.emitStats()}recordTurnUsage(t,e){if(!this.usageRecorder)return;let s=t.usage,n=h=>typeof h=="number"?h:0,a=n(s?.input_tokens),i=n(s?.output_tokens),o=n(s?.cache_read_input_tokens),c=n(s?.cache_creation_input_tokens),l=a+i+o+c;l===0&&e===0||this.usageRecorder({ts:new Date().toISOString(),agent:this.agent.name,source:this.channelName?"channel":"chat",model:this.stats.concreteModel??this.agent.model,inputTokens:a,outputTokens:i,cacheReadTokens:o,cacheCreateTokens:c,totalTokens:l,...e>0?{costUsd:e}:{}})}evaluateAutoCompact(){if(this.isCodex)return;let t=this.agent.autoCompactThreshold??0;if(t<=0||t>=100)return;let e=this.stats.contextWindow,s=this.stats.contextTokensUsed;!e||!s||s/e*100<t||Date.now()-this.lastCompactTriggerAt<3e4||(this.needsCompactBeforeNextTurn=!0)}handleTurnEnd(){this.lastActiveAt=Date.now();let t=this.turnResponseText,e=t;if(this.agent.memory){let n=qs(t);e=At(t),n.length>0&&this.memoryWriter.capture(this.agent,n,this.captureSource(),new Date().toISOString()).catch(a=>console.warn(`Agent Fleet: chat memory capture failed for "${this.agent.name}"`,a)),this.memoryWriter.drainPending(this.agent,new Date().toISOString()).catch(a=>console.warn(`Agent Fleet: chat pending drain failed for "${this.agent.name}"`,a))}e.trim()&&this.messages.push({id:Cs(),role:"assistant",content:e,timestamp:new Date().toISOString(),toolCalls:this.turnToolCalls.length>0?[...this.turnToolCalls]:void 0});let s={text:this.turnResponseText,toolCalls:[...this.turnToolCalls]};if(this.activeOnEvent?.({type:"result",content:"",toolCalls:[...this.turnToolCalls]}),this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns--,this.isCodex){let n=this.codexQueue.shift();if(n!==void 0){this.armWatchdog(),this.startCodexTurn(n).catch(a=>{this.handleProcessError(a instanceof Error?a:new Error(String(a)))});return}this.pendingTurns=0}if(this.pendingTurns<=0){this.pendingTurns=0,this.clearWatchdog(),this.setStreaming(!1),this.persist();let n=this.turnResolve;this.turnResolve=null,this.turnReject=null,n?.(s)}}handleProcessError(t){this.isProcessAlive=!1,this.process=null,this.pendingTurns=0,this.turnResponseText="",this.turnToolCalls=[],this.codexQueue=[],this.clearWatchdog(),this.setStreaming(!1),Dt(this.settingsState),this.settingsState=null,this.codexPermState?.restore(),this.codexPermState=null,xt(this.mcpProjection),this.mcpProjection=null;let e=this.turnReject;this.turnResolve=null,this.turnReject=null,e?.(t)}handleProcessClose(){if(this.isProcessAlive=!1,this.process=null,Dt(this.settingsState),this.settingsState=null,xt(this.mcpProjection),this.mcpProjection=null,this.turnResolve){this.claudeResumeAttempted&&!this.turnResponseText.trim()&&this.clearSessionId();let t={text:this.turnResponseText,toolCalls:[...this.turnToolCalls]};this.turnResponseText.trim()&&this.messages.push({id:Cs(),role:"assistant",content:this.turnResponseText,timestamp:new Date().toISOString(),toolCalls:this.turnToolCalls.length>0?[...this.turnToolCalls]:void 0}),this.pendingTurns=0,this.turnResponseText="",this.turnToolCalls=[],this.clearWatchdog(),this.setStreaming(!1),this.persist();let e=this.turnResolve;this.turnResolve=null,this.turnReject=null,e?.(t)}}async sendMessage(t,e,s,n){this.lastActiveAt=Date.now(),this.messages.push({id:Cs(),role:"user",content:t,timestamp:new Date().toISOString(),attachments:n&&n.length>0?n:void 0});let a=s??t;if(this.basePromptSent||(a=`${await this.buildBasePrompt()}
11996
11907
 
11997
11908
  ## Task
11998
- ${a}`,this.basePromptSent=!0),this.isCodex){this.stats.lastCompact&&(this.stats.lastCompact=void 0,this.emitStats()),this.activeOnEvent=e,this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns=1,this.setStreaming(!0),this.armWatchdog();try{await this.startCodexTurn(a)}catch(c){throw this.pendingTurns=0,this.clearWatchdog(),this.setStreaming(!1),new Error(`Failed to start Codex process: ${c instanceof Error?c.message:String(c)}`)}return new Promise((c,l)=>{this.turnResolve=c,this.turnReject=l})}await this.ensureProcess();let r=this.needsCompactBeforeNextTurn;r&&(this.needsCompactBeforeNextTurn=!1,this.lastCompactTriggerAt=Date.now()),this.stats.lastCompact&&(this.stats.lastCompact=void 0,this.emitStats()),this.activeOnEvent=e,this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns=r?2:1,this.setStreaming(!0),this.armWatchdog();let o=c=>{let l=JSON.stringify({type:"user",message:{role:"user",content:c}});this.process.stdin.write(l+`
11999
- `)};try{r&&o("/compact"),o(a)}catch(c){throw this.pendingTurns=0,this.setStreaming(!1),new Error(`Failed to write to Claude process stdin: ${c instanceof Error?c.message:String(c)}`)}return new Promise((c,l)=>{this.turnResolve=c,this.turnReject=l})}scheduleCompact(){this.isCodex||(this.needsCompactBeforeNextTurn=!0)}async startCodexTurn(t){this.refreshAgent();let e=Dt(null,this.agent,this.settings),s=await vs.buildExec({prompt:t,model:ws(e.value)?e.value:"",modelSource:e.source,effort:this.agent.effort??"",agent:this.agent,settings:this.settings,streaming:!0,resumeSessionId:this.claudeSessionId});e.value&&this.stats.concreteModel!==e.value&&(this.stats.concreteModel=e.value,this.emitStats()),this.codexTurnState=ya(),this.codexResumeAttempted=!!this.claudeSessionId,this.codexTurnErrors=[],this.codexStderr="";let n=this.agent.cwd?.trim()?this.agent.cwd:this.repository.getVaultBasePath()??".",a=this.buildMcpProjection(n);s.args.push(...a.args),this.codexPermState=await vs.setupPermissions(n,this.agent,this.settings);let r=dt(s.cliPath,s.args,{cwd:n,env:{...process.env,AWS_REGION:this.settings.awsRegion,...a.env,...this.codexPermState?.env??{}}});this.process=r,this.isProcessAlive=!0,this.stdoutBuffer="",this.processListeners={onStdout:o=>this.handleStdout(o),onStderr:o=>{this.codexStderr+=o.toString()},onError:o=>this.handleProcessError(o),onClose:o=>this.handleCodexProcessClose(o)},r.stdout.on("data",this.processListeners.onStdout),r.stderr.on("data",this.processListeners.onStderr),r.on("error",this.processListeners.onError),r.on("close",this.processListeners.onClose);try{r.stdin.write(s.stdinPayload??t),r.stdin.end()}catch(o){try{r.kill()}catch{}throw o instanceof Error?o:new Error(String(o))}}handleCodexProcessClose(t){if(this.detachProcessListeners(),this.isProcessAlive=!1,this.process=null,this.stdoutBuffer="",this.codexPermState?.restore(),this.codexPermState=null,xt(this.mcpProjection),this.mcpProjection=null,t!==0&&!this.turnResponseText.trim()){let s=this.codexTurnErrors.join("; ")||this.codexStderr.trim().slice(-500)||`Codex CLI exited with code ${t??"unknown"}`;this.codexResumeAttempted&&this.clearSessionId(),this.codexQueue=[],this.activeOnEvent?.({type:"error",content:"",errorMessage:s})}this.handleTurnEnd()}injectMessage(t,e,s){if(this.isCodex){if(!this.isStreaming&&this.pendingTurns===0)return;this.messages.push({id:Cs(),role:"user",content:t,timestamp:new Date().toISOString(),attachments:s&&s.length>0?s:void 0}),this.codexQueue.push(e??t),this.pendingTurns++;return}if(!this.process||!this.isProcessAlive)return;this.messages.push({id:Cs(),role:"user",content:t,timestamp:new Date().toISOString(),attachments:s&&s.length>0?s:void 0});let a=JSON.stringify({type:"user",message:{role:"user",content:e??t}});try{this.process.stdin.write(a+`
12000
- `)}catch(r){console.warn("Agent Fleet: injectMessage stdin write failed",r);return}this.pendingTurns++}abort(){this.detachProcessListeners(),this.process&&(this.process.kill(),this.process=null),this.isProcessAlive=!1,this.stdoutBuffer="",this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns=0,this.codexQueue=[],this.clearWatchdog(),this.setStreaming(!1),Rt(this.settingsState),this.settingsState=null,this.codexPermState?.restore(),this.codexPermState=null,xt(this.mcpProjection),this.mcpProjection=null;let t=this.turnReject;this.turnResolve=null,this.turnReject=null,t?.(new Error("Aborted"))}dispose(){for(let t of this.threads.values())t.abort();this.threads.clear(),this.threadIndex={},this.abort()}hibernate(){this.isStreaming||this.pendingTurns>0||(this.detachProcessListeners(),this.process&&(this.process.kill(),this.process=null),this.isProcessAlive=!1,this.stdoutBuffer="",Rt(this.settingsState),this.settingsState=null,xt(this.mcpProjection),this.mcpProjection=null)}clearSessionId(){this.claudeSessionId=null,this.basePromptSent=!1,this.claudeResumeAttempted=!1}async buildBasePrompt(){let t=[this.agent.body.trim()];for(let s of this.agent.skills){let n=this.repository.getSkillByName(s);if(n){let a=[n.body.trim()];n.toolsBody.trim()&&a.push(`### Tools
11909
+ ${a}`,this.basePromptSent=!0),this.isCodex){this.stats.lastCompact&&(this.stats.lastCompact=void 0,this.emitStats()),this.activeOnEvent=e,this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns=1,this.setStreaming(!0),this.armWatchdog();try{await this.startCodexTurn(a)}catch(c){throw this.pendingTurns=0,this.clearWatchdog(),this.setStreaming(!1),new Error(`Failed to start Codex process: ${c instanceof Error?c.message:String(c)}`)}return new Promise((c,l)=>{this.turnResolve=c,this.turnReject=l})}await this.ensureProcess();let i=this.needsCompactBeforeNextTurn;i&&(this.needsCompactBeforeNextTurn=!1,this.lastCompactTriggerAt=Date.now()),this.stats.lastCompact&&(this.stats.lastCompact=void 0,this.emitStats()),this.activeOnEvent=e,this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns=i?2:1,this.setStreaming(!0),this.armWatchdog();let o=c=>{let l=JSON.stringify({type:"user",message:{role:"user",content:c}});this.process.stdin.write(l+`
11910
+ `)};try{i&&o("/compact"),o(a)}catch(c){throw this.pendingTurns=0,this.setStreaming(!1),new Error(`Failed to write to Claude process stdin: ${c instanceof Error?c.message:String(c)}`)}return new Promise((c,l)=>{this.turnResolve=c,this.turnReject=l})}scheduleCompact(){this.isCodex||(this.needsCompactBeforeNextTurn=!0)}async startCodexTurn(t){this.refreshAgent();let e=It(null,this.agent,this.settings),s=await vs.buildExec({prompt:t,model:ws(e.value)?e.value:"",modelSource:e.source,effort:this.agent.effort??"",agent:this.agent,settings:this.settings,streaming:!0,resumeSessionId:this.claudeSessionId});e.value&&this.stats.concreteModel!==e.value&&(this.stats.concreteModel=e.value,this.emitStats()),this.codexTurnState=ya(),this.codexResumeAttempted=!!this.claudeSessionId,this.codexTurnErrors=[],this.codexStderr="";let n=this.agent.cwd?.trim()?this.agent.cwd:this.repository.getVaultBasePath()??".",a=this.buildMcpProjection(n);s.args.push(...a.args),this.codexPermState=await vs.setupPermissions(n,this.agent,this.settings);let i=dt(s.cliPath,s.args,{cwd:n,env:{...process.env,AWS_REGION:this.settings.awsRegion,...a.env,...this.codexPermState?.env??{}}});this.process=i,this.isProcessAlive=!0,this.stdoutBuffer="",this.processListeners={onStdout:o=>this.handleStdout(o),onStderr:o=>{this.codexStderr+=o.toString()},onError:o=>this.handleProcessError(o),onClose:o=>this.handleCodexProcessClose(o)},i.stdout.on("data",this.processListeners.onStdout),i.stderr.on("data",this.processListeners.onStderr),i.on("error",this.processListeners.onError),i.on("close",this.processListeners.onClose);try{i.stdin.write(s.stdinPayload??t),i.stdin.end()}catch(o){try{i.kill()}catch{}throw o instanceof Error?o:new Error(String(o))}}handleCodexProcessClose(t){if(this.detachProcessListeners(),this.isProcessAlive=!1,this.process=null,this.stdoutBuffer="",this.codexPermState?.restore(),this.codexPermState=null,xt(this.mcpProjection),this.mcpProjection=null,t!==0&&!this.turnResponseText.trim()){let s=this.codexTurnErrors.join("; ")||this.codexStderr.trim().slice(-500)||`Codex CLI exited with code ${t??"unknown"}`;this.codexResumeAttempted&&this.clearSessionId(),this.codexQueue=[],this.activeOnEvent?.({type:"error",content:"",errorMessage:s})}this.handleTurnEnd()}injectMessage(t,e,s){if(this.isCodex){if(!this.isStreaming&&this.pendingTurns===0)return;this.messages.push({id:Cs(),role:"user",content:t,timestamp:new Date().toISOString(),attachments:s&&s.length>0?s:void 0}),this.codexQueue.push(e??t),this.pendingTurns++;return}if(!this.process||!this.isProcessAlive)return;this.messages.push({id:Cs(),role:"user",content:t,timestamp:new Date().toISOString(),attachments:s&&s.length>0?s:void 0});let a=JSON.stringify({type:"user",message:{role:"user",content:e??t}});try{this.process.stdin.write(a+`
11911
+ `)}catch(i){console.warn("Agent Fleet: injectMessage stdin write failed",i);return}this.pendingTurns++}abort(){this.detachProcessListeners(),this.process&&(this.process.kill(),this.process=null),this.isProcessAlive=!1,this.stdoutBuffer="",this.turnResponseText="",this.turnToolCalls=[],this.pendingTurns=0,this.codexQueue=[],this.clearWatchdog(),this.setStreaming(!1),Dt(this.settingsState),this.settingsState=null,this.codexPermState?.restore(),this.codexPermState=null,xt(this.mcpProjection),this.mcpProjection=null;let t=this.turnReject;this.turnResolve=null,this.turnReject=null,t?.(new Error("Aborted"))}dispose(){for(let t of this.threads.values())t.abort();this.threads.clear(),this.threadIndex={},this.abort()}hibernate(){this.isStreaming||this.pendingTurns>0||(this.detachProcessListeners(),this.process&&(this.process.kill(),this.process=null),this.isProcessAlive=!1,this.stdoutBuffer="",Dt(this.settingsState),this.settingsState=null,xt(this.mcpProjection),this.mcpProjection=null)}clearSessionId(){this.claudeSessionId=null,this.basePromptSent=!1,this.claudeResumeAttempted=!1}async buildBasePrompt(){let t=[this.agent.body.trim()];for(let s of this.agent.skills){let n=this.repository.getSkillByName(s);if(n){let a=[n.body.trim()];n.toolsBody.trim()&&a.push(`### Tools
12001
11912
  ${n.toolsBody.trim()}`),n.referencesBody.trim()&&a.push(`### References
12002
11913
  ${n.referencesBody.trim()}`),n.examplesBody.trim()&&a.push(`### Examples
12003
11914
  ${n.examplesBody.trim()}`),t.push(`## Skill: ${n.name}
@@ -12006,22 +11917,22 @@ ${a.join(`
12006
11917
  `)}`)}}if(this.agent.skillsBody.trim()&&t.push(`## Agent Skills
12007
11918
  ${this.agent.skillsBody.trim()}`),this.agent.contextBody.trim()&&t.push(`## Agent Context
12008
11919
  ${this.agent.contextBody.trim()}`),this.agent.memory){let s=await this.repository.readWorkingMemory(this.agent.name),n=Gs(this.agent,s);n&&t.push(n)}this.channelContext&&this.channelContext.trim()&&t.push(`## Channel Context
12009
- ${this.channelContext.trim()}`);let e=rn(this.agent,this.repository);if(e&&t.push(e),this.isThread&&this.parentSession&&this.threadAnchorIndex!==void 0){let s="You are continuing a side thread from this conversation. The user is following up on one of your earlier replies and wants to explore something specific without adding to the main thread. Your answers here stay in this thread only and will NOT be added back to the main conversation.",n=this.parentSession.messages.slice(0,this.threadAnchorIndex+1),a=["## Conversation so far"];for(let r of n){let o=r.role==="user"?"User":"Assistant";a.push(`${o}: ${r.content.trim()}`)}t.push(`## Thread Mode
11920
+ ${this.channelContext.trim()}`);let e=rn(this.agent,this.repository);if(e&&t.push(e),this.isThread&&this.parentSession&&this.threadAnchorIndex!==void 0){let s="You are continuing a side thread from this conversation. The user is following up on one of your earlier replies and wants to explore something specific without adding to the main thread. Your answers here stay in this thread only and will NOT be added back to the main conversation.",n=this.parentSession.messages.slice(0,this.threadAnchorIndex+1),a=["## Conversation so far"];for(let i of n){let o=i.role==="user"?"User":"Assistant";a.push(`${o}: ${i.content.trim()}`)}t.push(`## Thread Mode
12010
11921
  ${s}
12011
11922
 
12012
11923
  ${a.join(`
12013
11924
  `)}`)}return t.filter(Boolean).join(`
12014
11925
 
12015
- `)}async openOrCreateThread(t){if(this.isThread)throw new Error("Nested threads are not supported.");let e=this.threads.get(t);if(e)return e;let s=this.messages.findIndex(r=>r.id===t);if(s<0)throw new Error(`Thread anchor message "${t}" not found in parent.`);let n=new i(this.agent,this.settings,this.repository,this.vault,{threadAnchorId:t,parentSession:this,mcpAuth:this.mcpAuth});return n.threadAnchorIndex=s,await n.loadPersistedState()||(n.threadAnchorIndex=s),this.threads.set(t,n),n}closeThread(t){let e=this.threads.get(t);e&&(e.abort(),this.threads.delete(t))}hibernateIdleThreads(t){let e=Date.now();for(let s of this.threads.values())s.isProcessAlive&&e-s.lastActiveAt>t&&s.hibernate()}parseStreamEvent(t){let e=t.type;if(e==="assistant"){let s=t.message;if(s?.content&&Array.isArray(s.content))for(let n of s.content){if(n.type==="text"&&typeof n.text=="string")return{type:"text",content:n.text};if(n.type==="tool_use"){let a=String(n.name??"tool"),r=n.input,o=r?.command??r?.content??r?.file_path??r?.path??"";return{type:"tool_use",content:o?String(o).slice(0,150):"",toolName:a}}}}if(e==="content_block_delta"){let s=t.delta;if(s?.type==="text_delta"&&typeof s.text=="string")return{type:"text",content:s.text}}return null}};var vn=class{constructor(t){this.config=t;this.now=t.now??Date.now}buckets=new Map;now;tryConsume(t){let e=this.now(),s=e-this.config.windowMs,a=(this.buckets.get(t)??[]).filter(r=>r>s);return a.length>=this.config.maxPerWindow?(this.buckets.set(t,a),!1):(a.push(e),this.buckets.set(t,a),!0)}currentCount(t){let s=this.now()-this.config.windowMs;return(this.buckets.get(t)??[]).filter(a=>a>s).length}reset(t){this.buckets.delete(t)}resetAll(){this.buckets.clear()}};function Ts(i){let t=uc(i),e=[];for(let s of t)if(s.kind==="code"){let n=s.text.replace(/^```[^\n]*\n/,"```\n");e.push(n)}else e.push(pc(s.text));return e.join("")}function uc(i){let t=[],e=0,s=!1,n=0;for(;e<i.length;)i.startsWith("```",e)?s?(e+=3,i[e]===`
12016
- `&&(e+=1),t.push({kind:"code",text:i.slice(n,e)}),n=e,s=!1):(e>n&&t.push({kind:"prose",text:i.slice(n,e)}),n=e,s=!0,e+=3):e+=1;return n<i.length&&t.push({kind:s?"code":"prose",text:i.slice(n)}),t}function pc(i){return mc(i).map(s=>{if(s.isCode)return s.text;let n=s.text;return n=n.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;"),n=n.replace(/\[([^\]]+)\]\(([^)]+)\)/g,(a,r,o)=>`<${o}|${r}>`),n=n.replace(/\*\*([^*]+)\*\*/g,"*$1*"),n=n.replace(/^#{1,6}\s+(.+)$/gm,"*$1*"),n}).join("")}function mc(i){let t=[],e=/`([^`\n]+)`/g,s=0,n;for(;n=e.exec(i);)n.index>s&&t.push({text:i.slice(s,n.index),isCode:!1}),t.push({text:n[0],isCode:!0}),s=n.index+n[0].length;return s<i.length&&t.push({text:i.slice(s),isCode:!1}),t}function br(i,t=3e3){if(i.length<=t)return[i];let e=[],s=i;for(;s.length>t;){let n=s.slice(0,t),a=fc(n)%2===1,r;if(a){let o=gc(n);if(o>0)r=o;else{e.push(n+"\n```"),s="```\n"+s.slice(t);continue}}else{if(r=n.lastIndexOf(`
11926
+ `)}async openOrCreateThread(t){if(this.isThread)throw new Error("Nested threads are not supported.");let e=this.threads.get(t);if(e)return e;let s=this.messages.findIndex(i=>i.id===t);if(s<0)throw new Error(`Thread anchor message "${t}" not found in parent.`);let n=new r(this.agent,this.settings,this.repository,this.vault,{threadAnchorId:t,parentSession:this,mcpAuth:this.mcpAuth});return n.threadAnchorIndex=s,await n.loadPersistedState()||(n.threadAnchorIndex=s),this.threads.set(t,n),n}closeThread(t){let e=this.threads.get(t);e&&(e.abort(),this.threads.delete(t))}hibernateIdleThreads(t){let e=Date.now();for(let s of this.threads.values())s.isProcessAlive&&e-s.lastActiveAt>t&&s.hibernate()}parseStreamEvent(t){let e=t.type;if(e==="assistant"){let s=t.message;if(s?.content&&Array.isArray(s.content))for(let n of s.content){if(n.type==="text"&&typeof n.text=="string")return{type:"text",content:n.text};if(n.type==="tool_use"){let a=String(n.name??"tool"),i=n.input,o=i?.command??i?.content??i?.file_path??i?.path??"";return{type:"tool_use",content:o?String(o).slice(0,150):"",toolName:a}}}}if(e==="content_block_delta"){let s=t.delta;if(s?.type==="text_delta"&&typeof s.text=="string")return{type:"text",content:s.text}}return null}};var vn=class{constructor(t){this.config=t;this.now=t.now??Date.now}buckets=new Map;now;tryConsume(t){let e=this.now(),s=e-this.config.windowMs,a=(this.buckets.get(t)??[]).filter(i=>i>s);return a.length>=this.config.maxPerWindow?(this.buckets.set(t,a),!1):(a.push(e),this.buckets.set(t,a),!0)}currentCount(t){let s=this.now()-this.config.windowMs;return(this.buckets.get(t)??[]).filter(a=>a>s).length}reset(t){this.buckets.delete(t)}resetAll(){this.buckets.clear()}};function Ts(r){let t=pc(r),e=[];for(let s of t)if(s.kind==="code"){let n=s.text.replace(/^```[^\n]*\n/,"```\n");e.push(n)}else e.push(mc(s.text));return e.join("")}function pc(r){let t=[],e=0,s=!1,n=0;for(;e<r.length;)r.startsWith("```",e)?s?(e+=3,r[e]===`
11927
+ `&&(e+=1),t.push({kind:"code",text:r.slice(n,e)}),n=e,s=!1):(e>n&&t.push({kind:"prose",text:r.slice(n,e)}),n=e,s=!0,e+=3):e+=1;return n<r.length&&t.push({kind:s?"code":"prose",text:r.slice(n)}),t}function mc(r){return fc(r).map(s=>{if(s.isCode)return s.text;let n=s.text;return n=n.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;"),n=n.replace(/\[([^\]]+)\]\(([^)]+)\)/g,(a,i,o)=>`<${o}|${i}>`),n=n.replace(/\*\*([^*]+)\*\*/g,"*$1*"),n=n.replace(/^#{1,6}\s+(.+)$/gm,"*$1*"),n}).join("")}function fc(r){let t=[],e=/`([^`\n]+)`/g,s=0,n;for(;n=e.exec(r);)n.index>s&&t.push({text:r.slice(s,n.index),isCode:!1}),t.push({text:n[0],isCode:!0}),s=n.index+n[0].length;return s<r.length&&t.push({text:r.slice(s),isCode:!1}),t}function ki(r,t=3e3){if(r.length<=t)return[r];let e=[],s=r;for(;s.length>t;){let n=s.slice(0,t),a=gc(n)%2===1,i;if(a){let o=yc(n);if(o>0)i=o;else{e.push(n+"\n```"),s="```\n"+s.slice(t);continue}}else{if(i=n.lastIndexOf(`
12017
11928
 
12018
- `),r<t/2){let o=n.lastIndexOf(`
12019
- `);o>t/2?r=o:r=t}r<=0&&(r=t)}e.push(s.slice(0,r)),s=s.slice(r).replace(/^\n+/,"")}return s.length>0&&e.push(s),e}function fc(i){let t=0,e=0;for(;(e=i.indexOf("```",e))!==-1;)t+=1,e+=3;return t}function gc(i){let t=0,e=0,s=0;for(;s<i.length;){let n=i.indexOf("```",s);if(n===-1)break;t+=1,s=n+3,t%2===0&&(e=s)}return e}function yc(i,t){let e=i.trim(),s=e.toLowerCase();for(let n of t){let a=n.toLowerCase(),r=[`use ${a}:`,`use ${a}`,`@${a}:`,`@${a}`,`${a}:`];for(let o of r)if(s.startsWith(o)){let c=e.slice(o.length).trim();return{agent:n,rest:c}}}return null}var wn=class{constructor(t){this.deps=t;this.now=t.now??(()=>Date.now());let e=t.getSettings();this.rateLimiter=new vn({maxPerWindow:Math.max(1,e.channelRateLimitPerConversation),windowMs:Math.max(1e3,e.channelRateLimitWindowMinutes*6e4),now:this.now})}adapters=new Map;adapterConfigs=new Map;sessions=new Map;conversationLocks=new Map;rateLimiter;metrics=new Map;statusListeners=new Set;adapterUnsubscribes=new Map;now;threadBindings=new Map;hibernationInterval=null;started=!1;async start(t){if(!this.started){this.started=!0;for(let e of t.channels)await this.bringUpChannel(e,t);this.hibernationInterval=window.setInterval(()=>{this.runHibernationSweep()},6e4)}}async stop(){if(!this.started)return;this.started=!1,this.hibernationInterval&&(window.clearInterval(this.hibernationInterval),this.hibernationInterval=null);let t=Array.from(this.adapters.values());await Promise.all(t.map(async e=>{try{await e.stop()}catch(s){console.warn(`Agent Fleet: channel adapter ${e.config.name} stop() failed`,s)}})),this.adapters.clear(),this.adapterConfigs.clear();for(let e of this.adapterUnsubscribes.values())for(let s of e)s();this.adapterUnsubscribes.clear(),this.conversationLocks.size>0&&await Promise.allSettled(Array.from(this.conversationLocks.values()));for(let e of this.sessions.values())try{e.session.isStreaming?e.session.abort():e.session.hibernate()}catch{}this.sessions.clear(),this.conversationLocks.clear(),this.conversationLockGen.clear(),this.rateLimiter.resetAll()}async reconcile(t){if(!this.started)return;let e=new Map;for(let s of t.channels)e.set(s.name,s);for(let[s,n]of this.adapters){let a=e.get(s),r=a&&this.isChannelRuntimeValid(a,t);if(!a||!a.enabled||!r){await this.tearDownChannel(s);continue}let o=this.adapterConfigs.get(s);o&&this.requiresRestart(o,a)?(await this.tearDownChannel(s),await this.bringUpChannel(a,t)):(this.adapterConfigs.set(s,a),n.config=a)}for(let[s,n]of e)!this.adapters.has(s)&&n.enabled&&this.isChannelRuntimeValid(n,t)&&await this.bringUpChannel(n,t);this.notifyStatusListeners()}getCredentials(){return this.deps.getChannelCredentials?.()??this.deps.getSettings().channelCredentials??{}}isChannelRuntimeValid(t,e){let s=e.agents.find(a=>a.name===t.defaultAgent);if(!s||s.approvalRequired.length>0)return!1;let n=this.getCredentials()[t.credentialRef];return!(!n||n.type!==t.type)}async bringUpChannel(t,e){if(!t.enabled||!this.isChannelRuntimeValid(t,e))return;let s=this.getCredentials()[t.credentialRef];if(!s)return;let n;try{n=this.deps.adapterFactory(t,s)}catch(r){console.error(`Agent Fleet: failed to build adapter for channel ${t.name}`,r);return}let a=[];a.push(n.onInbound(r=>{this.handleInbound(n,r)})),a.push(n.onStatusChange(()=>this.notifyStatusListeners())),n.onAgentSwitch&&a.push(n.onAgentSwitch((r,o,c)=>{let l=`${t.name}:${r}`;this.threadBindings.set(l,o),this.persistBindings(t.name)})),n.setAllowedAgentsResolver?.(()=>this.resolveAllowedAgents(t)),this.adapters.set(t.name,n),this.adapterConfigs.set(t.name,t),this.adapterUnsubscribes.set(t.name,a),this.ensureMetrics(t.name),await this.loadBindings(t.name);try{await n.start()}catch(r){console.error(`Agent Fleet: channel adapter ${t.name} start() failed`,r)}this.notifyStatusListeners()}async tearDownChannel(t){let e=this.adapters.get(t);if(!e)return;try{await e.stop()}catch(a){console.warn(`Agent Fleet: channel adapter ${t} stop() failed`,a)}let s=this.adapterUnsubscribes.get(t);if(s)for(let a of s)a();this.adapterUnsubscribes.delete(t),this.adapters.delete(t),this.adapterConfigs.delete(t);let n=`${t}:`;for(let[a,r]of this.sessions)if(a.startsWith(n)){try{r.session.isStreaming?r.session.abort():r.session.hibernate()}catch{}this.sessions.delete(a)}}requiresRestart(t,e){return t.type!==e.type||t.credentialRef!==e.credentialRef||JSON.stringify(t.transport)!==JSON.stringify(e.transport)}async handleInbound(t,e){let s=t.config,n=`${s.name}:${e.conversationId}`;if(!(s.allowedUsers.length>0&&(!e.externalUserId||!s.allowedUsers.includes(e.externalUserId)))){if(!this.rateLimiter.tryConsume(n)){console.warn(`Agent Fleet: rate-limited message from ${e.externalUserId} on ${s.name} (conversation ${e.conversationId})`);try{await t.send(e.conversationId,"_Rate limit exceeded. Please slow down and try again in a few minutes._")}catch(a){console.warn(`Agent Fleet: rate-limit reply failed on ${s.name}`,a)}return}await this.withConversationLock(n,async()=>{let a=this.ensureMetrics(s.name);a.messagesReceived+=1,a.lastMessageAt=this.now();let r=this.resolveAllowedAgents(s),o=yc(e.text,r),c,l;if(o){c=o.agent,l=o.rest;let d=this.threadBindings.get(n);if(this.threadBindings.set(n,c),this.persistBindings(s.name),d!==c)try{await t.setThreadTitle?.(e.conversationId,c)}catch{}if(!l){try{await t.send(e.conversationId,`_Now chatting with *${c}*. Send your next message to start._`)}catch{}return}}else{let d=`${s.name}:${e.conversationId.replace(/:thread:[^:]+$/,`:user:${e.externalUserId}`)}`;c=this.threadBindings.get(n)??this.threadBindings.get(d)??s.defaultAgent,l=e.text}try{await t.setTyping(e.conversationId,!0)}catch{}if(e.images&&e.images.length>0){let d=await this.saveInboundImages(e.images);d&&(l=d+(l||"Please analyze this image."))}let h="";try{let u=await(await this.getOrCreateSession(s,e.conversationId,c)).sendMessage(l,()=>{});if(h=u.text.trim(),u.toolCalls.length>0){let p=vc(u.toolCalls);p&&(h+=`
11929
+ `),i<t/2){let o=n.lastIndexOf(`
11930
+ `);o>t/2?i=o:i=t}i<=0&&(i=t)}e.push(s.slice(0,i)),s=s.slice(i).replace(/^\n+/,"")}return s.length>0&&e.push(s),e}function gc(r){let t=0,e=0;for(;(e=r.indexOf("```",e))!==-1;)t+=1,e+=3;return t}function yc(r){let t=0,e=0,s=0;for(;s<r.length;){let n=r.indexOf("```",s);if(n===-1)break;t+=1,s=n+3,t%2===0&&(e=s)}return e}function vc(r,t){let e=r.trim(),s=e.toLowerCase();for(let n of t){let a=n.toLowerCase(),i=[`use ${a}:`,`use ${a}`,`@${a}:`,`@${a}`,`${a}:`];for(let o of i)if(s.startsWith(o)){let c=e.slice(o.length).trim();return{agent:n,rest:c}}}return null}var wn=class{constructor(t){this.deps=t;this.now=t.now??(()=>Date.now());let e=t.getSettings();this.rateLimiter=new vn({maxPerWindow:Math.max(1,e.channelRateLimitPerConversation),windowMs:Math.max(1e3,e.channelRateLimitWindowMinutes*6e4),now:this.now})}adapters=new Map;adapterConfigs=new Map;sessions=new Map;conversationLocks=new Map;rateLimiter;metrics=new Map;statusListeners=new Set;adapterUnsubscribes=new Map;now;threadBindings=new Map;activeTurns=new Map;turnTails=new Map;hibernationInterval=null;started=!1;async start(t){if(!this.started){this.started=!0;for(let e of t.channels)await this.bringUpChannel(e,t);this.hibernationInterval=window.setInterval(()=>{this.runHibernationSweep()},6e4)}}async stop(){if(!this.started)return;this.started=!1,this.hibernationInterval&&(window.clearInterval(this.hibernationInterval),this.hibernationInterval=null);let t=Array.from(this.adapters.values());await Promise.all(t.map(async e=>{try{await e.stop()}catch(s){console.warn(`Agent Fleet: channel adapter ${e.config.name} stop() failed`,s)}})),this.adapters.clear(),this.adapterConfigs.clear();for(let e of this.adapterUnsubscribes.values())for(let s of e)s();this.adapterUnsubscribes.clear(),this.conversationLocks.size>0&&await Promise.allSettled(Array.from(this.conversationLocks.values()));for(let e of this.sessions.values())try{e.session.isStreaming?e.session.abort():e.session.hibernate()}catch{}this.sessions.clear(),this.conversationLocks.clear(),this.conversationLockGen.clear(),this.rateLimiter.resetAll()}async reconcile(t){if(!this.started)return;let e=new Map;for(let s of t.channels)e.set(s.name,s);for(let[s,n]of this.adapters){let a=e.get(s),i=a&&this.isChannelRuntimeValid(a,t);if(!a||!a.enabled||!i){await this.tearDownChannel(s);continue}let o=this.adapterConfigs.get(s);o&&this.requiresRestart(o,a)?(await this.tearDownChannel(s),await this.bringUpChannel(a,t)):(this.adapterConfigs.set(s,a),n.config=a)}for(let[s,n]of e)!this.adapters.has(s)&&n.enabled&&this.isChannelRuntimeValid(n,t)&&await this.bringUpChannel(n,t);this.notifyStatusListeners()}getCredentials(){return this.deps.getChannelCredentials?.()??this.deps.getSettings().channelCredentials??{}}isChannelRuntimeValid(t,e){let s=e.agents.find(a=>a.name===t.defaultAgent);if(!s||s.approvalRequired.length>0)return!1;let n=this.getCredentials()[t.credentialRef];return!(!n||n.type!==t.type)}async bringUpChannel(t,e){if(!t.enabled||!this.isChannelRuntimeValid(t,e))return;let s=this.getCredentials()[t.credentialRef];if(!s)return;let n;try{n=this.deps.adapterFactory(t,s)}catch(i){console.error(`Agent Fleet: failed to build adapter for channel ${t.name}`,i);return}let a=[];a.push(n.onInbound(i=>{this.handleInbound(n,i)})),a.push(n.onStatusChange(()=>this.notifyStatusListeners())),n.onAgentSwitch&&a.push(n.onAgentSwitch((i,o,c)=>{let l=`${t.name}:${i}`;this.threadBindings.set(l,o),this.persistBindings(t.name)})),n.setAllowedAgentsResolver?.(()=>this.resolveAllowedAgents(t)),this.adapters.set(t.name,n),this.adapterConfigs.set(t.name,t),this.adapterUnsubscribes.set(t.name,a),this.ensureMetrics(t.name),await this.loadBindings(t.name);try{await n.start()}catch(i){console.error(`Agent Fleet: channel adapter ${t.name} start() failed`,i)}this.notifyStatusListeners()}async tearDownChannel(t){let e=this.adapters.get(t);if(!e)return;try{await e.stop()}catch(a){console.warn(`Agent Fleet: channel adapter ${t} stop() failed`,a)}let s=this.adapterUnsubscribes.get(t);if(s)for(let a of s)a();this.adapterUnsubscribes.delete(t),this.adapters.delete(t),this.adapterConfigs.delete(t);let n=`${t}:`;for(let[a,i]of this.sessions)if(a.startsWith(n)){try{i.session.isStreaming?i.session.abort():i.session.hibernate()}catch{}this.sessions.delete(a)}}requiresRestart(t,e){return t.type!==e.type||t.credentialRef!==e.credentialRef||JSON.stringify(t.transport)!==JSON.stringify(e.transport)}async handleInbound(t,e){let s=t.config,n=`${s.name}:${e.conversationId}`;if(!(s.allowedUsers.length>0&&(!e.externalUserId||!s.allowedUsers.includes(e.externalUserId)))){if(!this.rateLimiter.tryConsume(n)){console.warn(`Agent Fleet: rate-limited message from ${e.externalUserId} on ${s.name} (conversation ${e.conversationId})`);try{await t.send(e.conversationId,"_Rate limit exceeded. Please slow down and try again in a few minutes._")}catch(a){console.warn(`Agent Fleet: rate-limit reply failed on ${s.name}`,a)}return}await this.withConversationLock(n,async()=>{let a=this.ensureMetrics(s.name);a.messagesReceived+=1,a.lastMessageAt=this.now();let i=this.resolveAllowedAgents(s),o=vc(e.text,i),c,l;if(o){c=o.agent,l=o.rest;let m=this.threadBindings.get(n);if(this.threadBindings.set(n,c),this.persistBindings(s.name),m!==c)try{await t.setThreadTitle?.(e.conversationId,c)}catch{}if(!l){try{await t.send(e.conversationId,`_Now chatting with *${c}*. Send your next message to start._`)}catch{}return}}else{let m=`${s.name}:${e.conversationId.replace(/:thread:[^:]+$/,`:user:${e.externalUserId}`)}`;c=this.threadBindings.get(n)??this.threadBindings.get(m)??s.defaultAgent,l=e.text}if(e.images&&e.images.length>0){let m=await this.saveInboundImages(e.images);m&&(l=m+(l||"Please analyze this image."))}let h=this.activeTurns.get(n);if(h&&h.agent===c&&h.session.isStreaming){h.session.injectMessage(l);return}let d=await this.getOrCreateSession(s,e.conversationId,c),p=(this.turnTails.get(n)??Promise.resolve()).catch(()=>{}).then(()=>(this.activeTurns.set(n,{agent:c,session:d}),this.runChannelTurn(t,s,e.conversationId,c,d,l).finally(()=>{this.activeTurns.delete(n)})));this.turnTails.set(n,p),p.finally(()=>{this.turnTails.get(n)===p&&this.turnTails.delete(n)})})}}async runChannelTurn(t,e,s,n,a,i){let o=this.ensureMetrics(e.name),c=e.allowedAgents.length>1||e.allowedAgents.length===0&&this.resolveAllowedAgents(e).length>1;try{await t.setTyping(s,!0)}catch{}let l=Promise.resolve(),h=m=>{l=l.then(async()=>{try{await this.deliverReply(t,s,m),o.messagesSent+=1}catch(f){console.error(`Agent Fleet: reply delivery failed on ${e.name}`,f)}})},d="",u=[],p=()=>{let m=At(d).trim();if(u.length>0){let f=wc(u);f&&(m+=`${m?`
12020
11931
 
12021
- _${p}_`)}}catch(d){console.error(`Agent Fleet: channel turn failed on ${s.name}/${e.conversationId}`,d),h=`_Sorry \u2014 the agent run failed. ${d instanceof Error?d.message:String(d)}_`}try{h&&((s.allowedAgents.length>1||s.allowedAgents.length===0&&this.resolveAllowedAgents(s).length>1)&&(h=`*[${c}]*
12022
- ${h}`),await this.deliverReply(t,e.conversationId,h),a.messagesSent+=1)}catch(d){console.error(`Agent Fleet: reply delivery failed on ${s.name}`,d)}finally{try{await t.setTyping(e.conversationId,!1)}catch{}}this.enforceHardCap()})}}async deliverReply(t,e,s){let n=br(s);for(let a of n)await t.send(e,a)}async saveInboundImages(t){let s=`${this.deps.getSettings().fleetFolder}/chat-images`,n=[];for(let a of t)try{this.deps.vault.getAbstractFileByPath((0,ut.normalizePath)(s))||await this.deps.vault.createFolder((0,ut.normalizePath)(s));let r=(0,ut.normalizePath)(`${s}/${a.filename}`),o=r;if(this.deps.vault.getAbstractFileByPath(r)){let d=a.filename.lastIndexOf("."),u=d>0?a.filename.slice(0,d):a.filename,p=d>0?a.filename.slice(d):"";o=(0,ut.normalizePath)(`${s}/${u}_${Date.now()}${p}`)}await this.deps.vault.createBinary(o,a.data);let l=this.deps.vault.adapter.basePath??"",h=l?`${l}/${o}`:o;n.push(`### Image: ${a.filename}
11932
+ `:""}_${f}_`)}d="",u=[],m&&(c&&(m=`*[${n}]*
11933
+ ${m}`),h(m))};try{await a.sendMessage(i,m=>{if(m.type==="text")d+=m.content;else if(m.type==="tool_use"&&m.toolName)u.push({name:m.toolName});else if(m.type==="error"){let f=m.errorMessage?.trim()||"the agent run failed";h(`_Sorry \u2014 ${f}_`)}else m.type==="result"&&p()})}catch(m){console.error(`Agent Fleet: channel turn failed on ${e.name}/${s}`,m),h(`_Sorry \u2014 the agent run failed. ${m instanceof Error?m.message:String(m)}_`)}finally{await l;try{await t.setTyping(s,!1)}catch{}this.enforceHardCap()}}async deliverReply(t,e,s){let n=ki(s);for(let a of n)await t.send(e,a)}async saveInboundImages(t){let s=`${this.deps.getSettings().fleetFolder}/chat-images`,n=[];for(let a of t)try{this.deps.vault.getAbstractFileByPath((0,ut.normalizePath)(s))||await this.deps.vault.createFolder((0,ut.normalizePath)(s));let i=(0,ut.normalizePath)(`${s}/${a.filename}`),o=i;if(this.deps.vault.getAbstractFileByPath(i)){let d=a.filename.lastIndexOf("."),u=d>0?a.filename.slice(0,d):a.filename,p=d>0?a.filename.slice(d):"";o=(0,ut.normalizePath)(`${s}/${u}_${Date.now()}${p}`)}await this.deps.vault.createBinary(o,a.data);let l=this.deps.vault.adapter.basePath??"",h=l?`${l}/${o}`:o;n.push(`### Image: ${a.filename}
12023
11934
  The image file is located at: ${h}
12024
- Please read and analyze this image.`)}catch(r){console.warn("Agent Fleet: failed to save inbound image",a.filename,r)}return n.length===0?"":`## Attached Images
11935
+ Please read and analyze this image.`)}catch(i){console.warn("Agent Fleet: failed to save inbound image",a.filename,i)}return n.length===0?"":`## Attached Images
12025
11936
 
12026
11937
  ${n.join(`
12027
11938
 
@@ -12029,26 +11940,26 @@ ${n.join(`
12029
11940
 
12030
11941
  ---
12031
11942
 
12032
- `}async getOrCreateSession(t,e,s){let n=`${t.name}:${e}:${s}`,a=this.sessions.get(n);if(a)return a.session;let r=this.deps.getRepository(),o=r.getAgentByName(s);if(!o)throw new Error(`Channel ${t.name} bound to missing agent ${s}`);let c=new Vt(o,this.deps.getSettings(),r,this.deps.vault,{channelName:t.name,conversationId:`${e}:${s}`,channelContext:t.channelContext||void 0,mcpAuth:this.deps.getMcpAuth?.()});this.deps.recordUsage&&c.setUsageRecorder(this.deps.recordUsage);try{await c.loadPersistedState()}catch{}return this.sessions.set(n,{session:c,channelName:t.name,conversationId:e,sessionKey:n}),c}resolveAllowedAgents(t){let e=new Set(this.deps.getRepository().getSnapshot().agents.filter(n=>n.enabled).map(n=>n.name));return(t.allowedAgents.length>0?t.allowedAgents:[...e]).filter(n=>e.has(n))}async persistBindings(t){let e=`${t}:`,s={};for(let[o,c]of this.threadBindings)o.startsWith(e)&&(s[o.slice(e.length)]=c);let n=this.deps.getSettings(),a=(0,ut.normalizePath)(`${n.fleetFolder}/channels/${t}/bindings.json`),r=JSON.stringify(s,null,2);try{let o=this.deps.vault.getAbstractFileByPath(a);if(o instanceof ut.TFile)await this.deps.vault.modify(o,r);else{let c=a.slice(0,a.lastIndexOf("/"));if(!this.deps.vault.getAbstractFileByPath(c))try{await this.deps.vault.createFolder(c)}catch{}await this.deps.vault.create(a,r)}}catch(o){console.warn(`Agent Fleet: failed to persist thread bindings for ${t}`,o)}}async loadBindings(t){let e=this.deps.getSettings(),s=(0,ut.normalizePath)(`${e.fleetFolder}/channels/${t}/bindings.json`);try{let n=this.deps.vault.getAbstractFileByPath(s);if(!(n instanceof ut.TFile))return;let a=await this.deps.vault.cachedRead(n),r=JSON.parse(a);for(let[o,c]of Object.entries(r))typeof c=="string"&&this.threadBindings.set(`${t}:${o}`,c)}catch{}}getThreadAgent(t,e){return this.threadBindings.get(`${t}:${e}`)}enforceHardCap(){let t=Math.max(1,this.deps.getSettings().maxConcurrentChannelSessions),e=Array.from(this.sessions.values()).filter(n=>n.session.isProcessAlive&&!n.session.isStreaming);if(e.length<=t)return;e.sort((n,a)=>n.session.lastActiveAt-a.session.lastActiveAt);let s=e.length-t;for(let n=0;n<s;n+=1){let a=e[n];if(!a)break;try{a.session.hibernate()}catch{}}}async runHibernationSweep(){let t=Math.max(6e4,this.deps.getSettings().channelIdleTimeoutMinutes*6e4),e=this.now()-t;for(let s of this.sessions.values())if(s.session.isProcessAlive&&!s.session.isStreaming&&s.session.lastActiveAt<e)try{s.session.hibernate()}catch{}}async broadcastToChannel(t,e){let s=this.adapters.get(t);if(!s){console.warn(`Agent Fleet: broadcastToChannel \u2014 no adapter for channel ${t}`);return}if(!s.broadcast){console.warn(`Agent Fleet: broadcastToChannel \u2014 adapter ${t} does not support broadcast`);return}await s.broadcast(e)}async postToChannelTarget(t,e,s){let n=this.adapters.get(t);if(!n){console.warn(`Agent Fleet: postToChannelTarget \u2014 no adapter for channel ${t}`);return}if(!n.sendToTarget){console.warn(`Agent Fleet: postToChannelTarget \u2014 adapter ${t} does not support sendToTarget`);return}await n.sendToTarget(e,s)}conversationLockGen=new Map;async withConversationLock(t,e){let s=this.conversationLocks.get(t)??Promise.resolve(),n,a=new Promise(o=>{n=o}),r=(this.conversationLockGen.get(t)??0)+1;this.conversationLockGen.set(t,r),this.conversationLocks.set(t,s.then(()=>a));try{await s,await e()}finally{n(),this.conversationLockGen.get(t)===r&&(this.conversationLocks.delete(t),this.conversationLockGen.delete(t))}}ensureMetrics(t){let e=this.metrics.get(t);return e||(e={messagesReceived:0,messagesSent:0,lastMessageAt:null},this.metrics.set(t,e)),e}getMetrics(t){return{...this.ensureMetrics(t)}}getChannelStatus(t){let e=this.adapters.get(t);return e?e.getStatus():"disabled"}getConnectedCount(){let t=0;for(let e of this.adapters.values())e.getStatus()==="connected"&&(t+=1);return t}getSessionCount(t){let e=0,s=`${t}:`;for(let n of this.sessions.keys())n.startsWith(s)&&(e+=1);return e}onStatusChange(t){return this.statusListeners.add(t),()=>this.statusListeners.delete(t)}notifyStatusListeners(){for(let t of this.statusListeners)try{t()}catch{}}};function vc(i){if(i.length===0)return"";let t=new Map;for(let s of i)t.set(s.name,(t.get(s.name)??0)+1);let e=[];for(let[s,n]of t)e.push(n>1?`${s}\xD7${n}`:s);return`Used ${i.length} tool${i.length===1?"":"s"}: ${e.join(", ")}`}var bn=class{credentials=new Map;secretStore;persistCallback;setSecretStore(t){this.secretStore=t}loadCredentials(t){if(this.credentials.clear(),this.secretStore?.available){let e=this.secretStore.listByPrefix(Gt);for(let s of e){let n=Gt+"-",a=s.startsWith(n)?s.slice(n.length):s,r=this.secretStore.getJson(Gt,a);if(r){let o=r._ref??a,c={...r};delete c._ref,this.credentials.set(o,c)}}}if(this.credentials.size===0&&t){for(let[e,s]of Object.entries(t))this.credentials.set(e,s);this.secretStore?.available&&this.credentials.size>0&&this.persistToSecretStore()}}onChanged(t){this.persistCallback=t}get(t){return this.credentials.get(t)}set(t,e){this.credentials.set(t,e),this.persist()}delete(t){this.credentials.delete(t),this.secretStore?.delete(Gt,t),this.persist()}list(){return Array.from(this.credentials.entries()).map(([t,e])=>({ref:t,entry:e}))}toRecord(){let t={};for(let[e,s]of this.credentials.entries())t[e]=s;return t}persist(){this.persistToSecretStore(),this.persistCallback&&this.persistCallback(this.toRecord())}persistToSecretStore(){if(this.secretStore?.available)for(let[t,e]of this.credentials)this.secretStore.setJson(Gt,t,{...e,_ref:t})}};function xr(){return{status:"disconnected",scope:"user",tools:[],toolDetails:[]}}function wc(i,t,e){if(e)return"stdio";let s=(i??"").toLowerCase();return s==="sse"?"sse":s==="http"||s==="streamable-http"||s==="streamable_http"?"http":t?.endsWith("/sse")?"sse":"http"}function Sr(i){let t=[],e={},s;try{s=JSON.parse(i)}catch{return{servers:t,tokens:e}}let n=s.mcpServers;if(!n||typeof n!="object")return{servers:t,tokens:e};for(let[a,r]of Object.entries(n)){if(!r||typeof r!="object")continue;let o=r,c=typeof o.command=="string"?o.command:void 0,l=typeof o.url=="string"?o.url:void 0;if(!c&&!l)continue;let h=wc(typeof o.type=="string"?o.type:void 0,l,!!c),d={name:a,type:h,enabled:!0,source:"imported",...xr()};if(h==="stdio")d.command=c,Array.isArray(o.args)&&(d.args=o.args.filter(u=>typeof u=="string")),o.env&&typeof o.env=="object"&&(d.env=kr(o.env));else{d.url=l;let u=o.headers&&typeof o.headers=="object"?kr(o.headers):{},p=u.Authorization??u.authorization;p?.startsWith("Bearer ")?(e[a]=p.slice(7),delete u.Authorization,delete u.authorization,d.auth="oauth"):d.auth="none",Object.keys(u).length>0&&(d.headers=u)}t.push(d)}return{servers:t,tokens:e}}function Cr(i){let t=new Map,e=null;for(let n of i.split(/\r?\n/)){let a=n.trim();if(!a||a.startsWith("#"))continue;let r=a.match(/^\[(.+)\]$/);if(r){let u=r[1].trim().match(/^mcp_servers\.(.+)$/);if(!u){e=null;continue}let p=u[1],m=null;p.endsWith(".env")?(m="env",p=p.slice(0,-4)):p.endsWith(".oauth")&&(m="oauth",p=p.slice(0,-6));let f=kn(p);if(!f){e=null;continue}t.has(f)||t.set(f,{name:f,enabled:!0}),e={name:f,sub:m};continue}if(!e)continue;let o=a.match(/^([^=]+?)\s*=\s*(.+)$/);if(!o)continue;let c=kn(o[1].trim()),l=o[2].trim(),h=t.get(e.name);if(e.sub==="env")h.env??={},h.env[c]=Yt(l);else if(e.sub==="oauth")c==="client_id"&&(h.oauthClientId=Yt(l));else switch(c){case"command":h.command=Yt(l);break;case"args":h.args=bc(l);break;case"url":h.url=Yt(l);break;case"bearer_token_env_var":h.bearerEnvVar=Yt(l);break;case"oauth_resource":h.oauthResource=Yt(l);break;case"enabled":h.enabled=l==="true";break;default:break}}let s=[];for(let n of t.values()){if(!n.command&&!n.url)continue;let a=n.command?"stdio":n.url?.endsWith("/sse")?"sse":"http",r={name:n.name,type:a,enabled:n.enabled,source:"imported",...xr()};a==="stdio"?(r.command=n.command,n.args&&n.args.length>0&&(r.args=n.args),n.env&&Object.keys(n.env).length>0&&(r.env=n.env)):(r.url=n.url,n.oauthClientId||n.oauthResource?(r.auth="oauth",r.oauth={clientId:n.oauthClientId,resource:n.oauthResource}):n.bearerEnvVar?(r.auth="bearer",r.envSecretKeys=[n.bearerEnvVar]):r.auth="none"),s.push(r)}return{servers:s,tokens:{}}}function Tr(...i){let t=new Map,e={};for(let s of i){for(let n of s.servers){let a=n.name.trim().toLowerCase();t.has(a)||t.set(a,n)}Object.assign(e,s.tokens)}return{servers:[...t.values()],tokens:e}}function kn(i){let t=i.trim();return t.startsWith('"')&&t.endsWith('"')||t.startsWith("'")&&t.endsWith("'")?t.slice(1,-1):t}function Yt(i){let t=i.trim();if(t.startsWith('"')||t.startsWith("'")){let e=t[0],s=t.indexOf(e,1);if(s>0)return t.slice(1,s)}return kn(t.replace(/\s+#.*$/,""))}function bc(i){let t=i.trim();try{let s=JSON.parse(t);if(Array.isArray(s))return s.filter(n=>typeof n=="string")}catch{}return t.replace(/^\[/,"").replace(/\]$/,"").split(",").map(s=>kn(s.trim())).filter(Boolean)}function kr(i){let t={};for(let[e,s]of Object.entries(i))typeof s=="string"&&(t[e]=s);return t}var jd=Ye(ko(),1),Wd=Ye(Rn(),1),Hd=Ye(Xt(),1),qd=Ye(La(),1),zd=Ye(Na(),1),Gd=Ye(Ha(),1),Ro=Ye(Ln(),1),Vd=Ye(Po(),1);var ns=Ro.default;var Do=require("obsidian");var Yd="https://slack.com/api";function Kd(i){let t=i.team??"unknown",e=i.channel??"unknown",s=i.thread_ts??i.ts??"unknown";return`slack:${t}:${e}:thread:${s}`}function Jd(i){let t=i.split(":");return t.length>=3&&t[0]==="slack"?t[2]??null:null}function Xd(i){let t=i.split(":");if(t[3]==="thread"&&t[4])return t[4]}var On=class{type="slack";config;credential;ws=null;status="stopped";stopping=!1;backoffMs=1e3;reconnectTimer=null;inboundHandlers=new Set;statusHandlers=new Set;agentSwitchHandlers=new Set;allowedAgentsResolver;sendQueues=new Map;threadContext=new Map;constructor(t,e){if(e.type!=="slack")throw new Error(`SlackAdapter requires a slack credential, got ${e.type}`);this.config=t,this.credential=e}async start(){this.stopping=!1,await this.connect()}async stop(){if(this.stopping=!0,this.reconnectTimer&&(window.clearTimeout(this.reconnectTimer),this.reconnectTimer=null),this.ws){try{this.ws.close()}catch{}this.ws=null}this.threadContext.clear(),this.sendQueues.clear(),this.setStatus("stopped")}getStatus(){return this.status}async send(t,e){let s=Jd(t);if(!s){console.warn(`Agent Fleet: could not extract channel id from ${t}`);return}let n=Xd(t),a=Ts(e);await this.enqueueSend(s,async()=>{await this.slackApi("chat.postMessage",{channel:s,text:a,...n?{thread_ts:n}:{}})})}async sendToTarget(t,e){if(!t)return;let s=Ts(e);await this.enqueueSend(t,async()=>{await this.slackApi("chat.postMessage",{channel:t,text:s})})}async broadcast(t){let e=this.config.allowedUsers[0];if(!e){console.warn(`Agent Fleet: broadcast on ${this.config.name} skipped \u2014 no allowed users configured`);return}try{let n=(await this.slackApi("conversations.open",{users:e})).channel?.id;if(!n){console.warn(`Agent Fleet: broadcast \u2014 conversations.open returned no channel for user ${e}`);return}let a=Ts(t);await this.slackApi("chat.postMessage",{channel:n,text:a})}catch(s){console.error(`Agent Fleet: broadcast failed on ${this.config.name}`,s)}}async setThreadTitle(t,e){let s=this.threadContext.get(t);if(s)try{await this.slackApi("assistant.threads.setTitle",{channel_id:s.channelId,thread_ts:s.threadTs,title:e})}catch(n){console.warn(`Agent Fleet: assistant.threads.setTitle failed on ${this.config.name}`,n)}}async setTyping(t,e){let s=this.threadContext.get(t);if(s)try{await this.slackApi("assistant.threads.setStatus",{channel_id:s.channelId,thread_ts:s.threadTs,status:e?"is thinking...":""})}catch(n){console.warn(`Agent Fleet: assistant.threads.setStatus (${e?"on":"off"}) failed on ${this.config.name}`,n)}}onInbound(t){return this.inboundHandlers.add(t),()=>this.inboundHandlers.delete(t)}onStatusChange(t){return this.statusHandlers.add(t),()=>this.statusHandlers.delete(t)}setAllowedAgentsResolver(t){this.allowedAgentsResolver=t}onAgentSwitch(t){return this.agentSwitchHandlers.add(t),()=>this.agentSwitchHandlers.delete(t)}async connect(){if(this.stopping)return;this.setStatus(this.ws?"reconnecting":"connecting");let t;try{let e=await this.slackApi("apps.connections.open",{},{useAppToken:!0});if(!e.ok||!e.url)throw new Error(e.error??"apps.connections.open returned no URL");t=e.url}catch(e){console.error(`Agent Fleet: Slack apps.connections.open failed for channel ${this.config.name}`,e),this.setStatus("needs-auth"),this.scheduleReconnect();return}try{let e=new ns(t);e.on("open",()=>{}),e.on("message",a=>{this.handleSocketData(a)}),e.on("error",a=>{console.warn(`Agent Fleet: Slack WebSocket error on ${this.config.name}`,a),this.setStatus("error")}),e.on("close",()=>{this.ws=null,this.stopping||this.scheduleReconnect()}),this.ws=e;let s=window.setTimeout(()=>{if(this.status==="connecting"||this.status==="reconnecting"){console.warn(`Agent Fleet: Slack WebSocket connect timeout on ${this.config.name}`);try{e.close()}catch{}}},3e4);e.on("close",()=>window.clearTimeout(s));let n=this.onStatusChange(a=>{a==="connected"&&(window.clearTimeout(s),n())})}catch(e){console.error("Agent Fleet: Slack WebSocket open failed",e),this.setStatus("error"),this.scheduleReconnect();return}}handleSocketData(t){let e;try{e=JSON.parse(t.toString())}catch{return}if(e.type==="hello"){this.backoffMs=1e3,this.setStatus("connected");return}if(e.type==="disconnect"){try{this.ws?.close()}catch{}return}if(e.type==="events_api"&&e.envelope_id){this.ackEnvelope(e.envelope_id),this.routeEventPayload(e.payload);return}if(e.type==="slash_commands"&&e.envelope_id){this.ackEnvelope(e.envelope_id),this.handleSlashCommand(e.payload);return}if(e.type==="interactive"&&e.envelope_id){this.ackEnvelope(e.envelope_id),this.handleInteraction(e.payload);return}e.envelope_id&&this.ackEnvelope(e.envelope_id)}async handleSlashCommand(t){if(!t)return;let e=t.command,s=t.channel_id,n=t.user_id;if(!(!e||!s||!n)){if(e==="/agents"){let a=this.allowedAgentsResolver?this.allowedAgentsResolver():this.config.allowedAgents;if(a.length===0){await this.slackApi("chat.postEphemeral",{channel:s,user:n,text:"No agents available. Add existing, enabled agents to `allowed_agents` in the channel file."});return}let r=a.map(c=>({type:"button",text:{type:"plain_text",text:c===this.config.defaultAgent?`${c} \u2713`:c,emoji:!0},action_id:`switch_agent_${c}`,value:c})),o=[{type:"section",text:{type:"mrkdwn",text:"*Select an agent to chat with:*"}}];for(let c=0;c<r.length;c+=5)o.push({type:"actions",elements:r.slice(c,c+5)});try{await this.slackApi("chat.postEphemeral",{channel:s,user:n,text:"Select an agent",blocks:o})}catch(c){console.error("Agent Fleet: /agents response failed",c)}return}try{await this.slackApi("chat.postEphemeral",{channel:s,user:n,text:`Unknown command: ${e}`})}catch(a){console.error(`Agent Fleet: slash command response failed for ${e}`,a)}}}async handleInteraction(t){if(!t||t.type!=="block_actions")return;let s=t.actions,n=t.user,a=t.channel;if(!s?.length||!n||!a)return;let r=s[0];if(!r)return;let o=r.action_id,c=r.value;if(!o?.startsWith("switch_agent_")||!c)return;let l=n.id,h=a.id;if(!l||!h)return;let u=`slack:${t.team?.id??"unknown"}:${h}:user:${l}`;for(let p of this.agentSwitchHandlers)try{p(u,c,l)}catch(m){console.error("Agent Fleet: agent switch handler threw",m)}try{await this.slackApi("chat.postEphemeral",{channel:h,user:l,text:`Switched to *${c}*. Send your next message to start.`})}catch(p){console.warn("Agent Fleet: agent switch confirmation failed",p)}try{await this.setThreadTitle(u,c)}catch{}}ackEnvelope(t){if(!(!this.ws||this.ws.readyState!==ns.OPEN))try{this.ws.send(JSON.stringify({envelope_id:t}))}catch(e){console.warn("Agent Fleet: Slack envelope ACK failed",e)}}routeEventPayload(t){if(!t)return;let e=t.event;if(!e)return;let s=e.type;if(s==="assistant_thread_started"){let h=e.assistant_thread;h?.channel_id&&h.thread_ts&&console.debug(`Agent Fleet: assistant thread started on ${this.config.name} (channel=${h.channel_id}, thread_ts=${h.thread_ts}, user=${h.user_id})`);return}if(s==="assistant_thread_context_changed")return;let n=e,a=s==="message"&&n.subtype===void 0,r=s==="app_mention";if(!a&&!r||n.bot_id||!n.user||!n.text||r&&(n.text=n.text.replace(/^<@[A-Z0-9]+>\s*/,"").trim(),!n.text))return;let o=Kd(n),c=n.thread_ts??n.ts;if(n.channel&&c&&(this.threadContext.set(o,{channelId:n.channel,threadTs:c}),this.threadContext.size>500)){let d=this.threadContext.keys().next();d.done||this.threadContext.delete(d.value)}let l={conversationId:o,externalUserId:n.user,text:n.text,timestamp:new Date().toISOString(),meta:{slack_channel:n.channel,slack_ts:n.ts,thread_ts:n.thread_ts}};for(let h of this.inboundHandlers)try{h(l)}catch(d){console.error("Agent Fleet: Slack inbound handler threw",d)}}scheduleReconnect(){if(this.stopping||this.reconnectTimer)return;let t=this.backoffMs;this.backoffMs=Math.min(3e4,this.backoffMs*2),console.warn(`Agent Fleet: Slack channel ${this.config.name} scheduling reconnect in ${t}ms`),this.reconnectTimer=window.setTimeout(()=>{this.reconnectTimer=null,!this.stopping&&this.connect()},t)}setStatus(t){if(this.status!==t){this.status=t;for(let e of this.statusHandlers)try{e(t)}catch{}}}async slackApi(t,e,s={}){let n=s.useAppToken?this.credential.appToken:this.credential.botToken,a=`${Yd}/${t}`,r=await(0,Do.requestUrl)({url:a,method:"POST",contentType:"application/json; charset=utf-8",headers:{Authorization:`Bearer ${n}`},body:JSON.stringify(e),throw:!1});if(r.status===429){let c=Number(r.headers["retry-after"]??"1");return await new Promise(l=>window.setTimeout(l,Math.max(1e3,c*1e3))),this.slackApi(t,e,s)}if(r.status<200||r.status>=300)throw new Error(`Slack ${t} HTTP ${r.status}`);let o=r.json;if(o.ok===!1)throw new Error(`Slack ${t} error: ${o.error??"unknown"}`);return o}async enqueueSend(t,e){let n=(this.sendQueues.get(t)??Promise.resolve()).then(async()=>{try{await e()}finally{await new Promise(r=>window.setTimeout(r,1e3))}}),a=n.catch(r=>{console.warn(`Agent Fleet: Slack send queue error for ${t}`,r)});this.sendQueues.set(t,a),await n,this.sendQueues.get(t)===a&&this.sendQueues.delete(t)}};var Ga=require("obsidian"),Io="https://api.telegram.org",Nn=class{type="telegram";config;credential;status="stopped";stopping=!1;pollOffset=0;pollTimer=null;backoffMs=1e3;typingIntervals=new Map;pollAbort=null;inboundHandlers=new Set;statusHandlers=new Set;agentSwitchHandlers=new Set;allowedAgentsResolver;constructor(t,e){if(e.type!=="telegram")throw new Error(`TelegramAdapter requires a telegram credential, got ${e.type}`);this.config=t,this.credential=e}async start(){this.stopping=!1;try{let e=(await this.tgApi("getUpdates",{offset:-1,limit:1})).result?.[0];e&&(this.pollOffset=e.update_id+1)}catch{}try{await this.tgApi("setMyCommands",{commands:[{command:"agents",description:"List available agents"}]})}catch{}this.setStatus("connected"),this.poll()}async stop(){this.stopping=!0,this.pollTimer&&(window.clearTimeout(this.pollTimer),this.pollTimer=null),this.pollAbort?.abort(),this.pollAbort=null;for(let[,t]of this.typingIntervals)window.clearInterval(t);this.typingIntervals.clear(),this.setStatus("stopped")}getStatus(){return this.status}async send(t,e){let s=Mo(t);if(!s)return;let n=Lo(t),a=za(e,4096);for(let r of a)await this.tgApi("sendMessage",{chat_id:s,text:r,parse_mode:"Markdown",...n?{message_thread_id:Number(n)}:{}})}async sendToTarget(t,e){if(!t)return;let s=za(e,4096);for(let n of s)await this.tgApi("sendMessage",{chat_id:t,text:n,parse_mode:"Markdown"})}async setTyping(t,e){let s=Mo(t);if(!s)return;let n=Lo(t),a={chat_id:s,action:"typing"};if(n&&(a.message_thread_id=Number(n)),e){let r=this.typingIntervals.get(t);r&&window.clearInterval(r);try{await this.tgApi("sendChatAction",a)}catch(c){console.warn("Agent Fleet: Telegram sendChatAction failed",c)}let o=window.setInterval(()=>{this.tgApi("sendChatAction",a).catch(()=>{})},4500);this.typingIntervals.set(t,o)}else{let r=this.typingIntervals.get(t);r&&(window.clearInterval(r),this.typingIntervals.delete(t))}}async setThreadTitle(t,e){}async broadcast(t){let e=this.config.allowedUsers[0];if(e)try{let s=za(t,4096);for(let n of s)await this.tgApi("sendMessage",{chat_id:e,text:n,parse_mode:"Markdown"})}catch(s){console.error(`Agent Fleet: Telegram broadcast failed on ${this.config.name}`,s)}}onInbound(t){return this.inboundHandlers.add(t),()=>this.inboundHandlers.delete(t)}onStatusChange(t){return this.statusHandlers.add(t),()=>this.statusHandlers.delete(t)}setAllowedAgentsResolver(t){this.allowedAgentsResolver=t}pickerAgents(){return this.allowedAgentsResolver?this.allowedAgentsResolver():this.config.allowedAgents}onAgentSwitch(t){return this.agentSwitchHandlers.add(t),()=>this.agentSwitchHandlers.delete(t)}poll(){this.stopping||(async()=>{try{this.pollAbort=new AbortController;let t=await this.tgApi("getUpdates",{offset:this.pollOffset,timeout:30,allowed_updates:["message","callback_query"]},this.pollAbort.signal);if(t.ok&&t.result)for(let e of t.result){if(this.pollOffset=e.update_id+1,e.callback_query){this.handleCallbackQuery(e.callback_query);continue}e.message&&this.routeMessage(e.message)}this.backoffMs=1e3,this.status!=="connected"&&this.setStatus("connected")}catch(t){if(t instanceof DOMException&&t.name==="AbortError")return;if(console.warn(`Agent Fleet: Telegram poll failed on ${this.config.name}`,t),this.status!=="error"&&this.status!=="needs-auth"){let e=t instanceof Error?t.message:String(t);this.setStatus(e.includes("401")||e.includes("Unauthorized")?"needs-auth":"error")}await new Promise(e=>window.setTimeout(e,this.backoffMs)),this.backoffMs=Math.min(3e4,this.backoffMs*2)}this.stopping||(this.pollTimer=window.setTimeout(()=>this.poll(),100))})()}routeMessage(t){let e=t.photo&&t.photo.length>0,s=t.document&&this.isImageMime(t.document.mime_type),n=!!t.text;if(!t.from||!n&&!e&&!s)return;let a=t.text??t.caption??"";if(a==="/agents"||a.startsWith("/agents@")){this.handleAgentsCommand(t);return}if(a.startsWith("/"))return;let r=String(t.from.id),o=String(t.chat.id),c=t.message_thread_id?String(t.message_thread_id):void 0,l=c?`tg:${o}:topic:${c}`:`tg:${o}`;this.buildAndEmitMessage(t,a,l,r,o,c)}async buildAndEmitMessage(t,e,s,n,a,r){let o=[];try{if(t.photo&&t.photo.length>0){let l=t.photo[t.photo.length-1],h=await this.downloadFile(l.file_id,`photo_${t.message_id}.jpg`,"image/jpeg");h&&o.push(h)}if(t.document&&this.isImageMime(t.document.mime_type)){let l=t.document.file_name??`doc_${t.message_id}`,h=t.document.mime_type??"image/jpeg",d=await this.downloadFile(t.document.file_id,l,h);d&&o.push(d)}}catch(l){console.warn("Agent Fleet: Telegram image download failed",l)}let c={conversationId:s,externalUserId:n,text:e,timestamp:new Date(t.date*1e3).toISOString(),meta:{telegram_chat_id:a,telegram_message_id:t.message_id,telegram_thread_id:r,chat_type:t.chat.type},...o.length>0?{images:o}:{}};for(let l of this.inboundHandlers)try{l(c)}catch(h){console.error("Agent Fleet: Telegram inbound handler threw",h)}}async downloadFile(t,e,s){let a=(await this.tgApi("getFile",{file_id:t})).result?.file_path;if(!a)return null;let r=`${Io}/file/bot${this.credential.botToken}/${a}`;return{data:(await(0,Ga.requestUrl)({url:r,method:"GET"})).arrayBuffer,filename:e,mimeType:s}}isImageMime(t){return t?t==="image/jpeg"||t==="image/png"||t==="image/gif"||t==="image/webp"||t==="image/bmp"||t==="image/tiff":!1}async handleAgentsCommand(t){let e=String(t.chat.id),s=t.message_thread_id,n=this.pickerAgents();if(n.length===0){await this.tgApi("sendMessage",{chat_id:e,text:"No agents available. Add existing, enabled agents to `allowed_agents` in the channel file.",...s?{message_thread_id:s}:{}});return}let a=n.map(r=>[{text:r===this.config.defaultAgent?`${r} \u2713`:r,callback_data:`switch:${r}`}]);await this.tgApi("sendMessage",{chat_id:e,text:"*Select an agent to chat with:*",parse_mode:"Markdown",reply_markup:{inline_keyboard:a},...s?{message_thread_id:s}:{}})}async handleCallbackQuery(t){let e=t.data;if(!e?.startsWith("switch:")){await this.tgApi("answerCallbackQuery",{callback_query_id:t.id});return}let s=e.slice(7),n=String(t.from.id),a=String(t.message?.chat?.id??t.from.id),r=t.message?.message_thread_id?String(t.message.message_thread_id):void 0,o=r?`tg:${a}:topic:${r}`:`tg:${a}`;for(let c of this.agentSwitchHandlers)try{c(o,s,n)}catch(l){console.error("Agent Fleet: Telegram agent switch handler threw",l)}if(await this.tgApi("answerCallbackQuery",{callback_query_id:t.id,text:`Switched to ${s}`}),t.message){let l=this.pickerAgents().map(h=>[{text:h===s?`${h} \u2713`:h,callback_data:`switch:${h}`}]);try{await this.tgApi("editMessageText",{chat_id:a,message_id:t.message.message_id,text:`*Active agent: ${s}*`,parse_mode:"Markdown",reply_markup:{inline_keyboard:l}})}catch{}}}async tgApi(t,e,s){if(s?.aborted)throw new DOMException("Aborted","AbortError");let n=`${Io}/bot${this.credential.botToken}/${t}`,a=(0,Ga.requestUrl)({url:n,method:"POST",contentType:"application/json",body:JSON.stringify(e),throw:!1}),r;if(s?r=await Promise.race([a,new Promise((o,c)=>{s.addEventListener("abort",()=>c(new DOMException("Aborted","AbortError")),{once:!0})})]):r=await a,r.status===401||r.status===403)throw new Error(`Telegram ${t} ${r.status} Unauthorized`);if(r.status===429){let c=r.json?.parameters?.retry_after??1;return await new Promise(l=>window.setTimeout(l,c*1e3)),this.tgApi(t,e)}if(r.status<200||r.status>=300)throw new Error(`Telegram ${t} HTTP ${r.status}: ${r.text}`);return r.json}setStatus(t){if(this.status!==t){this.status=t;for(let e of this.statusHandlers)try{e(t)}catch{}}}};function Mo(i){return i.startsWith("tg:")?i.split(":")[1]??null:null}function Lo(i){let t=i.split(":");if(t[2]==="topic"&&t[3])return t[3]}function za(i,t){if(i.length<=t)return[i];let e=[],s=i;for(;s.length>t;){let n=s.lastIndexOf(`
11943
+ `}async getOrCreateSession(t,e,s){let n=`${t.name}:${e}:${s}`,a=this.sessions.get(n);if(a)return a.session;let i=this.deps.getRepository(),o=i.getAgentByName(s);if(!o)throw new Error(`Channel ${t.name} bound to missing agent ${s}`);let c=new Yt(o,this.deps.getSettings(),i,this.deps.vault,{channelName:t.name,conversationId:`${e}:${s}`,channelContext:t.channelContext||void 0,mcpAuth:this.deps.getMcpAuth?.()});this.deps.recordUsage&&c.setUsageRecorder(this.deps.recordUsage);try{await c.loadPersistedState()}catch{}return this.sessions.set(n,{session:c,channelName:t.name,conversationId:e,sessionKey:n}),c}resolveAllowedAgents(t){let e=new Set(this.deps.getRepository().getSnapshot().agents.filter(n=>n.enabled).map(n=>n.name));return(t.allowedAgents.length>0?t.allowedAgents:[...e]).filter(n=>e.has(n))}async persistBindings(t){let e=`${t}:`,s={};for(let[o,c]of this.threadBindings)o.startsWith(e)&&(s[o.slice(e.length)]=c);let n=this.deps.getSettings(),a=(0,ut.normalizePath)(`${n.fleetFolder}/channels/${t}/bindings.json`),i=JSON.stringify(s,null,2);try{let o=this.deps.vault.getAbstractFileByPath(a);if(o instanceof ut.TFile)await this.deps.vault.modify(o,i);else{let c=a.slice(0,a.lastIndexOf("/"));if(!this.deps.vault.getAbstractFileByPath(c))try{await this.deps.vault.createFolder(c)}catch{}await this.deps.vault.create(a,i)}}catch(o){console.warn(`Agent Fleet: failed to persist thread bindings for ${t}`,o)}}async loadBindings(t){let e=this.deps.getSettings(),s=(0,ut.normalizePath)(`${e.fleetFolder}/channels/${t}/bindings.json`);try{let n=this.deps.vault.getAbstractFileByPath(s);if(!(n instanceof ut.TFile))return;let a=await this.deps.vault.cachedRead(n),i=JSON.parse(a);for(let[o,c]of Object.entries(i))typeof c=="string"&&this.threadBindings.set(`${t}:${o}`,c)}catch{}}getThreadAgent(t,e){return this.threadBindings.get(`${t}:${e}`)}enforceHardCap(){let t=Math.max(1,this.deps.getSettings().maxConcurrentChannelSessions),e=Array.from(this.sessions.values()).filter(n=>n.session.isProcessAlive&&!n.session.isStreaming);if(e.length<=t)return;e.sort((n,a)=>n.session.lastActiveAt-a.session.lastActiveAt);let s=e.length-t;for(let n=0;n<s;n+=1){let a=e[n];if(!a)break;try{a.session.hibernate()}catch{}}}async runHibernationSweep(){let t=Math.max(6e4,this.deps.getSettings().channelIdleTimeoutMinutes*6e4),e=this.now()-t;for(let s of this.sessions.values())if(s.session.isProcessAlive&&!s.session.isStreaming&&s.session.lastActiveAt<e)try{s.session.hibernate()}catch{}}async broadcastToChannel(t,e){let s=this.adapters.get(t);if(!s){console.warn(`Agent Fleet: broadcastToChannel \u2014 no adapter for channel ${t}`);return}if(!s.broadcast){console.warn(`Agent Fleet: broadcastToChannel \u2014 adapter ${t} does not support broadcast`);return}await s.broadcast(e)}async postToChannelTarget(t,e,s){let n=this.adapters.get(t);if(!n){console.warn(`Agent Fleet: postToChannelTarget \u2014 no adapter for channel ${t}`);return}if(!n.sendToTarget){console.warn(`Agent Fleet: postToChannelTarget \u2014 adapter ${t} does not support sendToTarget`);return}await n.sendToTarget(e,s)}conversationLockGen=new Map;async withConversationLock(t,e){let s=this.conversationLocks.get(t)??Promise.resolve(),n,a=new Promise(o=>{n=o}),i=(this.conversationLockGen.get(t)??0)+1;this.conversationLockGen.set(t,i),this.conversationLocks.set(t,s.then(()=>a));try{await s,await e()}finally{n(),this.conversationLockGen.get(t)===i&&(this.conversationLocks.delete(t),this.conversationLockGen.delete(t))}}ensureMetrics(t){let e=this.metrics.get(t);return e||(e={messagesReceived:0,messagesSent:0,lastMessageAt:null},this.metrics.set(t,e)),e}getMetrics(t){return{...this.ensureMetrics(t)}}getChannelStatus(t){let e=this.adapters.get(t);return e?e.getStatus():"disabled"}getConnectedCount(){let t=0;for(let e of this.adapters.values())e.getStatus()==="connected"&&(t+=1);return t}getSessionCount(t){let e=0,s=`${t}:`;for(let n of this.sessions.keys())n.startsWith(s)&&(e+=1);return e}onStatusChange(t){return this.statusListeners.add(t),()=>this.statusListeners.delete(t)}notifyStatusListeners(){for(let t of this.statusListeners)try{t()}catch{}}};function wc(r){if(r.length===0)return"";let t=new Map;for(let s of r)t.set(s.name,(t.get(s.name)??0)+1);let e=[];for(let[s,n]of t)e.push(n>1?`${s}\xD7${n}`:s);return`Used ${r.length} tool${r.length===1?"":"s"}: ${e.join(", ")}`}var bn=class{credentials=new Map;secretStore;persistCallback;setSecretStore(t){this.secretStore=t}loadCredentials(t){if(this.credentials.clear(),this.secretStore?.available){let e=this.secretStore.listByPrefix(Vt);for(let s of e){let n=Vt+"-",a=s.startsWith(n)?s.slice(n.length):s,i=this.secretStore.getJson(Vt,a);if(i){let o=i._ref??a,c={...i};delete c._ref,this.credentials.set(o,c)}}}if(this.credentials.size===0&&t){for(let[e,s]of Object.entries(t))this.credentials.set(e,s);this.secretStore?.available&&this.credentials.size>0&&this.persistToSecretStore()}}onChanged(t){this.persistCallback=t}get(t){return this.credentials.get(t)}set(t,e){this.credentials.set(t,e),this.persist()}delete(t){this.credentials.delete(t),this.secretStore?.delete(Vt,t),this.persist()}list(){return Array.from(this.credentials.entries()).map(([t,e])=>({ref:t,entry:e}))}toRecord(){let t={};for(let[e,s]of this.credentials.entries())t[e]=s;return t}persist(){this.persistToSecretStore(),this.persistCallback&&this.persistCallback(this.toRecord())}persistToSecretStore(){if(this.secretStore?.available)for(let[t,e]of this.credentials)this.secretStore.setJson(Vt,t,{...e,_ref:t})}};function Si(){return{status:"disconnected",scope:"user",tools:[],toolDetails:[]}}function bc(r,t,e){if(e)return"stdio";let s=(r??"").toLowerCase();return s==="sse"?"sse":s==="http"||s==="streamable-http"||s==="streamable_http"?"http":t?.endsWith("/sse")?"sse":"http"}function Ci(r){let t=[],e={},s;try{s=JSON.parse(r)}catch{return{servers:t,tokens:e}}let n=s.mcpServers;if(!n||typeof n!="object")return{servers:t,tokens:e};for(let[a,i]of Object.entries(n)){if(!i||typeof i!="object")continue;let o=i,c=typeof o.command=="string"?o.command:void 0,l=typeof o.url=="string"?o.url:void 0;if(!c&&!l)continue;let h=bc(typeof o.type=="string"?o.type:void 0,l,!!c),d={name:a,type:h,enabled:!0,source:"imported",...Si()};if(h==="stdio")d.command=c,Array.isArray(o.args)&&(d.args=o.args.filter(u=>typeof u=="string")),o.env&&typeof o.env=="object"&&(d.env=xi(o.env));else{d.url=l;let u=o.headers&&typeof o.headers=="object"?xi(o.headers):{},p=u.Authorization??u.authorization;p?.startsWith("Bearer ")?(e[a]=p.slice(7),delete u.Authorization,delete u.authorization,d.auth="oauth"):d.auth="none",Object.keys(u).length>0&&(d.headers=u)}t.push(d)}return{servers:t,tokens:e}}function Ti(r){let t=new Map,e=null;for(let n of r.split(/\r?\n/)){let a=n.trim();if(!a||a.startsWith("#"))continue;let i=a.match(/^\[(.+)\]$/);if(i){let u=i[1].trim().match(/^mcp_servers\.(.+)$/);if(!u){e=null;continue}let p=u[1],m=null;p.endsWith(".env")?(m="env",p=p.slice(0,-4)):p.endsWith(".oauth")&&(m="oauth",p=p.slice(0,-6));let f=kn(p);if(!f){e=null;continue}t.has(f)||t.set(f,{name:f,enabled:!0}),e={name:f,sub:m};continue}if(!e)continue;let o=a.match(/^([^=]+?)\s*=\s*(.+)$/);if(!o)continue;let c=kn(o[1].trim()),l=o[2].trim(),h=t.get(e.name);if(e.sub==="env")h.env??={},h.env[c]=Kt(l);else if(e.sub==="oauth")c==="client_id"&&(h.oauthClientId=Kt(l));else switch(c){case"command":h.command=Kt(l);break;case"args":h.args=kc(l);break;case"url":h.url=Kt(l);break;case"bearer_token_env_var":h.bearerEnvVar=Kt(l);break;case"oauth_resource":h.oauthResource=Kt(l);break;case"enabled":h.enabled=l==="true";break;default:break}}let s=[];for(let n of t.values()){if(!n.command&&!n.url)continue;let a=n.command?"stdio":n.url?.endsWith("/sse")?"sse":"http",i={name:n.name,type:a,enabled:n.enabled,source:"imported",...Si()};a==="stdio"?(i.command=n.command,n.args&&n.args.length>0&&(i.args=n.args),n.env&&Object.keys(n.env).length>0&&(i.env=n.env)):(i.url=n.url,n.oauthClientId||n.oauthResource?(i.auth="oauth",i.oauth={clientId:n.oauthClientId,resource:n.oauthResource}):n.bearerEnvVar?(i.auth="bearer",i.envSecretKeys=[n.bearerEnvVar]):i.auth="none"),s.push(i)}return{servers:s,tokens:{}}}function _i(...r){let t=new Map,e={};for(let s of r){for(let n of s.servers){let a=n.name.trim().toLowerCase();t.has(a)||t.set(a,n)}Object.assign(e,s.tokens)}return{servers:[...t.values()],tokens:e}}function kn(r){let t=r.trim();return t.startsWith('"')&&t.endsWith('"')||t.startsWith("'")&&t.endsWith("'")?t.slice(1,-1):t}function Kt(r){let t=r.trim();if(t.startsWith('"')||t.startsWith("'")){let e=t[0],s=t.indexOf(e,1);if(s>0)return t.slice(1,s)}return kn(t.replace(/\s+#.*$/,""))}function kc(r){let t=r.trim();try{let s=JSON.parse(t);if(Array.isArray(s))return s.filter(n=>typeof n=="string")}catch{}return t.replace(/^\[/,"").replace(/\]$/,"").split(",").map(s=>kn(s.trim())).filter(Boolean)}function xi(r){let t={};for(let[e,s]of Object.entries(r))typeof s=="string"&&(t[e]=s);return t}var Wd=Ye(xo(),1),Hd=Ye(Rn(),1),qd=Ye(Qt(),1),zd=Ye(La(),1),Gd=Ye(Na(),1),Vd=Ye(Ha(),1),Do=Ye(Ln(),1),Yd=Ye(Ro(),1);var as=Do.default;var Io=require("obsidian");var Kd="https://slack.com/api";function Jd(r){let t=r.team??"unknown",e=r.channel??"unknown",s=r.thread_ts??r.ts??"unknown";return`slack:${t}:${e}:thread:${s}`}function Xd(r){let t=r.split(":");return t.length>=3&&t[0]==="slack"?t[2]??null:null}function Qd(r){let t=r.split(":");if(t[3]==="thread"&&t[4])return t[4]}var On=class{type="slack";config;credential;ws=null;status="stopped";stopping=!1;backoffMs=1e3;reconnectTimer=null;inboundHandlers=new Set;statusHandlers=new Set;agentSwitchHandlers=new Set;allowedAgentsResolver;sendQueues=new Map;threadContext=new Map;constructor(t,e){if(e.type!=="slack")throw new Error(`SlackAdapter requires a slack credential, got ${e.type}`);this.config=t,this.credential=e}async start(){this.stopping=!1,await this.connect()}async stop(){if(this.stopping=!0,this.reconnectTimer&&(window.clearTimeout(this.reconnectTimer),this.reconnectTimer=null),this.ws){try{this.ws.close()}catch{}this.ws=null}this.threadContext.clear(),this.sendQueues.clear(),this.setStatus("stopped")}getStatus(){return this.status}async send(t,e){let s=Xd(t);if(!s){console.warn(`Agent Fleet: could not extract channel id from ${t}`);return}let n=Qd(t),a=Ts(e);await this.enqueueSend(s,async()=>{await this.slackApi("chat.postMessage",{channel:s,text:a,...n?{thread_ts:n}:{}})})}async sendToTarget(t,e){if(!t)return;let s=Ts(e);await this.enqueueSend(t,async()=>{await this.slackApi("chat.postMessage",{channel:t,text:s})})}async broadcast(t){let e=this.config.allowedUsers[0];if(!e){console.warn(`Agent Fleet: broadcast on ${this.config.name} skipped \u2014 no allowed users configured`);return}try{let n=(await this.slackApi("conversations.open",{users:e})).channel?.id;if(!n){console.warn(`Agent Fleet: broadcast \u2014 conversations.open returned no channel for user ${e}`);return}let a=Ts(t);await this.slackApi("chat.postMessage",{channel:n,text:a})}catch(s){console.error(`Agent Fleet: broadcast failed on ${this.config.name}`,s)}}async setThreadTitle(t,e){let s=this.threadContext.get(t);if(s)try{await this.slackApi("assistant.threads.setTitle",{channel_id:s.channelId,thread_ts:s.threadTs,title:e})}catch(n){console.warn(`Agent Fleet: assistant.threads.setTitle failed on ${this.config.name}`,n)}}async setTyping(t,e){let s=this.threadContext.get(t);if(s)try{await this.slackApi("assistant.threads.setStatus",{channel_id:s.channelId,thread_ts:s.threadTs,status:e?"is thinking...":""})}catch(n){console.warn(`Agent Fleet: assistant.threads.setStatus (${e?"on":"off"}) failed on ${this.config.name}`,n)}}onInbound(t){return this.inboundHandlers.add(t),()=>this.inboundHandlers.delete(t)}onStatusChange(t){return this.statusHandlers.add(t),()=>this.statusHandlers.delete(t)}setAllowedAgentsResolver(t){this.allowedAgentsResolver=t}onAgentSwitch(t){return this.agentSwitchHandlers.add(t),()=>this.agentSwitchHandlers.delete(t)}async connect(){if(this.stopping)return;this.setStatus(this.ws?"reconnecting":"connecting");let t;try{let e=await this.slackApi("apps.connections.open",{},{useAppToken:!0});if(!e.ok||!e.url)throw new Error(e.error??"apps.connections.open returned no URL");t=e.url}catch(e){console.error(`Agent Fleet: Slack apps.connections.open failed for channel ${this.config.name}`,e),this.setStatus("needs-auth"),this.scheduleReconnect();return}try{let e=new as(t);e.on("open",()=>{}),e.on("message",a=>{this.handleSocketData(a)}),e.on("error",a=>{console.warn(`Agent Fleet: Slack WebSocket error on ${this.config.name}`,a),this.setStatus("error")}),e.on("close",()=>{this.ws=null,this.stopping||this.scheduleReconnect()}),this.ws=e;let s=window.setTimeout(()=>{if(this.status==="connecting"||this.status==="reconnecting"){console.warn(`Agent Fleet: Slack WebSocket connect timeout on ${this.config.name}`);try{e.close()}catch{}}},3e4);e.on("close",()=>window.clearTimeout(s));let n=this.onStatusChange(a=>{a==="connected"&&(window.clearTimeout(s),n())})}catch(e){console.error("Agent Fleet: Slack WebSocket open failed",e),this.setStatus("error"),this.scheduleReconnect();return}}handleSocketData(t){let e;try{e=JSON.parse(t.toString())}catch{return}if(e.type==="hello"){this.backoffMs=1e3,this.setStatus("connected");return}if(e.type==="disconnect"){try{this.ws?.close()}catch{}return}if(e.type==="events_api"&&e.envelope_id){this.ackEnvelope(e.envelope_id),this.routeEventPayload(e.payload);return}if(e.type==="slash_commands"&&e.envelope_id){this.ackEnvelope(e.envelope_id),this.handleSlashCommand(e.payload);return}if(e.type==="interactive"&&e.envelope_id){this.ackEnvelope(e.envelope_id),this.handleInteraction(e.payload);return}e.envelope_id&&this.ackEnvelope(e.envelope_id)}async handleSlashCommand(t){if(!t)return;let e=t.command,s=t.channel_id,n=t.user_id;if(!(!e||!s||!n)){if(e==="/agents"){let a=this.allowedAgentsResolver?this.allowedAgentsResolver():this.config.allowedAgents;if(a.length===0){await this.slackApi("chat.postEphemeral",{channel:s,user:n,text:"No agents available. Add existing, enabled agents to `allowed_agents` in the channel file."});return}let i=a.map(c=>({type:"button",text:{type:"plain_text",text:c===this.config.defaultAgent?`${c} \u2713`:c,emoji:!0},action_id:`switch_agent_${c}`,value:c})),o=[{type:"section",text:{type:"mrkdwn",text:"*Select an agent to chat with:*"}}];for(let c=0;c<i.length;c+=5)o.push({type:"actions",elements:i.slice(c,c+5)});try{await this.slackApi("chat.postEphemeral",{channel:s,user:n,text:"Select an agent",blocks:o})}catch(c){console.error("Agent Fleet: /agents response failed",c)}return}try{await this.slackApi("chat.postEphemeral",{channel:s,user:n,text:`Unknown command: ${e}`})}catch(a){console.error(`Agent Fleet: slash command response failed for ${e}`,a)}}}async handleInteraction(t){if(!t||t.type!=="block_actions")return;let s=t.actions,n=t.user,a=t.channel;if(!s?.length||!n||!a)return;let i=s[0];if(!i)return;let o=i.action_id,c=i.value;if(!o?.startsWith("switch_agent_")||!c)return;let l=n.id,h=a.id;if(!l||!h)return;let u=`slack:${t.team?.id??"unknown"}:${h}:user:${l}`;for(let p of this.agentSwitchHandlers)try{p(u,c,l)}catch(m){console.error("Agent Fleet: agent switch handler threw",m)}try{await this.slackApi("chat.postEphemeral",{channel:h,user:l,text:`Switched to *${c}*. Send your next message to start.`})}catch(p){console.warn("Agent Fleet: agent switch confirmation failed",p)}try{await this.setThreadTitle(u,c)}catch{}}ackEnvelope(t){if(!(!this.ws||this.ws.readyState!==as.OPEN))try{this.ws.send(JSON.stringify({envelope_id:t}))}catch(e){console.warn("Agent Fleet: Slack envelope ACK failed",e)}}routeEventPayload(t){if(!t)return;let e=t.event;if(!e)return;let s=e.type;if(s==="assistant_thread_started"){let h=e.assistant_thread;h?.channel_id&&h.thread_ts&&console.debug(`Agent Fleet: assistant thread started on ${this.config.name} (channel=${h.channel_id}, thread_ts=${h.thread_ts}, user=${h.user_id})`);return}if(s==="assistant_thread_context_changed")return;let n=e,a=s==="message"&&n.subtype===void 0,i=s==="app_mention";if(!a&&!i||n.bot_id||!n.user||!n.text||i&&(n.text=n.text.replace(/^<@[A-Z0-9]+>\s*/,"").trim(),!n.text))return;let o=Jd(n),c=n.thread_ts??n.ts;if(n.channel&&c&&(this.threadContext.set(o,{channelId:n.channel,threadTs:c}),this.threadContext.size>500)){let d=this.threadContext.keys().next();d.done||this.threadContext.delete(d.value)}let l={conversationId:o,externalUserId:n.user,text:n.text,timestamp:new Date().toISOString(),meta:{slack_channel:n.channel,slack_ts:n.ts,thread_ts:n.thread_ts}};for(let h of this.inboundHandlers)try{h(l)}catch(d){console.error("Agent Fleet: Slack inbound handler threw",d)}}scheduleReconnect(){if(this.stopping||this.reconnectTimer)return;let t=this.backoffMs;this.backoffMs=Math.min(3e4,this.backoffMs*2),console.warn(`Agent Fleet: Slack channel ${this.config.name} scheduling reconnect in ${t}ms`),this.reconnectTimer=window.setTimeout(()=>{this.reconnectTimer=null,!this.stopping&&this.connect()},t)}setStatus(t){if(this.status!==t){this.status=t;for(let e of this.statusHandlers)try{e(t)}catch{}}}async slackApi(t,e,s={}){let n=s.useAppToken?this.credential.appToken:this.credential.botToken,a=`${Kd}/${t}`,i=await(0,Io.requestUrl)({url:a,method:"POST",contentType:"application/json; charset=utf-8",headers:{Authorization:`Bearer ${n}`},body:JSON.stringify(e),throw:!1});if(i.status===429){let c=Number(i.headers["retry-after"]??"1");return await new Promise(l=>window.setTimeout(l,Math.max(1e3,c*1e3))),this.slackApi(t,e,s)}if(i.status<200||i.status>=300)throw new Error(`Slack ${t} HTTP ${i.status}`);let o=i.json;if(o.ok===!1)throw new Error(`Slack ${t} error: ${o.error??"unknown"}`);return o}async enqueueSend(t,e){let n=(this.sendQueues.get(t)??Promise.resolve()).then(async()=>{try{await e()}finally{await new Promise(i=>window.setTimeout(i,1e3))}}),a=n.catch(i=>{console.warn(`Agent Fleet: Slack send queue error for ${t}`,i)});this.sendQueues.set(t,a),await n,this.sendQueues.get(t)===a&&this.sendQueues.delete(t)}};var Ga=require("obsidian"),Mo="https://api.telegram.org",Nn=class{type="telegram";config;credential;status="stopped";stopping=!1;pollOffset=0;pollTimer=null;backoffMs=1e3;typingIntervals=new Map;pollAbort=null;inboundHandlers=new Set;statusHandlers=new Set;agentSwitchHandlers=new Set;allowedAgentsResolver;constructor(t,e){if(e.type!=="telegram")throw new Error(`TelegramAdapter requires a telegram credential, got ${e.type}`);this.config=t,this.credential=e}async start(){this.stopping=!1;try{let e=(await this.tgApi("getUpdates",{offset:-1,limit:1})).result?.[0];e&&(this.pollOffset=e.update_id+1)}catch{}try{await this.tgApi("setMyCommands",{commands:[{command:"agents",description:"List available agents"}]})}catch{}this.setStatus("connected"),this.poll()}async stop(){this.stopping=!0,this.pollTimer&&(window.clearTimeout(this.pollTimer),this.pollTimer=null),this.pollAbort?.abort(),this.pollAbort=null;for(let[,t]of this.typingIntervals)window.clearInterval(t);this.typingIntervals.clear(),this.setStatus("stopped")}getStatus(){return this.status}async send(t,e){let s=Lo(t);if(!s)return;let n=Fo(t),a=za(e,4096);for(let i of a)await this.tgApi("sendMessage",{chat_id:s,text:i,parse_mode:"Markdown",...n?{message_thread_id:Number(n)}:{}})}async sendToTarget(t,e){if(!t)return;let s=za(e,4096);for(let n of s)await this.tgApi("sendMessage",{chat_id:t,text:n,parse_mode:"Markdown"})}async setTyping(t,e){let s=Lo(t);if(!s)return;let n=Fo(t),a={chat_id:s,action:"typing"};if(n&&(a.message_thread_id=Number(n)),e){let i=this.typingIntervals.get(t);i&&window.clearInterval(i);try{await this.tgApi("sendChatAction",a)}catch(c){console.warn("Agent Fleet: Telegram sendChatAction failed",c)}let o=window.setInterval(()=>{this.tgApi("sendChatAction",a).catch(()=>{})},4500);this.typingIntervals.set(t,o)}else{let i=this.typingIntervals.get(t);i&&(window.clearInterval(i),this.typingIntervals.delete(t))}}async setThreadTitle(t,e){}async broadcast(t){let e=this.config.allowedUsers[0];if(e)try{let s=za(t,4096);for(let n of s)await this.tgApi("sendMessage",{chat_id:e,text:n,parse_mode:"Markdown"})}catch(s){console.error(`Agent Fleet: Telegram broadcast failed on ${this.config.name}`,s)}}onInbound(t){return this.inboundHandlers.add(t),()=>this.inboundHandlers.delete(t)}onStatusChange(t){return this.statusHandlers.add(t),()=>this.statusHandlers.delete(t)}setAllowedAgentsResolver(t){this.allowedAgentsResolver=t}pickerAgents(){return this.allowedAgentsResolver?this.allowedAgentsResolver():this.config.allowedAgents}onAgentSwitch(t){return this.agentSwitchHandlers.add(t),()=>this.agentSwitchHandlers.delete(t)}poll(){this.stopping||(async()=>{try{this.pollAbort=new AbortController;let t=await this.tgApi("getUpdates",{offset:this.pollOffset,timeout:30,allowed_updates:["message","callback_query"]},this.pollAbort.signal);if(t.ok&&t.result)for(let e of t.result){if(this.pollOffset=e.update_id+1,e.callback_query){this.handleCallbackQuery(e.callback_query);continue}e.message&&this.routeMessage(e.message)}this.backoffMs=1e3,this.status!=="connected"&&this.setStatus("connected")}catch(t){if(t instanceof DOMException&&t.name==="AbortError")return;if(console.warn(`Agent Fleet: Telegram poll failed on ${this.config.name}`,t),this.status!=="error"&&this.status!=="needs-auth"){let e=t instanceof Error?t.message:String(t);this.setStatus(e.includes("401")||e.includes("Unauthorized")?"needs-auth":"error")}await new Promise(e=>window.setTimeout(e,this.backoffMs)),this.backoffMs=Math.min(3e4,this.backoffMs*2)}this.stopping||(this.pollTimer=window.setTimeout(()=>this.poll(),100))})()}routeMessage(t){let e=t.photo&&t.photo.length>0,s=t.document&&this.isImageMime(t.document.mime_type),n=!!t.text;if(!t.from||!n&&!e&&!s)return;let a=t.text??t.caption??"";if(a==="/agents"||a.startsWith("/agents@")){this.handleAgentsCommand(t);return}if(a.startsWith("/"))return;let i=String(t.from.id),o=String(t.chat.id),c=t.message_thread_id?String(t.message_thread_id):void 0,l=c?`tg:${o}:topic:${c}`:`tg:${o}`;this.buildAndEmitMessage(t,a,l,i,o,c)}async buildAndEmitMessage(t,e,s,n,a,i){let o=[];try{if(t.photo&&t.photo.length>0){let l=t.photo[t.photo.length-1],h=await this.downloadFile(l.file_id,`photo_${t.message_id}.jpg`,"image/jpeg");h&&o.push(h)}if(t.document&&this.isImageMime(t.document.mime_type)){let l=t.document.file_name??`doc_${t.message_id}`,h=t.document.mime_type??"image/jpeg",d=await this.downloadFile(t.document.file_id,l,h);d&&o.push(d)}}catch(l){console.warn("Agent Fleet: Telegram image download failed",l)}let c={conversationId:s,externalUserId:n,text:e,timestamp:new Date(t.date*1e3).toISOString(),meta:{telegram_chat_id:a,telegram_message_id:t.message_id,telegram_thread_id:i,chat_type:t.chat.type},...o.length>0?{images:o}:{}};for(let l of this.inboundHandlers)try{l(c)}catch(h){console.error("Agent Fleet: Telegram inbound handler threw",h)}}async downloadFile(t,e,s){let a=(await this.tgApi("getFile",{file_id:t})).result?.file_path;if(!a)return null;let i=`${Mo}/file/bot${this.credential.botToken}/${a}`;return{data:(await(0,Ga.requestUrl)({url:i,method:"GET"})).arrayBuffer,filename:e,mimeType:s}}isImageMime(t){return t?t==="image/jpeg"||t==="image/png"||t==="image/gif"||t==="image/webp"||t==="image/bmp"||t==="image/tiff":!1}async handleAgentsCommand(t){let e=String(t.chat.id),s=t.message_thread_id,n=this.pickerAgents();if(n.length===0){await this.tgApi("sendMessage",{chat_id:e,text:"No agents available. Add existing, enabled agents to `allowed_agents` in the channel file.",...s?{message_thread_id:s}:{}});return}let a=n.map(i=>[{text:i===this.config.defaultAgent?`${i} \u2713`:i,callback_data:`switch:${i}`}]);await this.tgApi("sendMessage",{chat_id:e,text:"*Select an agent to chat with:*",parse_mode:"Markdown",reply_markup:{inline_keyboard:a},...s?{message_thread_id:s}:{}})}async handleCallbackQuery(t){let e=t.data;if(!e?.startsWith("switch:")){await this.tgApi("answerCallbackQuery",{callback_query_id:t.id});return}let s=e.slice(7),n=String(t.from.id),a=String(t.message?.chat?.id??t.from.id),i=t.message?.message_thread_id?String(t.message.message_thread_id):void 0,o=i?`tg:${a}:topic:${i}`:`tg:${a}`;for(let c of this.agentSwitchHandlers)try{c(o,s,n)}catch(l){console.error("Agent Fleet: Telegram agent switch handler threw",l)}if(await this.tgApi("answerCallbackQuery",{callback_query_id:t.id,text:`Switched to ${s}`}),t.message){let l=this.pickerAgents().map(h=>[{text:h===s?`${h} \u2713`:h,callback_data:`switch:${h}`}]);try{await this.tgApi("editMessageText",{chat_id:a,message_id:t.message.message_id,text:`*Active agent: ${s}*`,parse_mode:"Markdown",reply_markup:{inline_keyboard:l}})}catch{}}}async tgApi(t,e,s){if(s?.aborted)throw new DOMException("Aborted","AbortError");let n=`${Mo}/bot${this.credential.botToken}/${t}`,a=(0,Ga.requestUrl)({url:n,method:"POST",contentType:"application/json",body:JSON.stringify(e),throw:!1}),i;if(s?i=await Promise.race([a,new Promise((o,c)=>{s.addEventListener("abort",()=>c(new DOMException("Aborted","AbortError")),{once:!0})})]):i=await a,i.status===401||i.status===403)throw new Error(`Telegram ${t} ${i.status} Unauthorized`);if(i.status===429){let c=i.json?.parameters?.retry_after??1;return await new Promise(l=>window.setTimeout(l,c*1e3)),this.tgApi(t,e)}if(i.status<200||i.status>=300)throw new Error(`Telegram ${t} HTTP ${i.status}: ${i.text}`);return i.json}setStatus(t){if(this.status!==t){this.status=t;for(let e of this.statusHandlers)try{e(t)}catch{}}}};function Lo(r){return r.startsWith("tg:")?r.split(":")[1]??null:null}function Fo(r){let t=r.split(":");if(t[2]==="topic"&&t[3])return t[3]}function za(r,t){if(r.length<=t)return[r];let e=[],s=r;for(;s.length>t;){let n=s.lastIndexOf(`
12033
11944
 
12034
11945
  `,t);n<t/2&&(n=s.lastIndexOf(`
12035
- `,t)),n<t/2&&(n=t),e.push(s.slice(0,n)),s=s.slice(n).replace(/^\n+/,"")}return s&&e.push(s),e}var Ya=require("obsidian"),Qd="https://discord.com/api/v10",Zd="wss://gateway.discord.gg",eh="?v=10&encoding=json",th="DiscordBot (https://github.com/denberek/obsidian-agent-fleet, 0.13.6)",sh=37376,nh=0,Va=1,ah=2,ih=6,rh=7,oh=9,lh=10,ch=11,dh=2,hh=3,Fo=4,uh=7,Oo=64,Bn=class{type="discord";config;credential;ws=null;status="stopped";stopping=!1;backoffMs=1e3;reconnectTimer=null;sessionId=null;resumeGatewayUrl=null;seq=null;canResume=!1;heartbeatTimer=null;heartbeatInitialTimer=null;heartbeatAcked=!0;selfUserId=null;applicationId=null;commandRegistered=!1;warnedEmptyContent=!1;typingIntervals=new Map;sendQueues=new Map;inboundHandlers=new Set;statusHandlers=new Set;agentSwitchHandlers=new Set;allowedAgentsResolver;constructor(t,e){if(e.type!=="discord")throw new Error(`DiscordAdapter requires a discord credential, got ${e.type}`);this.config=t,this.credential=e}async start(){this.stopping=!1,this.backoffMs=1e3,this.canResume=!1,await this.connect()}async stop(){this.stopping=!0,this.reconnectTimer&&(window.clearTimeout(this.reconnectTimer),this.reconnectTimer=null),this.clearHeartbeat();for(let[,t]of this.typingIntervals)window.clearInterval(t);this.typingIntervals.clear();try{this.ws?.close()}catch{}this.ws=null,this.setStatus("stopped")}getStatus(){return this.status}async send(t,e){let s=Bo(t);s&&await this.sendToTarget(s,e)}async sendToTarget(t,e){if(!t)return;let s=Uo(e,2e3);await this.enqueueSend(t,async()=>{for(let n of s)await this.discordApi("POST",`/channels/${t}/messages`,{content:n})})}async setTyping(t,e){let s=Bo(t);if(s)if(e){let n=this.typingIntervals.get(t);n&&window.clearInterval(n);try{await this.discordApi("POST",`/channels/${s}/typing`)}catch(r){console.warn("Agent Fleet: Discord typing trigger failed",r)}let a=window.setInterval(()=>{this.discordApi("POST",`/channels/${s}/typing`).catch(()=>{})},8e3);this.typingIntervals.set(t,a)}else{let n=this.typingIntervals.get(t);n&&(window.clearInterval(n),this.typingIntervals.delete(t))}}async broadcast(t){let e=this.config.allowedUsers[0];if(e)try{let n=(await this.discordApi("POST","/users/@me/channels",{recipient_id:e}))?.id;if(!n)return;let a=Uo(t,2e3);for(let r of a)await this.discordApi("POST",`/channels/${n}/messages`,{content:r})}catch(s){console.error(`Agent Fleet: Discord broadcast failed on ${this.config.name}`,s)}}onInbound(t){return this.inboundHandlers.add(t),()=>this.inboundHandlers.delete(t)}onStatusChange(t){return this.statusHandlers.add(t),()=>this.statusHandlers.delete(t)}onAgentSwitch(t){return this.agentSwitchHandlers.add(t),()=>this.agentSwitchHandlers.delete(t)}setAllowedAgentsResolver(t){this.allowedAgentsResolver=t}pickerAgents(){return this.allowedAgentsResolver?this.allowedAgentsResolver():this.config.allowedAgents}async connect(){if(this.stopping)return;this.setStatus(this.ws?"reconnecting":"connecting");let e=`${this.canResume&&this.resumeGatewayUrl?this.resumeGatewayUrl:Zd}${eh}`;try{let s=new ns(e);s.on("message",r=>{this.handleFrame(r)}),s.on("error",r=>{console.warn(`Agent Fleet: Discord WebSocket error on ${this.config.name}`,r)}),s.on("close",r=>{this.ws=null,this.clearHeartbeat(),this.handleClose(r)}),this.ws=s;let n=window.setTimeout(()=>{if(this.status==="connecting"||this.status==="reconnecting"){console.warn(`Agent Fleet: Discord WebSocket connect timeout on ${this.config.name}`);try{s.close()}catch{}}},3e4);s.on("close",()=>window.clearTimeout(n));let a=this.onStatusChange(r=>{r==="connected"&&(window.clearTimeout(n),a())})}catch(s){console.error("Agent Fleet: Discord WebSocket open failed",s),this.setStatus("error"),this.scheduleReconnect()}}handleClose(t){let e=!1;t===4014?(console.error(`Agent Fleet: Discord channel ${this.config.name} \u2014 disallowed intents (4014). Enable the Message Content intent in the Developer Portal \u2192 your app \u2192 Bot \u2192 Privileged Gateway Intents, then reload.`),this.canResume=!1,this.setStatus("needs-auth"),e=!0):t===4004||t===4013?(console.error(`Agent Fleet: Discord channel ${this.config.name} \u2014 gateway auth/intents error (${t}). Check the bot token credential, then reload.`),this.canResume=!1,this.setStatus("needs-auth"),e=!0):(t===4007||t===4009)&&(this.canResume=!1),!this.stopping&&!e&&this.scheduleReconnect()}handleFrame(t){let e;try{e=JSON.parse(t.toString())}catch{return}switch(typeof e.s=="number"&&(this.seq=e.s),e.op){case lh:{let s=e.d?.heartbeat_interval;this.startHeartbeat(s??41250),this.canResume&&this.sessionId&&this.seq!==null?this.sendGateway(ih,{token:this.credential.botToken,session_id:this.sessionId,seq:this.seq}):this.identify();return}case Va:{this.sendGateway(Va,this.seq);return}case ch:{this.heartbeatAcked=!0;return}case rh:{try{this.ws?.close()}catch{}return}case oh:{e.d===!0||(this.canResume=!1,this.sessionId=null),window.setTimeout(()=>{try{this.ws?.close()}catch{}},1e3+Math.floor(Math.random()*4e3));return}case nh:{this.handleDispatch(e.t??"",e.d);return}default:return}}handleDispatch(t,e){if(t==="READY"){let s=e;this.sessionId=s.session_id??null,this.resumeGatewayUrl=s.resume_gateway_url??null,this.selfUserId=s.user?.id??null,this.applicationId=s.application?.id??null,this.canResume=!0,this.backoffMs=1e3,this.setStatus("connected"),this.registerAgentsCommand();return}if(t==="RESUMED"){this.backoffMs=1e3,this.setStatus("connected");return}if(t==="MESSAGE_CREATE"){this.routeMessage(e);return}if(t==="INTERACTION_CREATE"){this.handleInteraction(e);return}}identify(){this.sendGateway(ah,{token:this.credential.botToken,intents:sh,properties:{os:"linux",browser:"agent-fleet",device:"agent-fleet"}})}sendGateway(t,e){if(!(!this.ws||this.ws.readyState!==ns.OPEN))try{this.ws.send(JSON.stringify({op:t,d:e}))}catch(s){console.warn("Agent Fleet: Discord gateway send failed",s)}}startHeartbeat(t){this.clearHeartbeat(),this.heartbeatAcked=!0;let e=()=>{if(!this.heartbeatAcked){console.warn(`Agent Fleet: Discord heartbeat not ACKed on ${this.config.name} \u2014 reconnecting`);try{this.ws?.close()}catch{}return}this.heartbeatAcked=!1,this.sendGateway(Va,this.seq)};this.heartbeatInitialTimer=window.setTimeout(()=>{e(),this.heartbeatTimer=window.setInterval(e,t)},Math.floor(t*Math.random()))}clearHeartbeat(){this.heartbeatInitialTimer&&(window.clearTimeout(this.heartbeatInitialTimer),this.heartbeatInitialTimer=null),this.heartbeatTimer&&(window.clearInterval(this.heartbeatTimer),this.heartbeatTimer=null)}scheduleReconnect(){if(this.stopping||this.reconnectTimer)return;let t=this.backoffMs;this.backoffMs=Math.min(3e4,this.backoffMs*2),console.warn(`Agent Fleet: Discord channel ${this.config.name} scheduling reconnect in ${t}ms`),this.reconnectTimer=window.setTimeout(()=>{this.reconnectTimer=null,!this.stopping&&this.connect()},t)}routeMessage(t){if(!t.author||t.author.bot||this.selfUserId&&t.author.id===this.selfUserId||!t.channel_id)return;let e=(t.attachments??[]).filter(a=>typeof a.content_type=="string"&&a.content_type.startsWith("image/")),s=ph(t.content??"",this.selfUserId);if(!s&&e.length===0){this.warnedEmptyContent||(this.warnedEmptyContent=!0,console.warn(`Agent Fleet: Discord channel ${this.config.name} received a message with empty content. If this persists, enable the Message Content intent in the Developer Portal.`));return}let n=No(t.guild_id,t.channel_id);this.buildAndEmitMessage(t,s,n,e)}async buildAndEmitMessage(t,e,s,n){let a=[];for(let o of n)try{let c=await(0,Ya.requestUrl)({url:o.url,method:"GET"});a.push({data:c.arrayBuffer,filename:o.filename||`attachment_${t.id}`,mimeType:o.content_type??"image/jpeg"})}catch(c){console.warn("Agent Fleet: Discord attachment download failed",c)}let r={conversationId:s,externalUserId:t.author.id,text:e,timestamp:t.timestamp??new Date().toISOString(),meta:{discord_guild_id:t.guild_id,discord_channel_id:t.channel_id,discord_message_id:t.id,is_dm:!t.guild_id},...a.length>0?{images:a}:{}};for(let o of this.inboundHandlers)try{o(r)}catch(c){console.error("Agent Fleet: Discord inbound handler threw",c)}}async registerAgentsCommand(){if(!(this.commandRegistered||!this.applicationId))try{await this.discordApi("PUT",`/applications/${this.applicationId}/commands`,[{name:"agents",description:"Switch the active agent",type:1}]),this.commandRegistered=!0}catch(t){console.warn(`Agent Fleet: Discord /agents command registration failed on ${this.config.name}`,t)}}async handleInteraction(t){if(t.type===dh){t.data?.name==="agents"&&await this.respondWithAgentPicker(t);return}if(t.type===hh){let e=t.data?.custom_id??"";if(!e.startsWith("switch:"))return;let s=e.slice(7),n=t.member?.user?.id??t.user?.id,a=t.channel_id;if(!n||!a)return;let r=No(t.guild_id,a);for(let o of this.agentSwitchHandlers)try{o(r,s,n)}catch(c){console.error("Agent Fleet: Discord agent switch handler threw",c)}await this.respondToInteraction(t,uh,{content:`Active agent: **${s}**`,components:this.buildAgentButtons(s)});return}}async respondWithAgentPicker(t){if(this.pickerAgents().length===0){await this.respondToInteraction(t,Fo,{content:"No agents available. Add existing, enabled agents to `allowed_agents` in the channel file.",flags:Oo});return}await this.respondToInteraction(t,Fo,{content:"Select an agent to chat with:",flags:Oo,components:this.buildAgentButtons(this.config.defaultAgent)})}buildAgentButtons(t){let e=this.pickerAgents().slice(0,25),s=[];for(let n=0;n<e.length;n+=5){let a=e.slice(n,n+5);s.push({type:1,components:a.map(r=>({type:2,style:r===t?1:2,label:r===t?`${r} \u2713`:r,custom_id:`switch:${r}`}))})}return s}async respondToInteraction(t,e,s){try{await this.discordApi("POST",`/interactions/${t.id}/${t.token}/callback`,{type:e,data:s})}catch(n){console.warn("Agent Fleet: Discord interaction response failed",n)}}async discordApi(t,e,s){let n=await(0,Ya.requestUrl)({url:`${Qd}${e}`,method:t,contentType:"application/json",headers:{Authorization:`Bot ${this.credential.botToken}`,"User-Agent":th},...s!==void 0?{body:JSON.stringify(s)}:{},throw:!1});if(n.status===429){let a=n.json?.retry_after??Number(n.headers["retry-after"]??"1");return await new Promise(r=>window.setTimeout(r,Math.max(1e3,a*1e3))),this.discordApi(t,e,s)}if(n.status===401||n.status===403)throw new Error(`Discord ${t} ${e} ${n.status}: ${mh(n.text)}`);if(n.status<200||n.status>=300)throw new Error(`Discord ${t} ${e} HTTP ${n.status}: ${n.text}`);if(n.status!==204)return n.json}async enqueueSend(t,e){let n=(this.sendQueues.get(t)??Promise.resolve()).then(e),a=n.catch(()=>{});this.sendQueues.set(t,a);try{await n}finally{this.sendQueues.get(t)===a&&this.sendQueues.delete(t)}}setStatus(t){if(this.status!==t){this.status=t;for(let e of this.statusHandlers)try{e(t)}catch{}}}};function No(i,t){return i?`discord:${i}:${t}`:`discord:dm:${t}`}function Bo(i){let t=i.split(":");return t[0]!=="discord"?null:t[2]??null}function ph(i,t){let e=i.trimStart();if(t){let s=new RegExp(`^<@!?${t}>\\s*`);e=e.replace(s,"")}return e.trim()}function mh(i){let t=i??"";try{let e=JSON.parse(t);if(e&&(e.message!==void 0||e.code!==void 0))return`${e.message??"error"} (code ${e.code??"?"})`}catch{}return t||"no body"}function Uo(i,t){if(i.length<=t)return[i];let e=[],s=i;for(;s.length>t;){let n=s.lastIndexOf(`
11946
+ `,t)),n<t/2&&(n=t),e.push(s.slice(0,n)),s=s.slice(n).replace(/^\n+/,"")}return s&&e.push(s),e}var Ya=require("obsidian"),Zd="https://discord.com/api/v10",eh="wss://gateway.discord.gg",th="?v=10&encoding=json",sh="DiscordBot (https://github.com/denberek/obsidian-agent-fleet, 0.13.6)",nh=37376,ah=0,Va=1,rh=2,ih=6,oh=7,lh=9,ch=10,dh=11,hh=2,uh=3,Oo=4,ph=7,No=64,Bn=class{type="discord";config;credential;ws=null;status="stopped";stopping=!1;backoffMs=1e3;reconnectTimer=null;sessionId=null;resumeGatewayUrl=null;seq=null;canResume=!1;heartbeatTimer=null;heartbeatInitialTimer=null;heartbeatAcked=!0;selfUserId=null;applicationId=null;commandRegistered=!1;warnedEmptyContent=!1;typingIntervals=new Map;sendQueues=new Map;inboundHandlers=new Set;statusHandlers=new Set;agentSwitchHandlers=new Set;allowedAgentsResolver;constructor(t,e){if(e.type!=="discord")throw new Error(`DiscordAdapter requires a discord credential, got ${e.type}`);this.config=t,this.credential=e}async start(){this.stopping=!1,this.backoffMs=1e3,this.canResume=!1,await this.connect()}async stop(){this.stopping=!0,this.reconnectTimer&&(window.clearTimeout(this.reconnectTimer),this.reconnectTimer=null),this.clearHeartbeat();for(let[,t]of this.typingIntervals)window.clearInterval(t);this.typingIntervals.clear();try{this.ws?.close()}catch{}this.ws=null,this.setStatus("stopped")}getStatus(){return this.status}async send(t,e){let s=Uo(t);s&&await this.sendToTarget(s,e)}async sendToTarget(t,e){if(!t)return;let s=$o(e,2e3);await this.enqueueSend(t,async()=>{for(let n of s)await this.discordApi("POST",`/channels/${t}/messages`,{content:n})})}async setTyping(t,e){let s=Uo(t);if(s)if(e){let n=this.typingIntervals.get(t);n&&window.clearInterval(n);try{await this.discordApi("POST",`/channels/${s}/typing`)}catch(i){console.warn("Agent Fleet: Discord typing trigger failed",i)}let a=window.setInterval(()=>{this.discordApi("POST",`/channels/${s}/typing`).catch(()=>{})},8e3);this.typingIntervals.set(t,a)}else{let n=this.typingIntervals.get(t);n&&(window.clearInterval(n),this.typingIntervals.delete(t))}}async broadcast(t){let e=this.config.allowedUsers[0];if(e)try{let n=(await this.discordApi("POST","/users/@me/channels",{recipient_id:e}))?.id;if(!n)return;let a=$o(t,2e3);for(let i of a)await this.discordApi("POST",`/channels/${n}/messages`,{content:i})}catch(s){console.error(`Agent Fleet: Discord broadcast failed on ${this.config.name}`,s)}}onInbound(t){return this.inboundHandlers.add(t),()=>this.inboundHandlers.delete(t)}onStatusChange(t){return this.statusHandlers.add(t),()=>this.statusHandlers.delete(t)}onAgentSwitch(t){return this.agentSwitchHandlers.add(t),()=>this.agentSwitchHandlers.delete(t)}setAllowedAgentsResolver(t){this.allowedAgentsResolver=t}pickerAgents(){return this.allowedAgentsResolver?this.allowedAgentsResolver():this.config.allowedAgents}async connect(){if(this.stopping)return;this.setStatus(this.ws?"reconnecting":"connecting");let e=`${this.canResume&&this.resumeGatewayUrl?this.resumeGatewayUrl:eh}${th}`;try{let s=new as(e);s.on("message",i=>{this.handleFrame(i)}),s.on("error",i=>{console.warn(`Agent Fleet: Discord WebSocket error on ${this.config.name}`,i)}),s.on("close",i=>{this.ws=null,this.clearHeartbeat(),this.handleClose(i)}),this.ws=s;let n=window.setTimeout(()=>{if(this.status==="connecting"||this.status==="reconnecting"){console.warn(`Agent Fleet: Discord WebSocket connect timeout on ${this.config.name}`);try{s.close()}catch{}}},3e4);s.on("close",()=>window.clearTimeout(n));let a=this.onStatusChange(i=>{i==="connected"&&(window.clearTimeout(n),a())})}catch(s){console.error("Agent Fleet: Discord WebSocket open failed",s),this.setStatus("error"),this.scheduleReconnect()}}handleClose(t){let e=!1;t===4014?(console.error(`Agent Fleet: Discord channel ${this.config.name} \u2014 disallowed intents (4014). Enable the Message Content intent in the Developer Portal \u2192 your app \u2192 Bot \u2192 Privileged Gateway Intents, then reload.`),this.canResume=!1,this.setStatus("needs-auth"),e=!0):t===4004||t===4013?(console.error(`Agent Fleet: Discord channel ${this.config.name} \u2014 gateway auth/intents error (${t}). Check the bot token credential, then reload.`),this.canResume=!1,this.setStatus("needs-auth"),e=!0):(t===4007||t===4009)&&(this.canResume=!1),!this.stopping&&!e&&this.scheduleReconnect()}handleFrame(t){let e;try{e=JSON.parse(t.toString())}catch{return}switch(typeof e.s=="number"&&(this.seq=e.s),e.op){case ch:{let s=e.d?.heartbeat_interval;this.startHeartbeat(s??41250),this.canResume&&this.sessionId&&this.seq!==null?this.sendGateway(ih,{token:this.credential.botToken,session_id:this.sessionId,seq:this.seq}):this.identify();return}case Va:{this.sendGateway(Va,this.seq);return}case dh:{this.heartbeatAcked=!0;return}case oh:{try{this.ws?.close()}catch{}return}case lh:{e.d===!0||(this.canResume=!1,this.sessionId=null),window.setTimeout(()=>{try{this.ws?.close()}catch{}},1e3+Math.floor(Math.random()*4e3));return}case ah:{this.handleDispatch(e.t??"",e.d);return}default:return}}handleDispatch(t,e){if(t==="READY"){let s=e;this.sessionId=s.session_id??null,this.resumeGatewayUrl=s.resume_gateway_url??null,this.selfUserId=s.user?.id??null,this.applicationId=s.application?.id??null,this.canResume=!0,this.backoffMs=1e3,this.setStatus("connected"),this.registerAgentsCommand();return}if(t==="RESUMED"){this.backoffMs=1e3,this.setStatus("connected");return}if(t==="MESSAGE_CREATE"){this.routeMessage(e);return}if(t==="INTERACTION_CREATE"){this.handleInteraction(e);return}}identify(){this.sendGateway(rh,{token:this.credential.botToken,intents:nh,properties:{os:"linux",browser:"agent-fleet",device:"agent-fleet"}})}sendGateway(t,e){if(!(!this.ws||this.ws.readyState!==as.OPEN))try{this.ws.send(JSON.stringify({op:t,d:e}))}catch(s){console.warn("Agent Fleet: Discord gateway send failed",s)}}startHeartbeat(t){this.clearHeartbeat(),this.heartbeatAcked=!0;let e=()=>{if(!this.heartbeatAcked){console.warn(`Agent Fleet: Discord heartbeat not ACKed on ${this.config.name} \u2014 reconnecting`);try{this.ws?.close()}catch{}return}this.heartbeatAcked=!1,this.sendGateway(Va,this.seq)};this.heartbeatInitialTimer=window.setTimeout(()=>{e(),this.heartbeatTimer=window.setInterval(e,t)},Math.floor(t*Math.random()))}clearHeartbeat(){this.heartbeatInitialTimer&&(window.clearTimeout(this.heartbeatInitialTimer),this.heartbeatInitialTimer=null),this.heartbeatTimer&&(window.clearInterval(this.heartbeatTimer),this.heartbeatTimer=null)}scheduleReconnect(){if(this.stopping||this.reconnectTimer)return;let t=this.backoffMs;this.backoffMs=Math.min(3e4,this.backoffMs*2),console.warn(`Agent Fleet: Discord channel ${this.config.name} scheduling reconnect in ${t}ms`),this.reconnectTimer=window.setTimeout(()=>{this.reconnectTimer=null,!this.stopping&&this.connect()},t)}routeMessage(t){if(!t.author||t.author.bot||this.selfUserId&&t.author.id===this.selfUserId||!t.channel_id)return;let e=(t.attachments??[]).filter(a=>typeof a.content_type=="string"&&a.content_type.startsWith("image/")),s=mh(t.content??"",this.selfUserId);if(!s&&e.length===0){this.warnedEmptyContent||(this.warnedEmptyContent=!0,console.warn(`Agent Fleet: Discord channel ${this.config.name} received a message with empty content. If this persists, enable the Message Content intent in the Developer Portal.`));return}let n=Bo(t.guild_id,t.channel_id);this.buildAndEmitMessage(t,s,n,e)}async buildAndEmitMessage(t,e,s,n){let a=[];for(let o of n)try{let c=await(0,Ya.requestUrl)({url:o.url,method:"GET"});a.push({data:c.arrayBuffer,filename:o.filename||`attachment_${t.id}`,mimeType:o.content_type??"image/jpeg"})}catch(c){console.warn("Agent Fleet: Discord attachment download failed",c)}let i={conversationId:s,externalUserId:t.author.id,text:e,timestamp:t.timestamp??new Date().toISOString(),meta:{discord_guild_id:t.guild_id,discord_channel_id:t.channel_id,discord_message_id:t.id,is_dm:!t.guild_id},...a.length>0?{images:a}:{}};for(let o of this.inboundHandlers)try{o(i)}catch(c){console.error("Agent Fleet: Discord inbound handler threw",c)}}async registerAgentsCommand(){if(!(this.commandRegistered||!this.applicationId))try{await this.discordApi("PUT",`/applications/${this.applicationId}/commands`,[{name:"agents",description:"Switch the active agent",type:1}]),this.commandRegistered=!0}catch(t){console.warn(`Agent Fleet: Discord /agents command registration failed on ${this.config.name}`,t)}}async handleInteraction(t){if(t.type===hh){t.data?.name==="agents"&&await this.respondWithAgentPicker(t);return}if(t.type===uh){let e=t.data?.custom_id??"";if(!e.startsWith("switch:"))return;let s=e.slice(7),n=t.member?.user?.id??t.user?.id,a=t.channel_id;if(!n||!a)return;let i=Bo(t.guild_id,a);for(let o of this.agentSwitchHandlers)try{o(i,s,n)}catch(c){console.error("Agent Fleet: Discord agent switch handler threw",c)}await this.respondToInteraction(t,ph,{content:`Active agent: **${s}**`,components:this.buildAgentButtons(s)});return}}async respondWithAgentPicker(t){if(this.pickerAgents().length===0){await this.respondToInteraction(t,Oo,{content:"No agents available. Add existing, enabled agents to `allowed_agents` in the channel file.",flags:No});return}await this.respondToInteraction(t,Oo,{content:"Select an agent to chat with:",flags:No,components:this.buildAgentButtons(this.config.defaultAgent)})}buildAgentButtons(t){let e=this.pickerAgents().slice(0,25),s=[];for(let n=0;n<e.length;n+=5){let a=e.slice(n,n+5);s.push({type:1,components:a.map(i=>({type:2,style:i===t?1:2,label:i===t?`${i} \u2713`:i,custom_id:`switch:${i}`}))})}return s}async respondToInteraction(t,e,s){try{await this.discordApi("POST",`/interactions/${t.id}/${t.token}/callback`,{type:e,data:s})}catch(n){console.warn("Agent Fleet: Discord interaction response failed",n)}}async discordApi(t,e,s){let n=await(0,Ya.requestUrl)({url:`${Zd}${e}`,method:t,contentType:"application/json",headers:{Authorization:`Bot ${this.credential.botToken}`,"User-Agent":sh},...s!==void 0?{body:JSON.stringify(s)}:{},throw:!1});if(n.status===429){let a=n.json?.retry_after??Number(n.headers["retry-after"]??"1");return await new Promise(i=>window.setTimeout(i,Math.max(1e3,a*1e3))),this.discordApi(t,e,s)}if(n.status===401||n.status===403)throw new Error(`Discord ${t} ${e} ${n.status}: ${fh(n.text)}`);if(n.status<200||n.status>=300)throw new Error(`Discord ${t} ${e} HTTP ${n.status}: ${n.text}`);if(n.status!==204)return n.json}async enqueueSend(t,e){let n=(this.sendQueues.get(t)??Promise.resolve()).then(e),a=n.catch(()=>{});this.sendQueues.set(t,a);try{await n}finally{this.sendQueues.get(t)===a&&this.sendQueues.delete(t)}}setStatus(t){if(this.status!==t){this.status=t;for(let e of this.statusHandlers)try{e(t)}catch{}}}};function Bo(r,t){return r?`discord:${r}:${t}`:`discord:dm:${t}`}function Uo(r){let t=r.split(":");return t[0]!=="discord"?null:t[2]??null}function mh(r,t){let e=r.trimStart();if(t){let s=new RegExp(`^<@!?${t}>\\s*`);e=e.replace(s,"")}return e.trim()}function fh(r){let t=r??"";try{let e=JSON.parse(t);if(e&&(e.message!==void 0||e.code!==void 0))return`${e.message??"error"} (code ${e.code??"?"})`}catch{}return t||"no body"}function $o(r,t){if(r.length<=t)return[r];let e=[],s=r;for(;s.length>t;){let n=s.lastIndexOf(`
12036
11947
 
12037
11948
  `,t);n<t/2&&(n=s.lastIndexOf(`
12038
- `,t)),n<t/2&&(n=t),e.push(s.slice(0,n)),s=s.slice(n).replace(/^\n+/,"")}return s&&e.push(s),e}var Ms=require("obsidian");var Un=class extends Ms.ItemView{constructor(e,s){super(e);this.plugin=s}getViewType(){return $t}getDisplayText(){return"Agent Fleet"}getIcon(){return"bot"}async onOpen(){this.plugin.subscribeView(this),await this.render()}async onClose(){this.plugin.unsubscribeView(this)}async render(){let e=this.contentEl;e.empty(),e.addClass("af-sidebar");let s=this.plugin.runtime.getSnapshot(),n=this.plugin.runtime.getFleetStatus(),a=e.createDiv({cls:"af-sidebar-section"});a.createDiv({cls:"af-sidebar-section-header",text:"AGENT FLEET"});let r=[{icon:"layout-dashboard",label:"Dashboard",page:"dashboard"},{icon:"bot",label:"Agents",page:"agents",badge:()=>s.agents.length},{icon:"columns-3",label:"Tasks Board",page:"kanban"},{icon:"scroll-text",label:"Run History",page:"runs"},{icon:"shield-check",label:"Approvals",page:"approvals",badge:()=>n.pending},{icon:"puzzle",label:"Skills",page:"skills",badge:()=>s.skills.length},{icon:"plug",label:"MCP Servers",page:"mcp",badge:()=>this.plugin.repository.getMcpServers().length},{icon:"radio",label:"Channels",page:"channels",badge:()=>this.plugin.channelManager?.getConnectedCount()??s.channels.length}];for(let l of r){let h=a.createDiv({cls:"af-sidebar-nav-item"}),d=h.createSpan({cls:"af-sidebar-nav-icon"});(0,Ms.setIcon)(d,l.icon),h.createSpan({cls:"af-sidebar-nav-label",text:l.label});let u=l.badge?.();u!==void 0&&u>0&&h.createSpan({cls:"af-sidebar-badge",text:String(u)}),h.setAttribute("role","button"),h.setAttribute("tabindex","0"),h.onclick=()=>void this.plugin.navigateDashboard(l.page),h.onkeydown=p=>{(p.key==="Enter"||p.key===" ")&&(p.preventDefault(),this.plugin.navigateDashboard(l.page))}}let o=e.createDiv({cls:"af-sidebar-section"});o.createDiv({cls:"af-sidebar-section-header",text:"AGENTS"}),s.agents.length===0&&o.createDiv({cls:"af-sidebar-empty",text:"No agents configured"});for(let l of s.agents){let h=this.plugin.runtime.getAgentState(l.name),d=o.createDiv({cls:"af-sidebar-agent-item"});d.createSpan({cls:`af-sidebar-agent-dot ${this.healthToClass(h.status)}`}),d.createSpan({cls:"af-sidebar-agent-name",text:l.name}),d.setAttribute("role","button"),d.setAttribute("tabindex","0"),d.onclick=()=>void this.plugin.navigateDashboard("agent-detail",l.name),d.onkeydown=u=>{(u.key==="Enter"||u.key===" ")&&(u.preventDefault(),this.plugin.navigateDashboard("agent-detail",l.name))}}let c=e.createDiv({cls:"af-sidebar-section"});c.createDiv({cls:"af-sidebar-section-header",text:"QUICK ACTIONS"}),this.renderQuickAction(c,"plus","New Agent",()=>void this.plugin.createAgentTemplate()),this.renderQuickAction(c,"plus","New Task",()=>void this.plugin.openCreateTask()),this.renderQuickAction(c,"plus","New Skill",()=>void this.plugin.createSkillTemplate())}renderQuickAction(e,s,n,a){let r=e.createDiv({cls:"af-sidebar-action-item"}),o=r.createSpan({cls:"af-sidebar-action-icon"});(0,Ms.setIcon)(o,s),r.createSpan({text:n}),r.setAttribute("role","button"),r.setAttribute("tabindex","0"),r.onclick=a,r.onkeydown=c=>{(c.key==="Enter"||c.key===" ")&&(c.preventDefault(),a())}}healthToClass(e){switch(e){case"running":return"running";case"error":return"error";case"pending":return"pending";case"disabled":return"disabled";default:return"idle"}}};var b=require("obsidian");var as=require("obsidian"),fh=["bot","brain","shield-check","search","file-text","rocket","wand","sparkles","zap","target","compass","eye","code","terminal","database","globe","mail","message-circle","book","pen-tool","palette","music","camera","chart-bar","clipboard","cpu","server","cloud","lock","key","bell","calendar","clock","heart","star","flag","bookmark"],$n=class extends as.Modal{constructor(e,s,n){super(e);this.onSelect=n;this.selectedIcon=s}searchQuery="";selectedIcon;allIcons=[];gridContainer;onOpen(){this.allIcons=(0,as.getIconIds)().sort();let{contentEl:e}=this;e.empty(),e.addClass("af-icon-picker-modal");let s=e.createEl("input",{cls:"af-icon-picker-search",attr:{type:"text",placeholder:"Search icons...",value:this.searchQuery}});this.gridContainer=e.createDiv({cls:"af-icon-picker-scroll"}),this.renderGrid(),s.addEventListener("input",()=>{this.searchQuery=s.value,this.renderGrid()}),window.setTimeout(()=>s.focus(),0)}renderGrid(){let e=this.gridContainer;e.empty();let s=this.searchQuery.toLowerCase().trim();if(!s)this.renderSection(e,"Popular",fh),this.renderSection(e,"All Icons",this.allIcons);else{let n=this.allIcons.filter(r=>r.includes(s)),a=n.length===0?"No results":`${n.length} result${n.length!==1?"s":""}`;this.renderSection(e,a,n)}}renderSection(e,s,n){let a=e.createDiv({cls:"af-icon-picker-section"});a.createDiv({cls:"af-icon-picker-section-label",text:s});let r=a.createDiv({cls:"af-icon-picker-grid"});for(let o of n)this.renderIconItem(r,o)}renderIconItem(e,s){let n=e.createDiv({cls:`af-icon-picker-item${this.selectedIcon===s?" selected":""}`});n.setAttribute("title",s),n.setAttribute("aria-label",s),(0,as.setIcon)(n,s),n.addEventListener("click",()=>{this.selectedIcon=s,this.onSelect(s),this.close()})}onClose(){this.contentEl.empty()}};var gh=[{match:/opus/i,rates:{input:15,output:75,cacheWrite:18.75,cacheRead:1.5}},{match:/sonnet/i,rates:{input:3,output:15,cacheWrite:3.75,cacheRead:.3}},{match:/haiku/i,rates:{input:1,output:5,cacheWrite:1.25,cacheRead:.1}},{match:/gpt-5|codex|o[0-9]/i,rates:{input:1.25,output:10,cacheWrite:1.25,cacheRead:.125}}],yh={input:3,output:15,cacheWrite:3.75,cacheRead:.3};function $o(i){for(let{match:t,rates:e}of gh)if(t.test(i))return e;return yh}function jo(i,t){let e=$o(i);return(t.inputTokens*e.input+t.outputTokens*e.output+t.cacheCreateTokens*e.cacheWrite+t.cacheReadTokens*e.cacheRead)/1e6}function Wo(i,t){let e=$o(i),s=.7*e.input+.3*e.output;return t*s/1e6}var Ho=require("obsidian");function R(i,t,e){let s=i.createSpan({cls:e??"af-icon"});return(0,Ho.setIcon)(s,t),s}function Ze(i,t={}){let e=activeDocument.createElementNS("http://www.w3.org/2000/svg",i);for(let[s,n]of Object.entries(t))e.setAttribute(s,n);return e}function vh(i){try{return new Date(i+"T12:00:00").toLocaleDateString(void 0,{month:"short",day:"numeric"})}catch{return i.slice(5)}}function qo(i,t){let e=t.length||1,s=1e3,n=6,a=4,r=4,o=s-a-r-n*(e-1),c=Math.floor(o/e),l=140,h=36,d=18,u=d+l+h,p=Math.max(1,...t.map(f=>f.success+f.failure+f.cancelled)),m=Ze("svg",{viewBox:`0 0 ${s} ${u}`,width:"100%",height:String(u),class:"af-chart-bar"});for(let f=0;f<=4;f++){let g=d+l-f/4*l;m.appendChild(Ze("line",{x1:String(a),y1:String(g),x2:String(s-r),y2:String(g),stroke:"var(--af-text-faint)","stroke-opacity":"0.15","stroke-width":"1"}))}for(let f=0;f<t.length;f++){let g=t[f],y=a+f*(c+n),v=g.success+g.failure+g.cancelled,k=v/p*l,w=g.success/p*l,S=g.cancelled/p*l,T=g.failure/p*l;if(g.success>0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-w),width:String(c),height:String(Math.max(w,2)),fill:"var(--af-green)",opacity:"0.85"})),g.cancelled>0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-w-S),width:String(c),height:String(Math.max(S,2)),fill:"var(--af-yellow)",opacity:"0.85"})),g.failure>0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-k),width:String(c),height:String(Math.max(T,2)),fill:"var(--af-red)",opacity:"0.85"})),v===0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-3),width:String(c),height:"3",rx:"1.5",fill:"var(--af-text-faint)",opacity:"0.2"})),v>0){let D=Ze("text",{x:String(y+c/2),y:String(d+l-k-5),"text-anchor":"middle","font-size":"10","font-weight":"600",fill:"var(--af-text-secondary)"});D.textContent=String(v),m.appendChild(D)}let _=Ze("text",{x:String(y+c/2),y:String(d+l+16),"text-anchor":"middle","font-size":"10",fill:"var(--af-text-muted)"});_.textContent=vh(g.date),m.appendChild(_)}i.appendChild(m)}function zo(i,t,e){let c=2*Math.PI*46,l=e>0?t/e:0,h=c*l,d=c-h,u=Ze("svg",{viewBox:"0 0 130 130",width:String(130),height:String(130),class:"af-chart-donut"});u.appendChild(Ze("circle",{cx:String(65),cy:String(65),r:String(46),fill:"none",stroke:e>0?"var(--af-red)":"var(--af-text-faint)","stroke-width":String(12),opacity:"0.2"})),l>0&&u.appendChild(Ze("circle",{cx:String(65),cy:String(65),r:String(46),fill:"none",stroke:"var(--af-green)","stroke-width":String(12),"stroke-dasharray":`${h} ${d}`,"stroke-dashoffset":String(c*.25),"stroke-linecap":"round"}));let p=Ze("text",{x:String(65),y:String(61),"text-anchor":"middle","dominant-baseline":"middle","font-size":"24","font-weight":"700",fill:"var(--af-text-primary)"});p.textContent=`${Math.round(l*100)}%`,u.appendChild(p);let m=Ze("text",{x:String(65),y:String(83),"text-anchor":"middle","font-size":"10",fill:"var(--af-text-muted)"});m.textContent=`${t}/${e} runs`,u.appendChild(m),i.appendChild(u)}function Go(i,t){i.draggable=!0,i.addEventListener("dragstart",e=>{e.dataTransfer?.setData("text/plain",t),i.addClass("af-dragging")}),i.addEventListener("dragend",()=>{i.removeClass("af-dragging")})}function Vo(i,t){i.addEventListener("dragover",e=>{e.preventDefault(),i.addClass("af-drag-over")}),i.addEventListener("dragleave",e=>{let s=e.relatedTarget;(!s||!i.contains(s))&&i.removeClass("af-drag-over")}),i.addEventListener("drop",e=>{e.preventDefault();let s=e.dataTransfer?.getData("text/plain");i.removeClass("af-drag-over"),s&&t(s)})}function Yo(i){let t=/^##\s+Lint\s+(\d{4}-\d{2}-\d{2})\s*$/gm,e,s=-1,n="";for(;(e=t.exec(i))!==null;)e.index>s&&(s=e.index,n=e[1]??"");if(s<0)return null;let a=i.slice(s),r=a.search(/\n##\s+(?!\s*#)/),o=r<0?a:a.slice(0,r);return{date:n,summary:jn(o,"Summary"),autoApplied:jn(o,"Auto-applied"),needsReview:jn(o,"Needs review"),refreshChained:jn(o,"Refresh chained")}}function jn(i,t){let e=t.replace(/[.*+?^${}()|[\]\\]/g,"\\$&"),n=new RegExp(`^###\\s+${e}\\s*$`,"m").exec(i);if(!n)return[];let a=n.index+n[0].length,r=i.slice(a),o=r.search(/\n###\s+/),l=(o<0?r:r.slice(0,o)).split(/\r?\n/).map(d=>d.trimEnd()),h=[];for(let d of l){let u=d.replace(/^\s+/,"");u.startsWith("- ")?h.push(u.slice(2).trim()):h.length>0&&(d.startsWith(" ")||d.startsWith(" "))&&(h[h.length-1]+=" "+u)}return h}var Ko={dashboard:"Dashboard",agents:"Agents",kanban:"Tasks Board",runs:"Run History",skills:"Skills Library",approvals:"Approvals",mcp:"MCP Servers",channels:"Channels","wiki-keepers":"Wiki Keepers","agent-detail":"Agent Details","task-detail":"Task Details","create-agent":"Create Agent","create-task":"Create Task","create-skill":"Create Skill","edit-agent":"Edit Agent","edit-task":"Edit Task","edit-skill":"Edit Skill","create-channel":"Create Channel","edit-channel":"Edit Channel","add-mcp-server":"Add MCP Server"},wh={dashboard:"layout-dashboard",agents:"bot",kanban:"columns-3",runs:"scroll-text",skills:"puzzle",approvals:"shield-check",mcp:"plug",channels:"radio","wiki-keepers":"library","agent-detail":"bot","task-detail":"circle-dot","create-agent":"plus","create-task":"plus","create-skill":"plus","edit-agent":"edit","edit-task":"edit","edit-skill":"edit","create-channel":"plus","edit-channel":"edit","add-mcp-server":"plus"},bh=["dashboard","agents","kanban","runs","approvals","skills","wiki-keepers","mcp","channels"],Jo=[["claude-code","Claude Code",!1],["codex","Codex",!1],["process","Process (coming soon)",!0],["http","HTTP (coming soon)",!0]],Qo=[["bypassPermissions","Bypass Permissions","Auto-approve everything except deny list"],["dontAsk","Don\u2019t Ask","Auto-approve all tool calls"],["acceptEdits","Accept Edits","Auto-approve file edits, block bash unless allowed"],["plan","Plan","Read-only mode, no writes or commands"],["default","Default","Ask for each tool call"]],Zo=[["bypassPermissions","Bypass (no sandbox)","No sandbox, auto-approve everything"],["workspace-write","Workspace Write","Sandboxed: writes only inside the working dir"],["read-only","Read Only","Sandboxed: no writes or side-effect commands"]];function Ls(i){let t=i.trim().toLowerCase();return t==="codex"||t==="openai-codex"}function Wn(i){return Ls(i)?Zo:Qo}function Hn(i,t){if(Ls(t))switch(i){case"acceptEdits":case"default":return"workspace-write";case"plan":return"read-only";case"dontAsk":return"bypassPermissions";default:return Zo.some(([e])=>e===i)?i:"bypassPermissions"}switch(i){case"workspace-write":return"acceptEdits";case"read-only":return"plan";case"danger-full-access":return"bypassPermissions";default:return Qo.some(([e])=>e===i)?i:"bypassPermissions"}}var Fs=class extends b.ItemView{constructor(e,s){super(e);this.plugin=s}currentPage="dashboard";detailContext;agentDetailTab="overview";streamingUnsubscribes=[];channelStatusUnsubscribe;authenticatingServers=new Set;getViewType(){return _t}getDisplayText(){return"Agent Fleet"}getIcon(){return"bot"}async onOpen(){this.plugin.subscribeView(this),this.channelStatusUnsubscribe=this.plugin.channelManager?.onStatusChange(()=>{this.currentPage==="channels"&&this.render()}),await this.render()}async onClose(){this.cleanupStreaming(),this.channelStatusUnsubscribe?.(),this.channelStatusUnsubscribe=void 0,this.plugin.unsubscribeView(this)}navigateTo(e,s){this.currentPage=e,this.detailContext=s,e!=="agent-detail"&&(this.agentDetailTab="overview"),this.render()}async render(){this.cleanupStreaming();let e=this.contentEl;e.empty(),e.addClass("af-root");let n=e.createDiv({cls:"af-app"}).createDiv({cls:"af-main-content"});this.renderTopBar(n),this.renderTabBar(n);let a=n.createDiv({cls:"af-page"});switch(this.currentPage){case"dashboard":this.renderDashboardPage(a);break;case"agents":this.renderAgentsPage(a);break;case"kanban":this.renderKanbanPage(a);break;case"runs":this.renderRunsPage(a);break;case"skills":this.renderSkillsPage(a);break;case"approvals":this.renderApprovalsPage(a);break;case"mcp":this.renderMcpPage(a);break;case"channels":this.renderChannelsPage(a);break;case"wiki-keepers":this.renderWikiKeepersPage(a);break;case"agent-detail":this.renderAgentDetailPage(a);break;case"task-detail":this.renderTaskDetailPage(a);break;case"create-agent":this.renderCreateAgentPage(a);break;case"create-task":this.renderCreateTaskPage(a);break;case"create-skill":this.renderCreateSkillPage(a);break;case"edit-agent":this.renderEditAgentPage(a);break;case"edit-task":this.renderEditTaskPage(a);break;case"edit-skill":this.renderEditSkillPage(a);break;case"create-channel":this.renderCreateChannelPage(a);break;case"edit-channel":this.renderEditChannelPage(a);break;case"add-mcp-server":this.renderAddMcpServerPage(a);break}}navigate(e,s){this.navigateTo(e,s)}cleanupStreaming(){for(let e of this.streamingUnsubscribes)e();this.streamingUnsubscribes=[]}renderTopBar(e){let s=e.createDiv({cls:"af-top-bar"}),n=s.createDiv({cls:"af-top-bar-title"});R(n,"bot","af-top-bar-icon"),n.createSpan({text:"Agent Fleet"});let a=s.createDiv({cls:"af-breadcrumb"}),r=a.createSpan({cls:"af-breadcrumb-sep"});(0,b.setIcon)(r,"chevron-right");let o=(m,f,g)=>{let y=a.createSpan({cls:f?"af-breadcrumb-link":"",text:m});f&&(y.onclick=()=>this.navigate(f,g))},c=()=>{let m=a.createSpan({cls:"af-breadcrumb-sep"});(0,b.setIcon)(m,"chevron-right")};switch(this.currentPage){case"agent-detail":o("Agents","agents"),c(),o(this.detailContext??"Agent");break;case"task-detail":o("Tasks Board","kanban"),c(),o(this.detailContext??"Task");break;case"edit-agent":o("Agents","agents"),c(),o(this.detailContext??"Agent","agent-detail",this.detailContext),c(),o("Edit");break;case"edit-task":o("Tasks Board","kanban"),c(),o(this.detailContext??"Task","task-detail",this.detailContext),c(),o("Edit");break;case"create-agent":o("Agents","agents"),c(),o("New Agent");break;case"create-task":o("Tasks Board","kanban"),c(),o("New Task");break;case"create-skill":o("Skills Library","skills"),c(),o("New Skill");break;case"edit-skill":o("Skills Library","skills"),c(),o(this.detailContext??"Skill"),c(),o("Edit");break;case"create-channel":o("Channels","channels"),c(),o("New Channel");break;case"edit-channel":o("Channels","channels"),c(),o(this.detailContext??"Channel"),c(),o("Edit");break;case"add-mcp-server":o("MCP Servers","mcp"),c(),o("Add Server");break;default:o(Ko[this.currentPage])}s.createDiv({cls:"af-top-bar-spacer"});let l=s.createDiv({cls:"af-search-wrap"});R(l,"search","af-search-icon");let h=l.createEl("input",{cls:"af-search-input",attr:{type:"text",placeholder:"Search agents, tasks, runs..."}});h.addEventListener("input",()=>{this.handleSearch(h.value,l)}),h.addEventListener("blur",()=>{window.setTimeout(()=>l.querySelector(".af-search-results")?.remove(),200)});let d=this.plugin.runtime.getFleetStatus(),u=s.createDiv({cls:"af-status-pills"});if(d.running>0){let m=u.createSpan({cls:"af-pill yellow"});m.createSpan({cls:"af-dot pulse"}),m.appendText(` ${d.running} Running`)}if(d.pending>0){let m=u.createSpan({cls:"af-pill blue"});m.createSpan({cls:"af-dot"}),m.appendText(` ${d.pending} Pending`)}let p=u.createSpan({cls:"af-pill green"});p.createSpan({cls:"af-dot"}),p.appendText(` ${d.completedToday} Today`)}renderTabBar(e){let s=e.createDiv({cls:"af-tab-bar"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.runtime.getFleetStatus();for(let r of bh){let o=this.currentPage===r||r==="agents"&&(this.currentPage==="agent-detail"||this.currentPage==="edit-agent"||this.currentPage==="create-agent")||r==="kanban"&&(this.currentPage==="task-detail"||this.currentPage==="edit-task")||r==="skills"&&(this.currentPage==="edit-skill"||this.currentPage==="create-skill")||r==="mcp"&&this.currentPage==="add-mcp-server",c=s.createEl("button",{cls:`af-tab-item${o?" active":""}`}),l=c.createSpan({cls:"af-tab-icon"});if((0,b.setIcon)(l,wh[r]),c.appendText(r==="dashboard"?"Overview":Ko[r]),r==="agents")c.createSpan({cls:"af-badge",text:String(n.agents.length)});else if(r==="skills")c.createSpan({cls:"af-badge",text:String(n.skills.length)});else if(r==="mcp"){let h=this.plugin.repository.getMcpServers().length;c.createSpan({cls:"af-badge",text:String(h)})}else r==="approvals"&&a.pending>0&&c.createSpan({cls:"af-badge af-badge-warn",text:String(a.pending)});c.onclick=()=>this.navigate(r)}}renderDashboardPage(e){let s=e.createDiv({cls:"af-dashboard"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.runtime.getRecentRuns(),r=this.plugin.runtime.getFleetStatus(),o=a.filter(S=>(S.approvals??[]).some(T=>T.status==="pending"));for(let S of o)for(let T of S.approvals??[])T.status==="pending"&&this.renderApprovalBanner(s,S,T.tool);let c=s.createDiv({cls:"af-dash-grid"}),l=n.agents.filter(S=>S.enabled).length,h=n.agents.length;this.renderStatCard(c,"Active Agents",`${l}`,`/ ${h}`,"bot",`${l} of ${h} enabled`);let d=this.toLocalDateStr(new Date),u=a.filter(S=>this.runToLocalDate(S.started)===d),p=u.filter(S=>S.status==="success").length,m=u.filter(S=>S.status==="failure"||S.status==="timeout").length;this.renderStatCard(c,"Runs Today",String(u.length),"","activity",`${p} passed \xB7 ${m} failed \xB7 ${r.running} running`);let f=this.plugin.runtime.getUsageRecords().filter(S=>this.runToLocalDate(S.ts)===d),{tokens:g,cost:y}=this.combinedTotals(u,f),v=y>0?` \xB7 $${y.toFixed(2)}`:"";this.renderStatCard(c,"Tokens Used",Ka(g),"","zap",`today${v}`);let k=n.tasks.filter(S=>S.enabled&&S.schedule);this.renderStatCard(c,"Scheduled Tasks",String(k.length),"","clock",k.length>0?`Next: ${this.getNextTaskLabel(k)}`:"No scheduled tasks"),this.renderChartsRow(s,a,this.plugin.runtime.getChartRuns()),this.renderStreamingCards(s);let w=s.createDiv({cls:"af-dash-split"});this.renderActivityTimeline(w,a),this.renderFleetStatusPanel(w,n)}renderChartsRow(e,s,n){let a=e.createDiv({cls:"af-charts-row"}),r=a.createDiv({cls:"af-section-card af-chart-section"}),c=r.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(c,"activity"),c.appendText(" Run Activity (14d)");let l=r.createDiv({cls:"af-chart-body"}),h=this.buildChartData(n,14);h.some(y=>y.success+y.failure+y.cancelled>0)?qo(l,h):this.renderEmptyState(l,"activity","No run data","Run agents to see activity");let d=a.createDiv({cls:"af-section-card af-chart-section"}),p=d.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(p,"target"),p.appendText(" Success Rate");let m=d.createDiv({cls:"af-chart-body af-chart-body-center"}),f=s.length,g=s.filter(y=>y.status==="success").length;f>0?zo(m,g,f):this.renderEmptyState(m,"target","No data","")}toLocalDateStr(e){let s=n=>String(n).padStart(2,"0");return`${e.getFullYear()}-${s(e.getMonth()+1)}-${s(e.getDate())}`}runToLocalDate(e){return this.toLocalDateStr(new Date(e))}runCost(e){return e.costUsd??Wo(e.model,e.tokensUsed??0)}usageCost(e){return e.costUsd??jo(e.model,{inputTokens:e.inputTokens,outputTokens:e.outputTokens,cacheReadTokens:e.cacheReadTokens,cacheCreateTokens:e.cacheCreateTokens})}combinedTotals(e,s){let n=e.reduce((r,o)=>r+(o.tokensUsed??0),0)+s.reduce((r,o)=>r+o.totalTokens,0),a=e.reduce((r,o)=>r+this.runCost(o),0)+s.reduce((r,o)=>r+this.usageCost(o),0);return{tokens:n,cost:a}}buildChartData(e,s){let n=[],a=new Date;for(let r=s-1;r>=0;r--){let o=new Date(a);o.setDate(o.getDate()-r);let c=this.toLocalDateStr(o),l=e.filter(h=>this.runToLocalDate(h.started)===c);n.push({date:c,success:l.filter(h=>h.status==="success").length,failure:l.filter(h=>h.status==="failure"||h.status==="timeout").length,cancelled:l.filter(h=>h.status==="cancelled").length})}return n}renderStreamingCards(e){let n=this.plugin.runtime.getSnapshot().agents.filter(c=>this.plugin.runtime.getAgentState(c.name).status==="running");if(n.length===0)return;let a=e.createDiv({cls:"af-streaming-section"}),o=a.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(o,"terminal"),o.appendText(" Active Agents");for(let c of n)this.renderStreamingCard(a,c.name)}renderStreamingCard(e,s){let n=e.createDiv({cls:"af-streaming-card"}),a=this.plugin.runtime.getAgentState(s),r=this.plugin.runtime.getSnapshot(),o=a.currentTaskId?r.tasks.find(m=>m.taskId===a.currentTaskId):void 0,c=o?` \u2192 ${o.taskId}`:a.currentTaskId?.startsWith("reflection-")?" \u2192 Reflection":a.currentTaskId?.startsWith("heartbeat-")||a.status==="running"?" \u2192 Heartbeat":"",l=n.createDiv({cls:"af-streaming-card-header"});l.createSpan({cls:"af-dot pulse",attr:{style:"background: var(--af-yellow)"}}),l.createSpan({cls:"af-streaming-card-agent",text:` ${s}`}),c&&l.createSpan({cls:"af-streaming-card-task",text:c});let h=n.createDiv({cls:"af-streaming-output"}),d=a.currentTaskId?.startsWith("reflection-")?"Consolidating memory\u2026":"Working\u2026",u=m=>{let f=ye(m).filter(g=>g.trim().length>0).slice(-4);h.setText(f.length>0?f.join(`
12039
- `):d)};u(this.plugin.runtime.getRunOutputBuffer(s));let p=this.plugin.runtime.onRunOutput(s,()=>{u(this.plugin.runtime.getRunOutputBuffer(s)),h.scrollTop=h.scrollHeight});this.streamingUnsubscribes.push(p)}renderApprovalBanner(e,s,n){let a=e.createDiv({cls:"af-approval-banner"}),r=a.createDiv({cls:"af-approval-icon"});(0,b.setIcon)(r,"shield-check");let o=a.createDiv({cls:"af-approval-text"});o.createDiv({cls:"af-approval-title",text:`${s.agent} wants to run: ${n}`}),o.createDiv({cls:"af-approval-desc",text:`Approval required for tool: ${n}`});let c=a.createDiv({cls:"af-approval-actions"}),l=c.createEl("button",{cls:"af-btn-approve",text:"Approve"});l.onclick=()=>void this.plugin.runtime.resolveApproval(s,n,"approved").then(()=>this.render());let h=c.createEl("button",{cls:"af-btn-reject",text:"Reject"});h.onclick=()=>void this.plugin.runtime.resolveApproval(s,n,"rejected").then(()=>this.render())}renderStatCard(e,s,n,a,r,o){let c=e.createDiv({cls:"af-stat-card"}),l=c.createDiv({cls:"af-stat-label"});R(l,r,"af-stat-icon"),l.appendText(` ${s}`);let h=c.createDiv({cls:"af-stat-value"});h.appendText(n),a&&h.createSpan({cls:"af-stat-value-suffix",text:a}),c.createDiv({cls:"af-stat-sub",text:o})}renderActivityTimeline(e,s){let n=e.createDiv({cls:"af-section-card"}),r=n.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(r,"inbox"),r.appendText(" Recent Activity");let o=n.createDiv({cls:"af-timeline"}),c=s.slice(0,8);if(c.length===0){this.renderEmptyState(o,"inbox","No runs yet","Run an agent to see activity here");return}for(let l of c)this.renderTimelineItem(o,l)}renderTimelineItem(e,s){let n=e.createDiv({cls:"af-timeline-item"}),a=this.statusToTimelineClass(s.status),r=n.createDiv({cls:`af-tl-icon ${a}`});(0,b.setIcon)(r,this.statusToIconName(s.status));let o=n.createDiv({cls:"af-tl-body"}),c=o.createDiv({cls:"af-tl-title"});c.createSpan({cls:"af-agent-tag",text:s.agent}),c.appendText(` ${s.task}`),o.createDiv({cls:"af-tl-desc",text:jt(s.output,100)});let l=o.createDiv({cls:"af-tl-meta"}),h=l.createSpan();if(R(h,"clock","af-meta-icon"),h.appendText(` ${this.formatStarted(s.started)} \xB7 ${this.formatDuration(s.durationSeconds)}`),s.tokensUsed){let d=l.createSpan();R(d,"zap","af-meta-icon"),d.appendText(` ${s.tokensUsed.toLocaleString()} tokens`)}n.onclick=()=>this.openSlideover(s)}renderFleetStatusPanel(e,s){let n=e.createDiv({cls:"af-section-card"}),a=n.createDiv({cls:"af-section-header"}),r=a.createDiv({cls:"af-section-title"});R(r,"bot"),r.appendText(" Fleet Status");let c=a.createDiv({cls:"af-section-actions"}).createEl("button",{cls:"af-btn-sm primary"});R(c,"plus","af-btn-icon"),c.appendText(" New Agent"),c.onclick=()=>void this.plugin.createAgentTemplate();let l=n.createDiv({cls:"af-agent-mini-list"});if(s.agents.length===0){this.renderEmptyState(l,"bot","No agents yet","Create your first agent to get started");return}for(let d of s.agents)this.renderAgentMini(l,d,s.tasks);let h=s.agents.filter(d=>d.enabled);if(h.length>0){let d=n.createDiv({cls:"af-quick-run"}),u=d.createDiv({cls:"af-quick-run-label"});R(u,"zap","af-meta-icon"),u.appendText(" Quick Run");let p=d.createDiv({cls:"af-quick-run-row"}),m=p.createEl("select",{cls:"af-select"});for(let g of h)m.createEl("option",{text:g.name,attr:{value:g.name}});let f=p.createEl("button",{cls:"af-btn-sm primary"});R(f,"play","af-btn-icon"),f.appendText(" Run"),f.onclick=()=>void this.plugin.runAgentPrompt(m.value)}}renderAgentMini(e,s,n){let a=this.plugin.runtime.getAgentState(s.name),r=n.filter(u=>u.agent===s.name),o=this.healthToClass(a.status),c=e.createDiv({cls:"af-agent-mini"}),l=c.createDiv({cls:`af-agent-avatar ${o}`});s.avatar?.trim()?this.renderAgentAvatar(l,s):l.setText(this.getInitials(s.name));let h=c.createDiv({cls:"af-agent-info"});h.createDiv({cls:"af-agent-name",text:s.name});let d="";if(a.status==="running")d=`Running now \xB7 ${r.length} task${r.length!==1?"s":""}`;else if(!s.enabled)d=`Disabled \xB7 ${r.length} task${r.length!==1?"s":""} paused`;else{let u=r.map(p=>p.nextRun).filter(Boolean).sort()[0];d=u?`Next: ${this.formatNextRun(u)} \xB7 ${r.length} task${r.length!==1?"s":""}`:`${r.length} task${r.length!==1?"s":""}`}h.createDiv({cls:"af-agent-desc",text:d}),c.createDiv({cls:`af-agent-status-dot ${o}`}),c.onclick=()=>this.navigate("agent-detail",s.name)}renderAgentsPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Agents"}),a.createDiv({cls:"af-toolbar-spacer"});let r=a.createEl("button",{cls:"af-btn-sm primary"});R(r,"plus","af-btn-icon"),r.appendText(" New Agent"),r.onclick=()=>void this.plugin.createAgentTemplate();let o=s.createDiv({cls:"af-agents-grid"});if(n.agents.length===0){this.renderEmptyState(o,"bot","No agents configured","Create your first agent to get started");return}for(let c of n.agents)this.renderAgentCard(o,c,n)}renderAgentCard(e,s,n){let a=this.plugin.runtime.getAgentState(s.name),r=this.plugin.runtime.getRecentRuns().filter(D=>D.agent===s.name),o=e.createDiv({cls:`af-agent-card${s.enabled?"":" disabled"}`}),c=o.createDiv({cls:"af-agent-card-header"}),l=s.enabled?this.healthToClass(a.status):"disabled",h=c.createDiv({cls:`af-agent-card-avatar ${l}`});this.renderAgentAvatar(h,s);let d=c.createDiv({cls:"af-agent-card-titleblock"}),u=d.createDiv({cls:"af-agent-card-name"});if(u.appendText(s.name),s.heartbeatEnabled&&s.heartbeatSchedule){let D=u.createSpan({cls:"af-heartbeat-indicator"});(0,b.setIcon)(D,"heart-pulse"),D.title=`Heartbeat: ${s.heartbeatSchedule}`}d.createDiv({cls:"af-agent-card-desc",text:s.description??"No description"});let p=c.createDiv({cls:`af-agent-card-toggle${s.enabled?" on":""}`});p.onclick=D=>{D.stopPropagation(),this.plugin.toggleAgent(s.name,!s.enabled)};let m=o.createDiv({cls:"af-agent-card-stats"}),f=r.length,g=r.filter(D=>D.status==="success").length,y=f>0?Math.round(g/f*100):0,v=f>0?Math.round(r.reduce((D,O)=>D+O.durationSeconds,0)/f):0,k=r.reduce((D,O)=>D+(O.tokensUsed??0),0);if(this.renderAgentStat(m,String(f),"Runs"),this.renderAgentStat(m,`${y}%`,"Success"),this.renderAgentStat(m,`${v}s`,"Avg Time"),this.renderAgentStat(m,Ka(k),"Tokens"),s.skills.length>0){let D=o.createDiv({cls:"af-agent-card-skills"});for(let O of s.skills)D.createSpan({cls:"af-skill-tag",text:O})}let w=o.createDiv({cls:"af-agent-card-footer"}),S=[`Model: ${s.model}`];s.approvalRequired.length>0&&S.push(`Approval: ${s.approvalRequired.join(", ")}`),s.memory&&S.push("Memory: on"),s.enabled||S.unshift("Disabled"),w.createSpan({cls:"af-agent-card-meta",text:S.join(" \xB7 ")});let T=w.createDiv({cls:"af-agent-card-actions"});if(!s.enabled){let D=T.createEl("button",{cls:"af-btn-sm",text:"Enable"});D.onclick=O=>{O.stopPropagation(),this.plugin.toggleAgent(s.name,!0)}}let _=T.createEl("button",{cls:"af-btn-sm"});if(R(_,"edit","af-btn-icon"),_.appendText(" Edit"),_.onclick=D=>{D.stopPropagation(),this.navigate("edit-agent",s.name)},s.enabled){let D=T.createEl("button",{cls:"af-btn-sm primary"});R(D,"play","af-btn-icon"),D.appendText(" Run"),D.onclick=O=>{O.stopPropagation(),this.plugin.runAgentPrompt(s.name)}}o.onclick=()=>this.navigate("agent-detail",s.name)}renderAgentStat(e,s,n){let a=e.createDiv({cls:"af-agent-stat"});a.createSpan({cls:"af-agent-stat-value",text:s}),a.createSpan({cls:"af-agent-stat-label",text:n})}renderAgentDetailPage(e){let s=e.createDiv({cls:"af-agent-detail-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"bot","No agent selected","Select an agent from the list");return}let a=this.plugin.runtime.getSnapshot().agents.find(k=>k.name===n);if(!a){this.renderEmptyState(s,"bot","Agent not found",`Agent "${n}" was not found`);return}let r=this.plugin.runtime.getAgentState(a.name),o=this.plugin.runtime.getRecentRuns().filter(k=>k.agent===a.name),c=s.createDiv({cls:"af-detail-header"}),l=c.createDiv({cls:"af-detail-header-left"}),h=l.createDiv({cls:`af-agent-card-avatar ${this.healthToClass(r.status)}`});this.renderAgentAvatar(h,a);let d=l.createDiv();d.createDiv({cls:"af-detail-header-name",text:a.name}),d.createDiv({cls:"af-detail-header-desc",text:a.description??"No description"});let u=c.createDiv({cls:"af-detail-header-actions"}),p=u.createEl("button",{cls:"af-btn-sm primary"});if(R(p,"message-circle","af-btn-icon"),p.appendText(" Chat"),p.onclick=()=>this.openChatSlideover(a),a.enabled){let k=u.createEl("button",{cls:"af-btn-sm"});R(k,"play","af-btn-icon"),k.appendText(" Run Now"),k.onclick=()=>void this.plugin.runAgentPrompt(a.name);let w=u.createEl("button",{cls:"af-btn-sm"});R(w,"pause","af-btn-icon"),w.appendText(" Disable"),w.onclick=()=>void this.plugin.toggleAgent(a.name,!1)}else{let k=u.createEl("button",{cls:"af-btn-sm"});R(k,"play","af-btn-icon"),k.appendText(" Enable"),k.onclick=()=>void this.plugin.toggleAgent(a.name,!0)}let m=u.createEl("button",{cls:"af-btn-sm"});R(m,"edit","af-btn-icon"),m.appendText(" Edit"),m.onclick=()=>this.navigate("edit-agent",a.name);let f=u.createEl("button",{cls:"af-btn-sm danger"});R(f,"trash-2","af-btn-icon"),f.appendText(" Delete"),f.onclick=()=>void this.plugin.deleteAgent(a.name);let g=s.createDiv({cls:"af-detail-tabs"}),y=[{id:"overview",label:"Overview",icon:"layout-dashboard"},{id:"config",label:"Config",icon:"settings"},{id:"runs",label:"Runs",icon:"scroll-text"},{id:"memory",label:"Memory",icon:"file-text"}];for(let k of y){let w=g.createEl("button",{cls:`af-detail-tab${this.agentDetailTab===k.id?" active":""}`});R(w,k.icon,"af-tab-icon"),w.appendText(` ${k.label}`),w.onclick=()=>{this.agentDetailTab=k.id,this.render()}}let v=s.createDiv({cls:"af-detail-tab-content"});switch(this.agentDetailTab){case"overview":this.renderAgentOverviewTab(v,a,o);break;case"config":this.renderAgentConfigTab(v,a);break;case"runs":this.renderAgentRunsTab(v,o);break;case"memory":this.renderAgentMemoryTab(v,a);break}}renderAgentOverviewTab(e,s,n){let a=e.createDiv({cls:"af-dash-grid"}),r=n.length,o=n.filter(w=>w.status==="success").length,c=r>0?Math.round(o/r*100):0,l=r>0?Math.round(n.reduce((w,S)=>w+S.durationSeconds,0)/r):0,h=this.plugin.runtime.getUsageRecords().filter(w=>w.agent===s.name),{tokens:d,cost:u}=this.combinedTotals(n,h),p=u>0?` \xB7 $${u.toFixed(2)}`:"";if(this.renderStatCard(a,"Total Runs",String(r),"","activity","all time"),this.renderStatCard(a,"Success Rate",`${c}%`,"","check-circle-2",`${o}/${r}`),this.renderStatCard(a,"Avg Time",`${l}s`,"","clock","per run"),this.renderStatCard(a,"Total Tokens",Ka(d),"","zap",`all time${p}`),s.isFolder&&(s.heartbeatBody.trim()||s.heartbeatEnabled)){let w=e.createDiv({cls:"af-section-card"}),S=w.createDiv({cls:"af-section-header"}),T=S.createDiv({cls:"af-section-title"});R(T,"heart-pulse"),T.appendText(" Heartbeat");let D=S.createDiv({cls:"af-detail-header-actions"}).createDiv({cls:`af-agent-card-toggle${s.heartbeatEnabled?" on":""}`});D.onclick=async()=>{let E=D.hasClass("on");await this.plugin.repository.updateHeartbeat(s.name,{enabled:!E}),await this.plugin.refreshFromVault(),new b.Notice(`Heartbeat ${E?"paused":"enabled"} for ${s.name}`)};let O=w.createDiv({cls:"af-config-form"});this.renderConfigRow(O,"Schedule",kh(s.heartbeatSchedule));let C=this.plugin.runtime.getNextHeartbeat(s.name);C&&s.heartbeatEnabled&&this.renderConfigRow(O,"Next run",this.timeUntil(C)),s.heartbeatChannel&&this.renderConfigRow(O,"Channel",s.heartbeatChannel)}if(s.skills.length>0){let w=e.createDiv({cls:"af-section-card"}),T=w.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(T,"puzzle"),T.appendText(" Skills");let _=w.createDiv({cls:"af-detail-skills-list"});for(let D of s.skills)_.createSpan({cls:"af-skill-tag",text:D})}let m=s.mcpServers??[];if(m.length>0){let w=e.createDiv({cls:"af-section-card"}),T=w.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(T,"plug"),T.appendText(" MCP Servers");let _=w.createDiv({cls:"af-mcp-overview-list"}),D=this.plugin.repository.getMcpServers();for(let O of m){let C=D.find(M=>M.name===O),E=_.createDiv({cls:"af-mcp-overview-row"}),P=E.createSpan({cls:`af-mcp-status-dot ${C?C.enabled?C.status:"disabled":"disconnected"}`});P.title=C?C.enabled?C.status:"disabled":"unknown",E.createSpan({cls:"af-mcp-overview-name",text:O});let N=C?.toolDetails.length??C?.tools.length??0;N>0?E.createSpan({cls:"af-mcp-overview-tools",text:`${N} tools`}):C&&!C.enabled?E.createSpan({cls:"af-mcp-overview-tools af-muted",text:"disabled"}):C?.status==="needs-auth"&&E.createSpan({cls:"af-mcp-overview-tools af-muted",text:"needs auth"})}}if(s.permissionRules.allow.length>0||s.permissionRules.deny.length>0||s.permissionMode&&s.permissionMode!=="default"){let w=e.createDiv({cls:"af-section-card"}),T=w.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(T,"shield-check"),T.appendText(" Permissions");let _=w.createDiv({cls:"af-config-form"});this.renderConfigRow(_,"Mode",s.permissionMode||"default"),s.permissionRules.allow.length>0&&this.renderConfigRow(_,"Allowed",s.permissionRules.allow.join(", ")),s.permissionRules.deny.length>0&&this.renderConfigRow(_,"Denied",s.permissionRules.deny.join(", "))}let g=e.createDiv({cls:"af-section-card"}),v=g.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(v,"scroll-text"),v.appendText(" Recent Runs");let k=g.createDiv({cls:"af-timeline"});if(n.length===0)this.renderEmptyState(k,"scroll-text","No runs yet","");else for(let w of n.slice(0,5))this.renderTimelineItem(k,w)}renderAgentConfigTab(e,s){let n=e.createDiv({cls:"af-config-form"});this.renderConfigRow(n,"Name",s.name),this.renderConfigRow(n,"Description",s.description??""),this.renderConfigRow(n,"Model",s.model),this.renderConfigRow(n,"Timeout",`${s.timeout}s`),this.renderConfigRow(n,"Working Directory",s.cwd??"(vault root)"),this.renderConfigRow(n,"Permission Mode",s.permissionMode||"default"),this.renderConfigRow(n,"Approval Required",s.approvalRequired.join(", ")||"none"),s.permissionRules.allow.length>0&&this.renderConfigRow(n,"Allowed Commands",s.permissionRules.allow.join(", ")),s.permissionRules.deny.length>0&&this.renderConfigRow(n,"Blocked Commands",s.permissionRules.deny.join(", ")),this.renderConfigRow(n,"Memory",s.memory?"Enabled":"Disabled"),this.renderConfigRow(n,"Auto-compact",s.autoCompactThreshold&&s.autoCompactThreshold>0?`at ${s.autoCompactThreshold}% context`:"disabled"),s.wikiReferences&&s.wikiReferences.length>0&&this.renderConfigRow(n,"Wiki access",s.wikiReferences.map(c=>c.agent).join(", ")),this.renderConfigRow(n,"Tags",s.tags.join(", ")||"none");let a=n.createDiv({cls:"af-config-prompt-section"});if(a.createDiv({cls:"af-slideover-section-title",text:"SYSTEM PROMPT"}),a.createDiv({cls:"af-output-block",text:s.body||"(empty)"}),s.heartbeatBody.trim()){let c=n.createDiv({cls:"af-config-prompt-section"});c.createDiv({cls:"af-slideover-section-title",text:"HEARTBEAT INSTRUCTION"}),c.createDiv({cls:"af-output-block",text:s.heartbeatBody})}let o=n.createDiv({cls:"af-slideover-actions"}).createEl("button",{cls:"af-btn-sm primary"});R(o,"edit","af-btn-icon"),o.appendText(" Edit Agent"),o.onclick=()=>this.navigate("edit-agent",s.name)}renderConfigRow(e,s,n){let a=e.createDiv({cls:"af-detail-row"});a.createSpan({cls:"af-detail-label",text:s}),a.createSpan({cls:"af-detail-value af-mono",text:n})}renderAgentRunsTab(e,s){if(s.length===0){this.renderEmptyState(e,"scroll-text","No runs yet","Run this agent to see history");return}for(let n of s){let a=e.createDiv({cls:"af-run-list-item"}),r=a.createDiv({cls:`af-tl-icon ${this.statusToTimelineClass(n.status)}`});(0,b.setIcon)(r,this.statusToIconName(n.status));let o=a.createDiv({cls:"af-tl-body"}),c=o.createDiv({cls:"af-tl-title"});c.createSpan({text:n.task}),c.createSpan({cls:`af-status-badge ${this.statusToBadgeClass(n.status)}`,text:this.statusToBadgeText(n.status)});let l=o.createDiv({cls:"af-tl-meta"});l.createSpan({text:`${this.formatStarted(n.started)} \xB7 ${this.formatDuration(n.durationSeconds)}`}),n.tokensUsed&&l.createSpan({text:`${n.tokensUsed.toLocaleString()} tokens`}),o.createDiv({cls:"af-tl-desc",text:jt(n.output,120)}),a.onclick=()=>this.openSlideover(n)}}async renderAgentMemoryTab(e,s){if(!s.memory){this.renderEmptyState(e,"file-text","Memory disabled","Enable memory in agent config");return}let n=await this.plugin.repository.readWorkingMemory(s.name),a=e.createDiv({cls:"af-form-help"}),r=n?.tokenEstimate??0,o=s.reflection.enabled?`reflection on (${s.reflection.schedule})`:"reflection off";a.setText(`~${r} / ${s.memoryTokenBudget} tokens \xB7 ${o}`),!n||n.sections.length===0?this.renderEmptyState(e,"file-text","No memories yet","Agent will learn from runs"):e.createDiv({cls:"af-output-block"}).setText(Ke(n.sections));let c=e.createDiv({cls:"af-slideover-actions"}),l=c.createEl("button",{cls:"af-btn-sm"});R(l,"moon","af-btn-icon"),l.appendText(" Reflect now"),l.onclick=async()=>{l.disabled=!0,l.setText(" Reflecting\u2026");let d=await this.plugin.runtime.runReflectionNow(s.name);new b.Notice(`Agent Fleet: ${d.message}`),await this.renderAgentMemoryTab((e.empty(),e),s)};let h=c.createEl("button",{cls:"af-btn-sm"});R(h,"external-link","af-btn-icon"),h.appendText(" Open in Editor"),h.onclick=()=>void this.plugin.openPath(this.plugin.repository.getWorkingMemoryPath(s.name))}timeAgo(e){let s=Math.round((Date.now()-e.getTime())/1e3);if(s<60)return"just now";let n=Math.round(s/60);if(n<60)return`${n}m ago`;let a=Math.round(n/60);return a<24?`${a}h ago`:`${Math.round(a/24)}d ago`}timeUntil(e){let s=Math.round((e.getTime()-Date.now())/1e3);if(s<60)return"< 1m";let n=Math.round(s/60);if(n<60)return`in ${n}m`;let a=Math.round(n/60);return a<24?`in ${a}h`:`in ${Math.round(a/24)}d`}renderKanbanPage(e){let s=e.createDiv({cls:"af-kanban-page"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.runtime.getRecentRuns(),r=s.createDiv({cls:"af-kanban-toolbar"});r.createDiv({cls:"af-page-title",text:"Tasks Board"}),r.createDiv({cls:"af-toolbar-spacer"});let o=r.createEl("button",{cls:"af-btn-sm primary"});R(o,"plus","af-btn-icon"),o.appendText(" New Task"),o.onclick=()=>this.navigate("create-task");let c=s.createDiv({cls:"af-kanban-board"}),l=[],h=[],d=[],u=[],p=[],m=new Set;for(let v of n.agents){let k=this.plugin.runtime.getAgentState(v.name);k.status==="running"&&k.currentTaskId&&m.add(k.currentTaskId)}let f=this.toLocalDateStr(new Date);for(let v of a)v.status==="success"&&this.runToLocalDate(v.started)===f?u.push(v):(v.status==="failure"||v.status==="timeout"||v.status==="cancelled")&&this.runToLocalDate(v.started)===f&&p.push(v);let g=new Set(u.map(v=>v.task)),y=new Set(p.map(v=>v.task));for(let v of n.tasks){let k=g.has(v.taskId)||y.has(v.taskId)||v.lastRun&&this.runToLocalDate(v.lastRun)===f;if(m.has(v.taskId))d.push(v);else{if(k&&!v.schedule)continue;v.schedule&&v.enabled?h.push(v):l.push(v)}}this.renderKanbanColumn(c,"Backlog","inbox",l.length,v=>{for(let k of l)this.renderKanbanTaskCard(v,k,n,!0)},"backlog"),this.renderKanbanColumn(c,"Scheduled","clock",h.length,v=>{for(let k of h)this.renderKanbanTaskCard(v,k,n,!0)},"scheduled"),this.renderKanbanColumn(c,"Running","loader-2",d.length,v=>{for(let k of d)this.renderKanbanRunningCard(v,k)},"running",!1,"running"),this.renderKanbanColumn(c,"Done","check-circle-2",u.length,v=>{for(let k of u.slice(0,10))this.renderKanbanCompletedCard(v,k)},"completed"),this.renderKanbanColumn(c,"Failed","x-circle",p.length,v=>{for(let k of p)this.renderKanbanFailedCard(v,k)},"failed",!1,"failed")}renderKanbanColumn(e,s,n,a,r,o,c=!1,l){let h=e.createDiv({cls:`af-kanban-column${l?` af-kanban-${l}`:""}`});(o==="backlog"||o==="scheduled")&&Vo(h,m=>this.handleTaskDrop(m,o));let u=h.createDiv({cls:"af-kanban-col-header"}).createDiv({cls:"af-kanban-col-title"});R(u,n),u.appendText(` ${s} `),u.createSpan({cls:"af-kanban-col-count",text:String(a)});let p=h.createDiv({cls:"af-kanban-col-body"});if(r(p),a===0&&p.createDiv({cls:"af-kanban-empty",text:"No items"}),c){let f=h.createDiv({cls:"af-kanban-col-add"}).createEl("button");R(f,"plus","af-btn-icon"),f.appendText(" Add task"),f.onclick=()=>{this.navigate("create-task")}}}handleTaskDrop(e,s){let n=this.plugin.runtime.getSnapshot().tasks.find(a=>a.taskId===e);if(n){if(s==="backlog")this.setTaskEnabled(n,!1).then(()=>{new b.Notice(`Task "${e}" moved to backlog (disabled)`)});else if(s==="scheduled"){if(!n.schedule&&!n.runAt){new b.Notice(`Task "${e}" needs a schedule. Open task details to set one.`),this.navigate("task-detail",e);return}this.setTaskEnabled(n,!0).then(()=>{new b.Notice(`Task "${e}" moved to scheduled (enabled)`)})}}}async setTaskEnabled(e,s){let n=this.plugin.app.vault.getAbstractFileByPath(e.filePath);if(!n||!(n instanceof b.TFile))return;let a=await this.plugin.app.vault.cachedRead(n),{frontmatter:r,body:o}=J(a);r.enabled=s,await this.plugin.app.vault.modify(n,H(r,o)),await this.plugin.refreshFromVault()}renderKanbanTaskCard(e,s,n,a){let r=e.createDiv({cls:`af-kanban-card af-priority-${s.priority}`});if(a){Go(r,s.taskId);let f=r.createDiv({cls:"af-kanban-card-grip"});(0,b.setIcon)(f,"grip-vertical")}let o=r.createDiv({cls:"af-kanban-card-header"});o.createDiv({cls:"af-kanban-card-title",text:s.taskId});let l=(n.agents.find(f=>f.name===s.agent)?.enabled??!1)&&s.enabled,h=o.createSpan({cls:`af-kanban-card-status ${l?"active":"inactive"}`});h.title=l?"Active":"Inactive";let d=r.createDiv({cls:"af-kanban-card-agent"}),u=d.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(u,"bot"),d.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let m=r.createDiv({cls:"af-kanban-card-footer"}).createSpan({cls:"af-kanban-card-schedule"});l?s.schedule?(R(m,"refresh-cw","af-meta-icon"),m.appendText(` ${this.humanizeCron(s.schedule)}`)):m.appendText(s.runAt??"Manual"):(R(m,"pause","af-meta-icon"),m.appendText(" Paused")),r.onclick=()=>this.navigate("task-detail",s.taskId)}renderKanbanRunningCard(e,s){let n=e.createDiv({cls:"af-kanban-card af-kanban-card-running"});n.createDiv({cls:"af-kanban-card-title",text:s.taskId});let a=n.createDiv({cls:"af-kanban-card-agent"}),r=a.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(r,"bot"),a.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let c=this.plugin.runtime.getSnapshot().agents.find(S=>S.name===s.agent)?.timeout??300,l=this.plugin.runtime.getAgentState(s.agent),h=l.runStarted?new Date(l.runStarted).getTime():Date.now(),p=n.createDiv({cls:"af-kanban-progress"}).createDiv({cls:"af-kanban-progress-track"}).createDiv({cls:"af-kanban-progress-bar af-kanban-progress-bar-real"}),m=(Date.now()-h)/1e3,f=Math.min(95,m/c*100);p.setCssStyles({width:`${f}%`});let g=n.createDiv({cls:"af-kanban-card-footer"}),y=g.createSpan({cls:"af-kanban-card-schedule"});R(y,"loader-2","af-meta-icon");let v=Math.round(m);y.appendText(` ${v}s / ${c}s`);let k=g.createEl("button",{cls:"af-kanban-stop-btn"});(0,b.setIcon)(k,"square"),k.title="Stop task",k.onclick=S=>{S.stopPropagation(),this.plugin.runtime.abortAgentRun(s.agent),new b.Notice(`Stopped task "${s.taskId}"`)};let w=window.setInterval(()=>{let S=(Date.now()-h)/1e3,T=Math.min(95,S/c*100);p.setCssStyles({width:`${T}%`});let _=Math.round(S);y.textContent="",(0,b.setIcon)(y,"loader-2"),y.appendText(` ${_}s / ${c}s`)},1e3);this.streamingUnsubscribes.push(()=>window.clearInterval(w))}renderKanbanCompletedCard(e,s){let n=e.createDiv({cls:"af-kanban-card"});n.createDiv({cls:"af-kanban-card-title",text:s.task});let a=n.createDiv({cls:"af-kanban-card-agent"}),r=a.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(r,"bot"),a.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let c=n.createDiv({cls:"af-kanban-card-footer"}).createSpan({cls:"af-kanban-card-schedule"});R(c,"check-circle-2","af-meta-icon"),c.appendText(` ${this.formatStarted(s.started)} \xB7 ${this.formatDuration(s.durationSeconds)}`),n.onclick=()=>this.openSlideover(s)}renderKanbanFailedCard(e,s){let n=s.status==="cancelled",a=e.createDiv({cls:`af-kanban-card ${n?"af-kanban-card-cancelled":"af-kanban-card-failed"}`});a.createDiv({cls:"af-kanban-card-title",text:s.task});let r=a.createDiv({cls:"af-kanban-card-agent"}),o=r.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(o,"bot"),r.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let c=n?`Stopped after ${s.durationSeconds}s`:s.status==="timeout"?`Timeout after ${s.durationSeconds}s`:jt(s.output,60),l=a.createDiv({cls:"af-kanban-card-error"});R(l,n?"square":"alert-triangle","af-meta-icon"),l.appendText(` ${c}`);let h=a.createDiv({cls:"af-kanban-card-footer"}),d=h.createSpan({cls:"af-kanban-card-schedule"});if(R(d,n?"square":"x-circle","af-meta-icon"),d.appendText(` ${this.formatStarted(s.started)}`),!n){let u=h.createEl("button",{cls:"af-btn-sm"});R(u,"refresh-cw","af-btn-icon"),u.appendText(" Retry"),u.onclick=p=>{p.stopPropagation(),this.plugin.runAgentPrompt(s.agent)}}a.onclick=()=>this.openSlideover(s)}renderRunsPage(e){let s=e.createDiv({cls:"af-runs-page"}),n=this.plugin.runtime.getRecentRuns(),a=s.createDiv({cls:"af-runs-toolbar"});a.createDiv({cls:"af-page-title",text:"Run History"}),a.createDiv({cls:"af-toolbar-spacer"});let r=s.createDiv({cls:"af-runs-table"});if(n.length===0){this.renderEmptyState(r,"scroll-text","No runs yet","Run an agent to see history here");return}let o=r.createEl("table"),l=o.createEl("thead").createEl("tr");for(let d of["Status","Agent","Task","Started","Duration","Tokens","Model"])l.createEl("th",{text:d});let h=o.createEl("tbody");for(let d of n.slice(0,50))this.renderRunRow(h,d)}renderRunRow(e,s){let n=e.createEl("tr"),r=n.createEl("td").createSpan({cls:`af-status-badge ${this.statusToBadgeClass(s.status)}`}),o=r.createSpan();(0,b.setIcon)(o,this.statusToIconName(s.status)),r.appendText(` ${this.statusToBadgeText(s.status)}`);let c=n.createEl("td",{cls:"af-agent-link"});c.setText(s.agent),c.onclick=l=>{l.stopPropagation(),this.navigate("agent-detail",s.agent)},n.createEl("td",{text:s.task}),n.createEl("td",{cls:"af-mono",text:this.formatStarted(s.started)}),n.createEl("td",{cls:"af-mono",text:this.formatDuration(s.durationSeconds)}),n.createEl("td",{cls:"af-mono",text:s.tokensUsed?s.tokensUsed.toLocaleString():"\u2014"}),n.createEl("td",{cls:"af-mono",text:s.model}),n.setCssStyles({cursor:"pointer"}),n.onclick=()=>this.openSlideover(s)}renderSkillsPage(e){let s=e.createDiv({cls:"af-skills-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Skills Library"}),a.createDiv({cls:"af-toolbar-spacer"});let r=a.createEl("button",{cls:"af-btn-sm primary"});R(r,"plus","af-btn-icon"),r.appendText(" New Skill"),r.onclick=()=>void this.plugin.createSkillTemplate();let o=s.createDiv({cls:"af-skills-grid"});if(n.skills.length===0){this.renderEmptyState(o,"puzzle","No skills yet","Create skills to give agents specialized abilities");return}for(let c of n.skills)this.renderSkillCard(o,c,n.agents)}renderSkillCard(e,s,n){let a=e.createDiv({cls:"af-skill-card"}),r=a.createDiv({cls:"af-skill-card-header"}),o=r.createDiv({cls:"af-skill-card-icon"});(0,b.setIcon)(o,this.getSkillIcon(s.name));let c=r.createEl("button",{cls:"af-btn-sm af-btn-xs"});R(c,"edit","af-btn-icon"),c.onclick=h=>{h.stopPropagation(),this.navigate("edit-skill",s.name)},a.createDiv({cls:"af-skill-card-name",text:s.name}),a.createDiv({cls:"af-skill-card-desc",text:s.description??"No description"});let l=n.filter(h=>h.skills.includes(s.name));if(l.length>0){let h=a.createDiv({cls:"af-skill-card-agents"});for(let d of l)h.createSpan({cls:"af-skill-card-agent-tag",text:d.name})}}renderChannelsPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Channels"}),a.createDiv({cls:"af-toolbar-spacer"});let r=a.createEl("button",{cls:"af-btn-sm primary"});R(r,"plus","af-btn-icon"),r.appendText(" New Channel"),r.onclick=()=>this.navigate("create-channel");let o=s.createDiv({cls:"af-agents-grid"});if(n.channels.length===0){this.renderEmptyState(o,"radio","No channels configured","Connect an agent to Slack or another chat platform");return}for(let c of n.channels)this.renderChannelCard(o,c,n.validationIssues)}async renderWikiKeepersPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Wiki Keepers"}),a.createDiv({cls:"af-toolbar-spacer"});let r=n.agents.filter(c=>c.wikiKeeper!==void 0);if(r.length===0){this.renderEmptyState(s,"library","No Wiki Keepers yet","Open Settings \u2192 Agent Fleet \u2192 Wiki Keepers \u2192 + Add to create one.");return}let o=s.createDiv({cls:"af-wk-list"});o.setCssStyles({display:"flex"}),o.setCssStyles({flexDirection:"column"}),o.setCssStyles({gap:"16px"});for(let c of r)await this.renderWikiKeeperCard(o,c)}async renderWikiKeeperCard(e,s){let n=s.wikiKeeper,a=e.createDiv({cls:"af-card"});a.setCssStyles({padding:"16px"}),a.setCssStyles({border:"1px solid var(--background-modifier-border)"}),a.setCssStyles({borderRadius:"8px"});let r=a.createDiv();r.setCssStyles({display:"flex"}),r.setCssStyles({alignItems:"center"}),r.setCssStyles({gap:"12px"}),r.setCssStyles({marginBottom:"12px"});let o=r.createDiv();o.setCssStyles({flex:"1"}),o.createEl("strong",{text:s.name});let c=n.scopeRoot||"(whole vault)";o.createEl("div",{text:`Scope: ${c} \xB7 topics: ${n.topicsRoot}/ \xB7 log: ${n.logPath}`,cls:"af-form-hint"});let l=r.createEl("button",{cls:"af-btn-sm"});l.appendText("Open log"),l.onclick=()=>{let g=n.scopeRoot?`${n.scopeRoot}/${n.logPath}`:n.logPath,y=this.plugin.app.vault.getAbstractFileByPath(g);y instanceof b.TFile?this.plugin.app.workspace.getLeaf().openFile(y):new b.Notice(`Log file not found: ${g}`)};let h=n.scopeRoot?`${n.scopeRoot}/${n.logPath}`:n.logPath,d=this.plugin.app.vault.getAbstractFileByPath(h),u=null;if(d instanceof b.TFile)try{let g=await this.plugin.app.vault.cachedRead(d);u=Yo(g)}catch{u=null}if(!u){a.createDiv({cls:"af-form-hint"}).setText("No lint report yet. Run wiki-lint manually or wait for the weekly task to fire.");return}let p=a.createDiv();if(p.setCssStyles({display:"flex"}),p.setCssStyles({alignItems:"baseline"}),p.setCssStyles({gap:"12px"}),p.setCssStyles({marginBottom:"8px"}),p.createEl("strong",{text:`Lint ${u.date}`}),p.createSpan({cls:"af-form-hint",text:`${u.summary.length} summary lines \xB7 ${u.autoApplied.length} auto-applied \xB7 ${u.needsReview.length} needs review`}),u.summary.length>0){let g=a.createEl("details");g.createEl("summary",{text:"Summary"});let y=g.createEl("ul");y.setCssStyles({marginTop:"4px"});for(let v of u.summary)y.createEl("li",{text:v})}if(u.autoApplied.length>0){let g=a.createEl("details");g.createEl("summary",{text:`Auto-applied (${u.autoApplied.length})`});let y=g.createEl("ul");y.setCssStyles({marginTop:"4px"});for(let v of u.autoApplied)y.createEl("li",{text:v})}if(u.refreshChained.length>0){let g=a.createEl("details");g.createEl("summary",{text:`Refresh chained (${u.refreshChained.length})`});let y=g.createEl("ul");y.setCssStyles({marginTop:"4px"});for(let v of u.refreshChained)y.createEl("li",{text:v})}let m=a.createDiv();if(m.setCssStyles({marginTop:"12px"}),m.createEl("strong",{text:`Needs review (${u.needsReview.length})`}),u.needsReview.length===0){m.createDiv({cls:"af-form-hint",text:"All clear. Nothing requires manual review from this lint pass."});return}let f=m.createDiv();f.setCssStyles({display:"flex"}),f.setCssStyles({flexDirection:"column"}),f.setCssStyles({gap:"6px"}),f.setCssStyles({marginTop:"8px"});for(let g of u.needsReview){let y=f.createDiv();y.setCssStyles({display:"flex"}),y.setCssStyles({alignItems:"flex-start"}),y.setCssStyles({gap:"8px"}),y.setCssStyles({padding:"8px 10px"}),y.setCssStyles({background:"var(--background-secondary)"}),y.setCssStyles({borderRadius:"4px"}),y.setCssStyles({fontSize:"13px"});let v=y.createDiv();v.setCssStyles({flex:"1"}),v.setText(g);let k=y.createEl("button",{cls:"af-btn-sm",text:"Dismiss"});k.title="Hide this item from the dashboard until the next lint pass (does not modify log.md).",k.onclick=()=>{y.remove()}}}renderChannelCard(e,s,n){let a=this.plugin.channelManager?.getChannelStatus(s.name)??"disabled",r=Xo(a),o=s.enabled&&a!=="disabled"?"af-agent-card":"af-agent-card disabled",c=e.createDiv({cls:o});c.setCssStyles({cursor:"default"});let l=c.createDiv({cls:"af-agent-card-header"}),h=l.createDiv({cls:`af-agent-card-avatar ${r}`});(0,b.setIcon)(h,"radio");let d=l.createDiv({cls:"af-agent-card-titleblock"});d.createDiv({cls:"af-agent-card-name",text:s.name}),d.createDiv({cls:"af-agent-card-desc",text:`Default: ${s.defaultAgent}`});let u=l.createSpan({cls:`af-pill ${xh(a)}`});if(u.createSpan({cls:"af-dot"}),u.appendText(` ${a}`),s.allowedAgents.length>0){let T=c.createDiv({cls:"af-agent-card-skills"});for(let _ of s.allowedAgents){let D=T.createSpan({cls:"af-skill-tag",text:_});_===s.defaultAgent&&D.setCssStyles({fontWeight:"700"})}}let p=c.createDiv({cls:"af-agent-card-stats"}),m=this.plugin.channelManager?.getSessionCount(s.name)??0,f=this.plugin.channelManager?.getMetrics(s.name),g=s.allowedAgents.length>0?String(s.allowedAgents.length):"all";this.renderAgentStat(p,g,"Agents"),this.renderAgentStat(p,String(m),"Sessions"),this.renderAgentStat(p,String(f?.messagesReceived??0),"In"),this.renderAgentStat(p,String(f?.messagesSent??0),"Out");let y=c.createDiv({cls:"af-agent-card-footer"}),v=[s.type];s.enabled||v.push("disabled"),s.allowedUsers.length>0?v.push(`${s.allowedUsers.length} user(s)`):v.push("allowlist empty"),y.createSpan({cls:"af-agent-card-meta",text:v.join(" \xB7 ")});let w=y.createDiv({cls:"af-agent-card-actions"}).createEl("button",{cls:"af-btn-sm"});R(w,"edit","af-btn-icon"),w.appendText(" Edit"),w.onclick=T=>{T.stopPropagation(),this.navigate("edit-channel",s.name)};let S=n.filter(T=>T.path===s.filePath);if(S.length>0){let T=c.createDiv({cls:"af-channel-issues"});for(let _ of S)T.createDiv({cls:"af-channel-issue-row",text:_.message})}}renderCreateChannelPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.channelCredentials.list(),r=s.createDiv({cls:"af-detail-header"}),o=r.createDiv({cls:"af-detail-header-left"}),c=o.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(c,"plus");let l=o.createDiv();l.createDiv({cls:"af-detail-header-name",text:"Create New Channel"}),l.createDiv({cls:"af-detail-header-desc",text:"Connect an external chat transport to an agent"});let h=r.createDiv({cls:"af-detail-header-actions"}),d={name:"",type:"slack",defaultAgent:n.agents[0]?.name??"",allowedAgents:[],credentialRef:a[0]?.ref??"",allowedUsers:"",perUserSessions:!0,channelContext:"",enabled:!0,tags:"",body:"",transportJson:""},u=s.createDiv({cls:"af-create-form"}),p=u.createDiv({cls:"af-create-section"}),m=p.createDiv({cls:"af-create-section-header"}),f=m.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(f,"radio"),m.createSpan({text:"Channel Details"}),this.createFormField(p,"Name","my-slack","Unique identifier for this channel",j=>{d.name=j});let g=p.createDiv({cls:"af-form-row"});g.createDiv({cls:"af-form-label",text:"Type"});let y=g.createEl("select",{cls:"af-form-select"});y.createEl("option",{text:"slack",attr:{value:"slack"}}),y.createEl("option",{text:"telegram",attr:{value:"telegram"}}),y.createEl("option",{text:"discord",attr:{value:"discord"}}),y.addEventListener("change",()=>{d.type=y.value});let v=p.createDiv({cls:"af-form-row"}),k=v.createDiv({cls:"af-form-label"});k.setText("Credential"),this.addTooltip(k,"Configured in Settings \u2192 Agent Fleet \u2192 Channel Credentials");let w=v.createEl("select",{cls:"af-form-select"});a.length===0&&w.createEl("option",{text:"(no credentials configured)",attr:{value:""}});for(let j of a)w.createEl("option",{text:`${j.ref} (${j.entry.type})`,attr:{value:j.ref}});w.addEventListener("change",()=>{d.credentialRef=w.value});let S=p.createDiv({cls:"af-form-row af-form-row-toggle"});S.createDiv({cls:"af-form-label",text:"Enabled"});let T=S.createDiv({cls:"af-agent-card-toggle on"});T.onclick=()=>{let j=T.hasClass("on");T.toggleClass("on",!j),d.enabled=!j};let _=u.createDiv({cls:"af-create-section"}),D=_.createDiv({cls:"af-create-section-header"}),O=D.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(O,"bot"),D.createSpan({text:"Agent Routing"});let C=_.createDiv({cls:"af-form-row"}),E=C.createDiv({cls:"af-form-label"});E.setText("Default agent"),this.addTooltip(E,"Used when no @agent-name prefix is given in a message");let P=C.createEl("select",{cls:"af-form-select"});for(let j of n.agents)P.createEl("option",{text:j.name,attr:{value:j.name}});P.addEventListener("change",()=>{d.defaultAgent=P.value});let N=_.createDiv({cls:"af-form-row"}),M=N.createDiv({cls:"af-form-label"});M.setText("Allowed agents"),this.addTooltip(M,"Agents reachable via @prefix. Leave unchecked to allow all agents.");let U=N.createDiv({cls:"af-form-checkboxes"});for(let j of n.agents){let Ce=U.createEl("label",{cls:"af-form-checkbox-label"}),xe=Ce.createEl("input",{attr:{type:"checkbox",value:j.name}});Ce.appendText(` ${j.name}`),xe.addEventListener("change",()=>{xe.checked?d.allowedAgents.includes(j.name)||d.allowedAgents.push(j.name):d.allowedAgents=d.allowedAgents.filter(he=>he!==j.name)})}let $=_.createDiv({cls:"af-form-row af-form-row-toggle"}),Z=$.createDiv({cls:"af-form-label"});Z.setText("Per-user sessions"),this.addTooltip(Z,"Each external user gets their own isolated Claude session");let de=$.createDiv({cls:"af-agent-card-toggle on"});de.onclick=()=>{let j=de.hasClass("on");de.toggleClass("on",!j),d.perUserSessions=!j};let me=u.createDiv({cls:"af-create-section"}),ge=me.createDiv({cls:"af-create-section-header"}),X=ge.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(X,"shield-check"),ge.createSpan({text:"Access Control"});let W=me.createDiv({cls:"af-form-label"});W.setText("Allowed users"),this.addTooltip(W,"User IDs, one per line \u2014 Slack (U...), Telegram (numeric), or Discord (numeric snowflakes). Only listed users can reach the bot.");let K=me.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`U0AQW6P37N1
11949
+ `,t)),n<t/2&&(n=t),e.push(s.slice(0,n)),s=s.slice(n).replace(/^\n+/,"")}return s&&e.push(s),e}var Ms=require("obsidian");var Un=class extends Ms.ItemView{constructor(e,s){super(e);this.plugin=s}getViewType(){return jt}getDisplayText(){return"Agent Fleet"}getIcon(){return"bot"}async onOpen(){this.plugin.subscribeView(this),await this.render()}async onClose(){this.plugin.unsubscribeView(this)}async render(){let e=this.contentEl;e.empty(),e.addClass("af-sidebar");let s=this.plugin.runtime.getSnapshot(),n=this.plugin.runtime.getFleetStatus(),a=e.createDiv({cls:"af-sidebar-section"});a.createDiv({cls:"af-sidebar-section-header",text:"AGENT FLEET"});let i=[{icon:"layout-dashboard",label:"Dashboard",page:"dashboard"},{icon:"bot",label:"Agents",page:"agents",badge:()=>s.agents.length},{icon:"columns-3",label:"Tasks Board",page:"kanban"},{icon:"scroll-text",label:"Run History",page:"runs"},{icon:"shield-check",label:"Approvals",page:"approvals",badge:()=>n.pending},{icon:"puzzle",label:"Skills",page:"skills",badge:()=>s.skills.length},{icon:"plug",label:"MCP Servers",page:"mcp",badge:()=>this.plugin.repository.getMcpServers().length},{icon:"radio",label:"Channels",page:"channels",badge:()=>this.plugin.channelManager?.getConnectedCount()??s.channels.length}];for(let l of i){let h=a.createDiv({cls:"af-sidebar-nav-item"}),d=h.createSpan({cls:"af-sidebar-nav-icon"});(0,Ms.setIcon)(d,l.icon),h.createSpan({cls:"af-sidebar-nav-label",text:l.label});let u=l.badge?.();u!==void 0&&u>0&&h.createSpan({cls:"af-sidebar-badge",text:String(u)}),h.setAttribute("role","button"),h.setAttribute("tabindex","0"),h.onclick=()=>void this.plugin.navigateDashboard(l.page),h.onkeydown=p=>{(p.key==="Enter"||p.key===" ")&&(p.preventDefault(),this.plugin.navigateDashboard(l.page))}}let o=e.createDiv({cls:"af-sidebar-section"});o.createDiv({cls:"af-sidebar-section-header",text:"AGENTS"}),s.agents.length===0&&o.createDiv({cls:"af-sidebar-empty",text:"No agents configured"});for(let l of s.agents){let h=this.plugin.runtime.getAgentState(l.name),d=o.createDiv({cls:"af-sidebar-agent-item"});d.createSpan({cls:`af-sidebar-agent-dot ${this.healthToClass(h.status)}`}),d.createSpan({cls:"af-sidebar-agent-name",text:l.name}),d.setAttribute("role","button"),d.setAttribute("tabindex","0"),d.onclick=()=>void this.plugin.navigateDashboard("agent-detail",l.name),d.onkeydown=u=>{(u.key==="Enter"||u.key===" ")&&(u.preventDefault(),this.plugin.navigateDashboard("agent-detail",l.name))}}let c=e.createDiv({cls:"af-sidebar-section"});c.createDiv({cls:"af-sidebar-section-header",text:"QUICK ACTIONS"}),this.renderQuickAction(c,"plus","New Agent",()=>void this.plugin.createAgentTemplate()),this.renderQuickAction(c,"plus","New Task",()=>void this.plugin.openCreateTask()),this.renderQuickAction(c,"plus","New Skill",()=>void this.plugin.createSkillTemplate())}renderQuickAction(e,s,n,a){let i=e.createDiv({cls:"af-sidebar-action-item"}),o=i.createSpan({cls:"af-sidebar-action-icon"});(0,Ms.setIcon)(o,s),i.createSpan({text:n}),i.setAttribute("role","button"),i.setAttribute("tabindex","0"),i.onclick=a,i.onkeydown=c=>{(c.key==="Enter"||c.key===" ")&&(c.preventDefault(),a())}}healthToClass(e){switch(e){case"running":return"running";case"error":return"error";case"pending":return"pending";case"disabled":return"disabled";default:return"idle"}}};var b=require("obsidian");var rs=require("obsidian"),gh=["bot","brain","shield-check","search","file-text","rocket","wand","sparkles","zap","target","compass","eye","code","terminal","database","globe","mail","message-circle","book","pen-tool","palette","music","camera","chart-bar","clipboard","cpu","server","cloud","lock","key","bell","calendar","clock","heart","star","flag","bookmark"],$n=class extends rs.Modal{constructor(e,s,n){super(e);this.onSelect=n;this.selectedIcon=s}searchQuery="";selectedIcon;allIcons=[];gridContainer;onOpen(){this.allIcons=(0,rs.getIconIds)().sort();let{contentEl:e}=this;e.empty(),e.addClass("af-icon-picker-modal");let s=e.createEl("input",{cls:"af-icon-picker-search",attr:{type:"text",placeholder:"Search icons...",value:this.searchQuery}});this.gridContainer=e.createDiv({cls:"af-icon-picker-scroll"}),this.renderGrid(),s.addEventListener("input",()=>{this.searchQuery=s.value,this.renderGrid()}),window.setTimeout(()=>s.focus(),0)}renderGrid(){let e=this.gridContainer;e.empty();let s=this.searchQuery.toLowerCase().trim();if(!s)this.renderSection(e,"Popular",gh),this.renderSection(e,"All Icons",this.allIcons);else{let n=this.allIcons.filter(i=>i.includes(s)),a=n.length===0?"No results":`${n.length} result${n.length!==1?"s":""}`;this.renderSection(e,a,n)}}renderSection(e,s,n){let a=e.createDiv({cls:"af-icon-picker-section"});a.createDiv({cls:"af-icon-picker-section-label",text:s});let i=a.createDiv({cls:"af-icon-picker-grid"});for(let o of n)this.renderIconItem(i,o)}renderIconItem(e,s){let n=e.createDiv({cls:`af-icon-picker-item${this.selectedIcon===s?" selected":""}`});n.setAttribute("title",s),n.setAttribute("aria-label",s),(0,rs.setIcon)(n,s),n.addEventListener("click",()=>{this.selectedIcon=s,this.onSelect(s),this.close()})}onClose(){this.contentEl.empty()}};var yh=[{match:/opus/i,rates:{input:15,output:75,cacheWrite:18.75,cacheRead:1.5}},{match:/sonnet/i,rates:{input:3,output:15,cacheWrite:3.75,cacheRead:.3}},{match:/haiku/i,rates:{input:1,output:5,cacheWrite:1.25,cacheRead:.1}},{match:/gpt-5|codex|o[0-9]/i,rates:{input:1.25,output:10,cacheWrite:1.25,cacheRead:.125}}],vh={input:3,output:15,cacheWrite:3.75,cacheRead:.3};function jo(r){for(let{match:t,rates:e}of yh)if(t.test(r))return e;return vh}function Wo(r,t){let e=jo(r);return(t.inputTokens*e.input+t.outputTokens*e.output+t.cacheCreateTokens*e.cacheWrite+t.cacheReadTokens*e.cacheRead)/1e6}function Ho(r,t){let e=jo(r),s=.7*e.input+.3*e.output;return t*s/1e6}var qo=require("obsidian");function R(r,t,e){let s=r.createSpan({cls:e??"af-icon"});return(0,qo.setIcon)(s,t),s}function Ze(r,t={}){let e=activeDocument.createElementNS("http://www.w3.org/2000/svg",r);for(let[s,n]of Object.entries(t))e.setAttribute(s,n);return e}function wh(r){try{return new Date(r+"T12:00:00").toLocaleDateString(void 0,{month:"short",day:"numeric"})}catch{return r.slice(5)}}function zo(r,t){let e=t.length||1,s=1e3,n=6,a=4,i=4,o=s-a-i-n*(e-1),c=Math.floor(o/e),l=140,h=36,d=18,u=d+l+h,p=Math.max(1,...t.map(f=>f.success+f.failure+f.cancelled)),m=Ze("svg",{viewBox:`0 0 ${s} ${u}`,width:"100%",height:String(u),class:"af-chart-bar"});for(let f=0;f<=4;f++){let g=d+l-f/4*l;m.appendChild(Ze("line",{x1:String(a),y1:String(g),x2:String(s-i),y2:String(g),stroke:"var(--af-text-faint)","stroke-opacity":"0.15","stroke-width":"1"}))}for(let f=0;f<t.length;f++){let g=t[f],y=a+f*(c+n),v=g.success+g.failure+g.cancelled,k=v/p*l,w=g.success/p*l,S=g.cancelled/p*l,T=g.failure/p*l;if(g.success>0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-w),width:String(c),height:String(Math.max(w,2)),fill:"var(--af-green)",opacity:"0.85"})),g.cancelled>0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-w-S),width:String(c),height:String(Math.max(S,2)),fill:"var(--af-yellow)",opacity:"0.85"})),g.failure>0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-k),width:String(c),height:String(Math.max(T,2)),fill:"var(--af-red)",opacity:"0.85"})),v===0&&m.appendChild(Ze("rect",{x:String(y),y:String(d+l-3),width:String(c),height:"3",rx:"1.5",fill:"var(--af-text-faint)",opacity:"0.2"})),v>0){let D=Ze("text",{x:String(y+c/2),y:String(d+l-k-5),"text-anchor":"middle","font-size":"10","font-weight":"600",fill:"var(--af-text-secondary)"});D.textContent=String(v),m.appendChild(D)}let _=Ze("text",{x:String(y+c/2),y:String(d+l+16),"text-anchor":"middle","font-size":"10",fill:"var(--af-text-muted)"});_.textContent=wh(g.date),m.appendChild(_)}r.appendChild(m)}function Go(r,t,e){let c=2*Math.PI*46,l=e>0?t/e:0,h=c*l,d=c-h,u=Ze("svg",{viewBox:"0 0 130 130",width:String(130),height:String(130),class:"af-chart-donut"});u.appendChild(Ze("circle",{cx:String(65),cy:String(65),r:String(46),fill:"none",stroke:e>0?"var(--af-red)":"var(--af-text-faint)","stroke-width":String(12),opacity:"0.2"})),l>0&&u.appendChild(Ze("circle",{cx:String(65),cy:String(65),r:String(46),fill:"none",stroke:"var(--af-green)","stroke-width":String(12),"stroke-dasharray":`${h} ${d}`,"stroke-dashoffset":String(c*.25),"stroke-linecap":"round"}));let p=Ze("text",{x:String(65),y:String(61),"text-anchor":"middle","dominant-baseline":"middle","font-size":"24","font-weight":"700",fill:"var(--af-text-primary)"});p.textContent=`${Math.round(l*100)}%`,u.appendChild(p);let m=Ze("text",{x:String(65),y:String(83),"text-anchor":"middle","font-size":"10",fill:"var(--af-text-muted)"});m.textContent=`${t}/${e} runs`,u.appendChild(m),r.appendChild(u)}function Vo(r,t){r.draggable=!0,r.addEventListener("dragstart",e=>{e.dataTransfer?.setData("text/plain",t),r.addClass("af-dragging")}),r.addEventListener("dragend",()=>{r.removeClass("af-dragging")})}function Yo(r,t){r.addEventListener("dragover",e=>{e.preventDefault(),r.addClass("af-drag-over")}),r.addEventListener("dragleave",e=>{let s=e.relatedTarget;(!s||!r.contains(s))&&r.removeClass("af-drag-over")}),r.addEventListener("drop",e=>{e.preventDefault();let s=e.dataTransfer?.getData("text/plain");r.removeClass("af-drag-over"),s&&t(s)})}function Ko(r){let t=/^##\s+Lint\s+(\d{4}-\d{2}-\d{2})\s*$/gm,e,s=-1,n="";for(;(e=t.exec(r))!==null;)e.index>s&&(s=e.index,n=e[1]??"");if(s<0)return null;let a=r.slice(s),i=a.search(/\n##\s+(?!\s*#)/),o=i<0?a:a.slice(0,i);return{date:n,summary:jn(o,"Summary"),autoApplied:jn(o,"Auto-applied"),needsReview:jn(o,"Needs review"),refreshChained:jn(o,"Refresh chained")}}function jn(r,t){let e=t.replace(/[.*+?^${}()|[\]\\]/g,"\\$&"),n=new RegExp(`^###\\s+${e}\\s*$`,"m").exec(r);if(!n)return[];let a=n.index+n[0].length,i=r.slice(a),o=i.search(/\n###\s+/),l=(o<0?i:i.slice(0,o)).split(/\r?\n/).map(d=>d.trimEnd()),h=[];for(let d of l){let u=d.replace(/^\s+/,"");u.startsWith("- ")?h.push(u.slice(2).trim()):h.length>0&&(d.startsWith(" ")||d.startsWith(" "))&&(h[h.length-1]+=" "+u)}return h}var Jo={dashboard:"Dashboard",agents:"Agents",kanban:"Tasks Board",runs:"Run History",skills:"Skills Library",approvals:"Approvals",mcp:"MCP Servers",channels:"Channels","wiki-keepers":"Wiki Keepers","agent-detail":"Agent Details","task-detail":"Task Details","create-agent":"Create Agent","create-task":"Create Task","create-skill":"Create Skill","edit-agent":"Edit Agent","edit-task":"Edit Task","edit-skill":"Edit Skill","create-channel":"Create Channel","edit-channel":"Edit Channel","add-mcp-server":"Add MCP Server"},bh={dashboard:"layout-dashboard",agents:"bot",kanban:"columns-3",runs:"scroll-text",skills:"puzzle",approvals:"shield-check",mcp:"plug",channels:"radio","wiki-keepers":"library","agent-detail":"bot","task-detail":"circle-dot","create-agent":"plus","create-task":"plus","create-skill":"plus","edit-agent":"edit","edit-task":"edit","edit-skill":"edit","create-channel":"plus","edit-channel":"edit","add-mcp-server":"plus"},kh=["dashboard","agents","kanban","runs","approvals","skills","wiki-keepers","mcp","channels"],Xo=[["claude-code","Claude Code",!1],["codex","Codex",!1],["process","Process (coming soon)",!0],["http","HTTP (coming soon)",!0]],Zo=[["bypassPermissions","Bypass Permissions","Auto-approve everything except deny list"],["dontAsk","Don\u2019t Ask","Auto-approve all tool calls"],["acceptEdits","Accept Edits","Auto-approve file edits, block bash unless allowed"],["plan","Plan","Read-only mode, no writes or commands"],["default","Default","Ask for each tool call"]],el=[["bypassPermissions","Bypass (no sandbox)","No sandbox, auto-approve everything"],["workspace-write","Workspace Write","Sandboxed: writes only inside the working dir"],["read-only","Read Only","Sandboxed: no writes or side-effect commands"]];function Ls(r){let t=r.trim().toLowerCase();return t==="codex"||t==="openai-codex"}function Wn(r){return Ls(r)?el:Zo}function Hn(r,t){if(Ls(t))switch(r){case"acceptEdits":case"default":return"workspace-write";case"plan":return"read-only";case"dontAsk":return"bypassPermissions";default:return el.some(([e])=>e===r)?r:"bypassPermissions"}switch(r){case"workspace-write":return"acceptEdits";case"read-only":return"plan";case"danger-full-access":return"bypassPermissions";default:return Zo.some(([e])=>e===r)?r:"bypassPermissions"}}var Fs=class extends b.ItemView{constructor(e,s){super(e);this.plugin=s}currentPage="dashboard";detailContext;agentDetailTab="overview";streamingUnsubscribes=[];channelStatusUnsubscribe;authenticatingServers=new Set;getViewType(){return _t}getDisplayText(){return"Agent Fleet"}getIcon(){return"bot"}async onOpen(){this.plugin.subscribeView(this),this.channelStatusUnsubscribe=this.plugin.channelManager?.onStatusChange(()=>{this.currentPage==="channels"&&this.render()}),await this.render()}async onClose(){this.cleanupStreaming(),this.channelStatusUnsubscribe?.(),this.channelStatusUnsubscribe=void 0,this.plugin.unsubscribeView(this)}navigateTo(e,s){this.currentPage=e,this.detailContext=s,e!=="agent-detail"&&(this.agentDetailTab="overview"),this.render()}async render(){this.cleanupStreaming();let e=this.contentEl;e.empty(),e.addClass("af-root");let n=e.createDiv({cls:"af-app"}).createDiv({cls:"af-main-content"});this.renderTopBar(n),this.renderTabBar(n);let a=n.createDiv({cls:"af-page"});switch(this.currentPage){case"dashboard":this.renderDashboardPage(a);break;case"agents":this.renderAgentsPage(a);break;case"kanban":this.renderKanbanPage(a);break;case"runs":this.renderRunsPage(a);break;case"skills":this.renderSkillsPage(a);break;case"approvals":this.renderApprovalsPage(a);break;case"mcp":this.renderMcpPage(a);break;case"channels":this.renderChannelsPage(a);break;case"wiki-keepers":this.renderWikiKeepersPage(a);break;case"agent-detail":this.renderAgentDetailPage(a);break;case"task-detail":this.renderTaskDetailPage(a);break;case"create-agent":this.renderCreateAgentPage(a);break;case"create-task":this.renderCreateTaskPage(a);break;case"create-skill":this.renderCreateSkillPage(a);break;case"edit-agent":this.renderEditAgentPage(a);break;case"edit-task":this.renderEditTaskPage(a);break;case"edit-skill":this.renderEditSkillPage(a);break;case"create-channel":this.renderCreateChannelPage(a);break;case"edit-channel":this.renderEditChannelPage(a);break;case"add-mcp-server":this.renderAddMcpServerPage(a);break}}navigate(e,s){this.navigateTo(e,s)}cleanupStreaming(){for(let e of this.streamingUnsubscribes)e();this.streamingUnsubscribes=[]}renderTopBar(e){let s=e.createDiv({cls:"af-top-bar"}),n=s.createDiv({cls:"af-top-bar-title"});R(n,"bot","af-top-bar-icon"),n.createSpan({text:"Agent Fleet"});let a=s.createDiv({cls:"af-breadcrumb"}),i=a.createSpan({cls:"af-breadcrumb-sep"});(0,b.setIcon)(i,"chevron-right");let o=(m,f,g)=>{let y=a.createSpan({cls:f?"af-breadcrumb-link":"",text:m});f&&(y.onclick=()=>this.navigate(f,g))},c=()=>{let m=a.createSpan({cls:"af-breadcrumb-sep"});(0,b.setIcon)(m,"chevron-right")};switch(this.currentPage){case"agent-detail":o("Agents","agents"),c(),o(this.detailContext??"Agent");break;case"task-detail":o("Tasks Board","kanban"),c(),o(this.detailContext??"Task");break;case"edit-agent":o("Agents","agents"),c(),o(this.detailContext??"Agent","agent-detail",this.detailContext),c(),o("Edit");break;case"edit-task":o("Tasks Board","kanban"),c(),o(this.detailContext??"Task","task-detail",this.detailContext),c(),o("Edit");break;case"create-agent":o("Agents","agents"),c(),o("New Agent");break;case"create-task":o("Tasks Board","kanban"),c(),o("New Task");break;case"create-skill":o("Skills Library","skills"),c(),o("New Skill");break;case"edit-skill":o("Skills Library","skills"),c(),o(this.detailContext??"Skill"),c(),o("Edit");break;case"create-channel":o("Channels","channels"),c(),o("New Channel");break;case"edit-channel":o("Channels","channels"),c(),o(this.detailContext??"Channel"),c(),o("Edit");break;case"add-mcp-server":o("MCP Servers","mcp"),c(),o("Add Server");break;default:o(Jo[this.currentPage])}s.createDiv({cls:"af-top-bar-spacer"});let l=s.createDiv({cls:"af-search-wrap"});R(l,"search","af-search-icon");let h=l.createEl("input",{cls:"af-search-input",attr:{type:"text",placeholder:"Search agents, tasks, runs..."}});h.addEventListener("input",()=>{this.handleSearch(h.value,l)}),h.addEventListener("blur",()=>{window.setTimeout(()=>l.querySelector(".af-search-results")?.remove(),200)});let d=this.plugin.runtime.getFleetStatus(),u=s.createDiv({cls:"af-status-pills"});if(d.running>0){let m=u.createSpan({cls:"af-pill yellow"});m.createSpan({cls:"af-dot pulse"}),m.appendText(` ${d.running} Running`)}if(d.pending>0){let m=u.createSpan({cls:"af-pill blue"});m.createSpan({cls:"af-dot"}),m.appendText(` ${d.pending} Pending`)}let p=u.createSpan({cls:"af-pill green"});p.createSpan({cls:"af-dot"}),p.appendText(` ${d.completedToday} Today`)}renderTabBar(e){let s=e.createDiv({cls:"af-tab-bar"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.runtime.getFleetStatus();for(let i of kh){let o=this.currentPage===i||i==="agents"&&(this.currentPage==="agent-detail"||this.currentPage==="edit-agent"||this.currentPage==="create-agent")||i==="kanban"&&(this.currentPage==="task-detail"||this.currentPage==="edit-task")||i==="skills"&&(this.currentPage==="edit-skill"||this.currentPage==="create-skill")||i==="mcp"&&this.currentPage==="add-mcp-server",c=s.createEl("button",{cls:`af-tab-item${o?" active":""}`}),l=c.createSpan({cls:"af-tab-icon"});if((0,b.setIcon)(l,bh[i]),c.appendText(i==="dashboard"?"Overview":Jo[i]),i==="agents")c.createSpan({cls:"af-badge",text:String(n.agents.length)});else if(i==="skills")c.createSpan({cls:"af-badge",text:String(n.skills.length)});else if(i==="mcp"){let h=this.plugin.repository.getMcpServers().length;c.createSpan({cls:"af-badge",text:String(h)})}else i==="approvals"&&a.pending>0&&c.createSpan({cls:"af-badge af-badge-warn",text:String(a.pending)});c.onclick=()=>this.navigate(i)}}renderDashboardPage(e){let s=e.createDiv({cls:"af-dashboard"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.runtime.getRecentRuns(),i=this.plugin.runtime.getFleetStatus(),o=a.filter(S=>(S.approvals??[]).some(T=>T.status==="pending"));for(let S of o)for(let T of S.approvals??[])T.status==="pending"&&this.renderApprovalBanner(s,S,T.tool);let c=s.createDiv({cls:"af-dash-grid"}),l=n.agents.filter(S=>S.enabled).length,h=n.agents.length;this.renderStatCard(c,"Active Agents",`${l}`,`/ ${h}`,"bot",`${l} of ${h} enabled`);let d=this.toLocalDateStr(new Date),u=a.filter(S=>this.runToLocalDate(S.started)===d),p=u.filter(S=>S.status==="success").length,m=u.filter(S=>S.status==="failure"||S.status==="timeout").length;this.renderStatCard(c,"Runs Today",String(u.length),"","activity",`${p} passed \xB7 ${m} failed \xB7 ${i.running} running`);let f=this.plugin.runtime.getUsageRecords().filter(S=>this.runToLocalDate(S.ts)===d),{tokens:g,cost:y}=this.combinedTotals(u,f),v=y>0?` \xB7 $${y.toFixed(2)}`:"";this.renderStatCard(c,"Tokens Used",Ka(g),"","zap",`today${v}`);let k=n.tasks.filter(S=>S.enabled&&S.schedule);this.renderStatCard(c,"Scheduled Tasks",String(k.length),"","clock",k.length>0?`Next: ${this.getNextTaskLabel(k)}`:"No scheduled tasks"),this.renderChartsRow(s,a,this.plugin.runtime.getChartRuns()),this.renderStreamingCards(s);let w=s.createDiv({cls:"af-dash-split"});this.renderActivityTimeline(w,a),this.renderFleetStatusPanel(w,n)}renderChartsRow(e,s,n){let a=e.createDiv({cls:"af-charts-row"}),i=a.createDiv({cls:"af-section-card af-chart-section"}),c=i.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(c,"activity"),c.appendText(" Run Activity (14d)");let l=i.createDiv({cls:"af-chart-body"}),h=this.buildChartData(n,14);h.some(y=>y.success+y.failure+y.cancelled>0)?zo(l,h):this.renderEmptyState(l,"activity","No run data","Run agents to see activity");let d=a.createDiv({cls:"af-section-card af-chart-section"}),p=d.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(p,"target"),p.appendText(" Success Rate");let m=d.createDiv({cls:"af-chart-body af-chart-body-center"}),f=s.length,g=s.filter(y=>y.status==="success").length;f>0?Go(m,g,f):this.renderEmptyState(m,"target","No data","")}toLocalDateStr(e){let s=n=>String(n).padStart(2,"0");return`${e.getFullYear()}-${s(e.getMonth()+1)}-${s(e.getDate())}`}runToLocalDate(e){return this.toLocalDateStr(new Date(e))}runCost(e){return e.costUsd??Ho(e.model,e.tokensUsed??0)}usageCost(e){return e.costUsd??Wo(e.model,{inputTokens:e.inputTokens,outputTokens:e.outputTokens,cacheReadTokens:e.cacheReadTokens,cacheCreateTokens:e.cacheCreateTokens})}combinedTotals(e,s){let n=e.reduce((i,o)=>i+(o.tokensUsed??0),0)+s.reduce((i,o)=>i+o.totalTokens,0),a=e.reduce((i,o)=>i+this.runCost(o),0)+s.reduce((i,o)=>i+this.usageCost(o),0);return{tokens:n,cost:a}}buildChartData(e,s){let n=[],a=new Date;for(let i=s-1;i>=0;i--){let o=new Date(a);o.setDate(o.getDate()-i);let c=this.toLocalDateStr(o),l=e.filter(h=>this.runToLocalDate(h.started)===c);n.push({date:c,success:l.filter(h=>h.status==="success").length,failure:l.filter(h=>h.status==="failure"||h.status==="timeout").length,cancelled:l.filter(h=>h.status==="cancelled").length})}return n}renderStreamingCards(e){let n=this.plugin.runtime.getSnapshot().agents.filter(c=>this.plugin.runtime.getAgentState(c.name).status==="running");if(n.length===0)return;let a=e.createDiv({cls:"af-streaming-section"}),o=a.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(o,"terminal"),o.appendText(" Active Agents");for(let c of n)this.renderStreamingCard(a,c.name)}renderStreamingCard(e,s){let n=e.createDiv({cls:"af-streaming-card"}),a=this.plugin.runtime.getAgentState(s),i=this.plugin.runtime.getSnapshot(),o=a.currentTaskId?i.tasks.find(m=>m.taskId===a.currentTaskId):void 0,c=o?` \u2192 ${o.taskId}`:a.currentTaskId?.startsWith("reflection-")?" \u2192 Reflection":a.currentTaskId?.startsWith("heartbeat-")||a.status==="running"?" \u2192 Heartbeat":"",l=n.createDiv({cls:"af-streaming-card-header"});l.createSpan({cls:"af-dot pulse",attr:{style:"background: var(--af-yellow)"}}),l.createSpan({cls:"af-streaming-card-agent",text:` ${s}`}),c&&l.createSpan({cls:"af-streaming-card-task",text:c});let h=n.createDiv({cls:"af-streaming-output"}),d=a.currentTaskId?.startsWith("reflection-")?"Consolidating memory\u2026":"Working\u2026",u=m=>{let f=ye(m).filter(g=>g.trim().length>0).slice(-4);h.setText(f.length>0?f.join(`
11950
+ `):d)};u(this.plugin.runtime.getRunOutputBuffer(s));let p=this.plugin.runtime.onRunOutput(s,()=>{u(this.plugin.runtime.getRunOutputBuffer(s)),h.scrollTop=h.scrollHeight});this.streamingUnsubscribes.push(p)}renderApprovalBanner(e,s,n){let a=e.createDiv({cls:"af-approval-banner"}),i=a.createDiv({cls:"af-approval-icon"});(0,b.setIcon)(i,"shield-check");let o=a.createDiv({cls:"af-approval-text"});o.createDiv({cls:"af-approval-title",text:`${s.agent} wants to run: ${n}`}),o.createDiv({cls:"af-approval-desc",text:`Approval required for tool: ${n}`});let c=a.createDiv({cls:"af-approval-actions"}),l=c.createEl("button",{cls:"af-btn-approve",text:"Approve"});l.onclick=()=>void this.plugin.runtime.resolveApproval(s,n,"approved").then(()=>this.render());let h=c.createEl("button",{cls:"af-btn-reject",text:"Reject"});h.onclick=()=>void this.plugin.runtime.resolveApproval(s,n,"rejected").then(()=>this.render())}renderStatCard(e,s,n,a,i,o){let c=e.createDiv({cls:"af-stat-card"}),l=c.createDiv({cls:"af-stat-label"});R(l,i,"af-stat-icon"),l.appendText(` ${s}`);let h=c.createDiv({cls:"af-stat-value"});h.appendText(n),a&&h.createSpan({cls:"af-stat-value-suffix",text:a}),c.createDiv({cls:"af-stat-sub",text:o})}renderActivityTimeline(e,s){let n=e.createDiv({cls:"af-section-card"}),i=n.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(i,"inbox"),i.appendText(" Recent Activity");let o=n.createDiv({cls:"af-timeline"}),c=s.slice(0,8);if(c.length===0){this.renderEmptyState(o,"inbox","No runs yet","Run an agent to see activity here");return}for(let l of c)this.renderTimelineItem(o,l)}renderTimelineItem(e,s){let n=e.createDiv({cls:"af-timeline-item"}),a=this.statusToTimelineClass(s.status),i=n.createDiv({cls:`af-tl-icon ${a}`});(0,b.setIcon)(i,this.statusToIconName(s.status));let o=n.createDiv({cls:"af-tl-body"}),c=o.createDiv({cls:"af-tl-title"});c.createSpan({cls:"af-agent-tag",text:s.agent}),c.appendText(` ${s.task}`),o.createDiv({cls:"af-tl-desc",text:Wt(s.output,100)});let l=o.createDiv({cls:"af-tl-meta"}),h=l.createSpan();if(R(h,"clock","af-meta-icon"),h.appendText(` ${this.formatStarted(s.started)} \xB7 ${this.formatDuration(s.durationSeconds)}`),s.tokensUsed){let d=l.createSpan();R(d,"zap","af-meta-icon"),d.appendText(` ${s.tokensUsed.toLocaleString()} tokens`)}n.onclick=()=>this.openSlideover(s)}renderFleetStatusPanel(e,s){let n=e.createDiv({cls:"af-section-card"}),a=n.createDiv({cls:"af-section-header"}),i=a.createDiv({cls:"af-section-title"});R(i,"bot"),i.appendText(" Fleet Status");let c=a.createDiv({cls:"af-section-actions"}).createEl("button",{cls:"af-btn-sm primary"});R(c,"plus","af-btn-icon"),c.appendText(" New Agent"),c.onclick=()=>void this.plugin.createAgentTemplate();let l=n.createDiv({cls:"af-agent-mini-list"});if(s.agents.length===0){this.renderEmptyState(l,"bot","No agents yet","Create your first agent to get started");return}for(let d of s.agents)this.renderAgentMini(l,d,s.tasks);let h=s.agents.filter(d=>d.enabled);if(h.length>0){let d=n.createDiv({cls:"af-quick-run"}),u=d.createDiv({cls:"af-quick-run-label"});R(u,"zap","af-meta-icon"),u.appendText(" Quick Run");let p=d.createDiv({cls:"af-quick-run-row"}),m=p.createEl("select",{cls:"af-select"});for(let g of h)m.createEl("option",{text:g.name,attr:{value:g.name}});let f=p.createEl("button",{cls:"af-btn-sm primary"});R(f,"play","af-btn-icon"),f.appendText(" Run"),f.onclick=()=>void this.plugin.runAgentPrompt(m.value)}}renderAgentMini(e,s,n){let a=this.plugin.runtime.getAgentState(s.name),i=n.filter(u=>u.agent===s.name),o=this.healthToClass(a.status),c=e.createDiv({cls:"af-agent-mini"}),l=c.createDiv({cls:`af-agent-avatar ${o}`});s.avatar?.trim()?this.renderAgentAvatar(l,s):l.setText(this.getInitials(s.name));let h=c.createDiv({cls:"af-agent-info"});h.createDiv({cls:"af-agent-name",text:s.name});let d="";if(a.status==="running")d=`Running now \xB7 ${i.length} task${i.length!==1?"s":""}`;else if(!s.enabled)d=`Disabled \xB7 ${i.length} task${i.length!==1?"s":""} paused`;else{let u=i.map(p=>p.nextRun).filter(Boolean).sort()[0];d=u?`Next: ${this.formatNextRun(u)} \xB7 ${i.length} task${i.length!==1?"s":""}`:`${i.length} task${i.length!==1?"s":""}`}h.createDiv({cls:"af-agent-desc",text:d}),c.createDiv({cls:`af-agent-status-dot ${o}`}),c.onclick=()=>this.navigate("agent-detail",s.name)}renderAgentsPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Agents"}),a.createDiv({cls:"af-toolbar-spacer"});let i=a.createEl("button",{cls:"af-btn-sm primary"});R(i,"plus","af-btn-icon"),i.appendText(" New Agent"),i.onclick=()=>void this.plugin.createAgentTemplate();let o=s.createDiv({cls:"af-agents-grid"});if(n.agents.length===0){this.renderEmptyState(o,"bot","No agents configured","Create your first agent to get started");return}for(let c of n.agents)this.renderAgentCard(o,c,n)}renderAgentCard(e,s,n){let a=this.plugin.runtime.getAgentState(s.name),i=this.plugin.runtime.getRecentRuns().filter(D=>D.agent===s.name),o=e.createDiv({cls:`af-agent-card${s.enabled?"":" disabled"}`}),c=o.createDiv({cls:"af-agent-card-header"}),l=s.enabled?this.healthToClass(a.status):"disabled",h=c.createDiv({cls:`af-agent-card-avatar ${l}`});this.renderAgentAvatar(h,s);let d=c.createDiv({cls:"af-agent-card-titleblock"}),u=d.createDiv({cls:"af-agent-card-name"});if(u.appendText(s.name),s.heartbeatEnabled&&s.heartbeatSchedule){let D=u.createSpan({cls:"af-heartbeat-indicator"});(0,b.setIcon)(D,"heart-pulse"),D.title=`Heartbeat: ${s.heartbeatSchedule}`}d.createDiv({cls:"af-agent-card-desc",text:s.description??"No description"});let p=c.createDiv({cls:`af-agent-card-toggle${s.enabled?" on":""}`});p.onclick=D=>{D.stopPropagation(),this.plugin.toggleAgent(s.name,!s.enabled)};let m=o.createDiv({cls:"af-agent-card-stats"}),f=i.length,g=i.filter(D=>D.status==="success").length,y=f>0?Math.round(g/f*100):0,v=f>0?Math.round(i.reduce((D,O)=>D+O.durationSeconds,0)/f):0,k=i.reduce((D,O)=>D+(O.tokensUsed??0),0);if(this.renderAgentStat(m,String(f),"Runs"),this.renderAgentStat(m,`${y}%`,"Success"),this.renderAgentStat(m,`${v}s`,"Avg Time"),this.renderAgentStat(m,Ka(k),"Tokens"),s.skills.length>0){let D=o.createDiv({cls:"af-agent-card-skills"});for(let O of s.skills)D.createSpan({cls:"af-skill-tag",text:O})}let w=o.createDiv({cls:"af-agent-card-footer"}),S=[`Model: ${s.model}`];s.approvalRequired.length>0&&S.push(`Approval: ${s.approvalRequired.join(", ")}`),s.memory&&S.push("Memory: on"),s.enabled||S.unshift("Disabled"),w.createSpan({cls:"af-agent-card-meta",text:S.join(" \xB7 ")});let T=w.createDiv({cls:"af-agent-card-actions"});if(!s.enabled){let D=T.createEl("button",{cls:"af-btn-sm",text:"Enable"});D.onclick=O=>{O.stopPropagation(),this.plugin.toggleAgent(s.name,!0)}}let _=T.createEl("button",{cls:"af-btn-sm"});if(R(_,"edit","af-btn-icon"),_.appendText(" Edit"),_.onclick=D=>{D.stopPropagation(),this.navigate("edit-agent",s.name)},s.enabled){let D=T.createEl("button",{cls:"af-btn-sm primary"});R(D,"play","af-btn-icon"),D.appendText(" Run"),D.onclick=O=>{O.stopPropagation(),this.plugin.runAgentPrompt(s.name)}}o.onclick=()=>this.navigate("agent-detail",s.name)}renderAgentStat(e,s,n){let a=e.createDiv({cls:"af-agent-stat"});a.createSpan({cls:"af-agent-stat-value",text:s}),a.createSpan({cls:"af-agent-stat-label",text:n})}renderAgentDetailPage(e){let s=e.createDiv({cls:"af-agent-detail-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"bot","No agent selected","Select an agent from the list");return}let a=this.plugin.runtime.getSnapshot().agents.find(k=>k.name===n);if(!a){this.renderEmptyState(s,"bot","Agent not found",`Agent "${n}" was not found`);return}let i=this.plugin.runtime.getAgentState(a.name),o=this.plugin.runtime.getRecentRuns().filter(k=>k.agent===a.name),c=s.createDiv({cls:"af-detail-header"}),l=c.createDiv({cls:"af-detail-header-left"}),h=l.createDiv({cls:`af-agent-card-avatar ${this.healthToClass(i.status)}`});this.renderAgentAvatar(h,a);let d=l.createDiv();d.createDiv({cls:"af-detail-header-name",text:a.name}),d.createDiv({cls:"af-detail-header-desc",text:a.description??"No description"});let u=c.createDiv({cls:"af-detail-header-actions"}),p=u.createEl("button",{cls:"af-btn-sm primary"});if(R(p,"message-circle","af-btn-icon"),p.appendText(" Chat"),p.onclick=()=>this.openChatSlideover(a),a.enabled){let k=u.createEl("button",{cls:"af-btn-sm"});R(k,"play","af-btn-icon"),k.appendText(" Run Now"),k.onclick=()=>void this.plugin.runAgentPrompt(a.name);let w=u.createEl("button",{cls:"af-btn-sm"});R(w,"pause","af-btn-icon"),w.appendText(" Disable"),w.onclick=()=>void this.plugin.toggleAgent(a.name,!1)}else{let k=u.createEl("button",{cls:"af-btn-sm"});R(k,"play","af-btn-icon"),k.appendText(" Enable"),k.onclick=()=>void this.plugin.toggleAgent(a.name,!0)}let m=u.createEl("button",{cls:"af-btn-sm"});R(m,"edit","af-btn-icon"),m.appendText(" Edit"),m.onclick=()=>this.navigate("edit-agent",a.name);let f=u.createEl("button",{cls:"af-btn-sm danger"});R(f,"trash-2","af-btn-icon"),f.appendText(" Delete"),f.onclick=()=>void this.plugin.deleteAgent(a.name);let g=s.createDiv({cls:"af-detail-tabs"}),y=[{id:"overview",label:"Overview",icon:"layout-dashboard"},{id:"config",label:"Config",icon:"settings"},{id:"runs",label:"Runs",icon:"scroll-text"},{id:"memory",label:"Memory",icon:"file-text"}];for(let k of y){let w=g.createEl("button",{cls:`af-detail-tab${this.agentDetailTab===k.id?" active":""}`});R(w,k.icon,"af-tab-icon"),w.appendText(` ${k.label}`),w.onclick=()=>{this.agentDetailTab=k.id,this.render()}}let v=s.createDiv({cls:"af-detail-tab-content"});switch(this.agentDetailTab){case"overview":this.renderAgentOverviewTab(v,a,o);break;case"config":this.renderAgentConfigTab(v,a);break;case"runs":this.renderAgentRunsTab(v,o);break;case"memory":this.renderAgentMemoryTab(v,a);break}}renderAgentOverviewTab(e,s,n){let a=e.createDiv({cls:"af-dash-grid"}),i=n.length,o=n.filter(w=>w.status==="success").length,c=i>0?Math.round(o/i*100):0,l=i>0?Math.round(n.reduce((w,S)=>w+S.durationSeconds,0)/i):0,h=this.plugin.runtime.getUsageRecords().filter(w=>w.agent===s.name),{tokens:d,cost:u}=this.combinedTotals(n,h),p=u>0?` \xB7 $${u.toFixed(2)}`:"";if(this.renderStatCard(a,"Total Runs",String(i),"","activity","all time"),this.renderStatCard(a,"Success Rate",`${c}%`,"","check-circle-2",`${o}/${i}`),this.renderStatCard(a,"Avg Time",`${l}s`,"","clock","per run"),this.renderStatCard(a,"Total Tokens",Ka(d),"","zap",`all time${p}`),s.isFolder&&(s.heartbeatBody.trim()||s.heartbeatEnabled)){let w=e.createDiv({cls:"af-section-card"}),S=w.createDiv({cls:"af-section-header"}),T=S.createDiv({cls:"af-section-title"});R(T,"heart-pulse"),T.appendText(" Heartbeat");let D=S.createDiv({cls:"af-detail-header-actions"}).createDiv({cls:`af-agent-card-toggle${s.heartbeatEnabled?" on":""}`});D.onclick=async()=>{let E=D.hasClass("on");await this.plugin.repository.updateHeartbeat(s.name,{enabled:!E}),await this.plugin.refreshFromVault(),new b.Notice(`Heartbeat ${E?"paused":"enabled"} for ${s.name}`)};let O=w.createDiv({cls:"af-config-form"});this.renderConfigRow(O,"Schedule",xh(s.heartbeatSchedule));let C=this.plugin.runtime.getNextHeartbeat(s.name);C&&s.heartbeatEnabled&&this.renderConfigRow(O,"Next run",this.timeUntil(C)),s.heartbeatChannel&&this.renderConfigRow(O,"Channel",s.heartbeatChannel)}if(s.skills.length>0){let w=e.createDiv({cls:"af-section-card"}),T=w.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(T,"puzzle"),T.appendText(" Skills");let _=w.createDiv({cls:"af-detail-skills-list"});for(let D of s.skills)_.createSpan({cls:"af-skill-tag",text:D})}let m=s.mcpServers??[];if(m.length>0){let w=e.createDiv({cls:"af-section-card"}),T=w.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(T,"plug"),T.appendText(" MCP Servers");let _=w.createDiv({cls:"af-mcp-overview-list"}),D=this.plugin.repository.getMcpServers();for(let O of m){let C=D.find(M=>M.name===O),E=_.createDiv({cls:"af-mcp-overview-row"}),P=E.createSpan({cls:`af-mcp-status-dot ${C?C.enabled?C.status:"disabled":"disconnected"}`});P.title=C?C.enabled?C.status:"disabled":"unknown",E.createSpan({cls:"af-mcp-overview-name",text:O});let N=C?.toolDetails.length??C?.tools.length??0;N>0?E.createSpan({cls:"af-mcp-overview-tools",text:`${N} tools`}):C&&!C.enabled?E.createSpan({cls:"af-mcp-overview-tools af-muted",text:"disabled"}):C?.status==="needs-auth"&&E.createSpan({cls:"af-mcp-overview-tools af-muted",text:"needs auth"})}}if(s.permissionRules.allow.length>0||s.permissionRules.deny.length>0||s.permissionMode&&s.permissionMode!=="default"){let w=e.createDiv({cls:"af-section-card"}),T=w.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(T,"shield-check"),T.appendText(" Permissions");let _=w.createDiv({cls:"af-config-form"});this.renderConfigRow(_,"Mode",s.permissionMode||"default"),s.permissionRules.allow.length>0&&this.renderConfigRow(_,"Allowed",s.permissionRules.allow.join(", ")),s.permissionRules.deny.length>0&&this.renderConfigRow(_,"Denied",s.permissionRules.deny.join(", "))}let g=e.createDiv({cls:"af-section-card"}),v=g.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(v,"scroll-text"),v.appendText(" Recent Runs");let k=g.createDiv({cls:"af-timeline"});if(n.length===0)this.renderEmptyState(k,"scroll-text","No runs yet","");else for(let w of n.slice(0,5))this.renderTimelineItem(k,w)}renderAgentConfigTab(e,s){let n=e.createDiv({cls:"af-config-form"});this.renderConfigRow(n,"Name",s.name),this.renderConfigRow(n,"Description",s.description??""),this.renderConfigRow(n,"Model",s.model),this.renderConfigRow(n,"Timeout",`${s.timeout}s`),this.renderConfigRow(n,"Working Directory",s.cwd??"(vault root)"),this.renderConfigRow(n,"Permission Mode",s.permissionMode||"default"),this.renderConfigRow(n,"Approval Required",s.approvalRequired.join(", ")||"none"),s.permissionRules.allow.length>0&&this.renderConfigRow(n,"Allowed Commands",s.permissionRules.allow.join(", ")),s.permissionRules.deny.length>0&&this.renderConfigRow(n,"Blocked Commands",s.permissionRules.deny.join(", ")),this.renderConfigRow(n,"Memory",s.memory?"Enabled":"Disabled"),this.renderConfigRow(n,"Auto-compact",s.autoCompactThreshold&&s.autoCompactThreshold>0?`at ${s.autoCompactThreshold}% context`:"disabled"),s.wikiReferences&&s.wikiReferences.length>0&&this.renderConfigRow(n,"Wiki access",s.wikiReferences.map(c=>c.agent).join(", ")),this.renderConfigRow(n,"Tags",s.tags.join(", ")||"none");let a=n.createDiv({cls:"af-config-prompt-section"});if(a.createDiv({cls:"af-slideover-section-title",text:"SYSTEM PROMPT"}),a.createDiv({cls:"af-output-block",text:s.body||"(empty)"}),s.heartbeatBody.trim()){let c=n.createDiv({cls:"af-config-prompt-section"});c.createDiv({cls:"af-slideover-section-title",text:"HEARTBEAT INSTRUCTION"}),c.createDiv({cls:"af-output-block",text:s.heartbeatBody})}let o=n.createDiv({cls:"af-slideover-actions"}).createEl("button",{cls:"af-btn-sm primary"});R(o,"edit","af-btn-icon"),o.appendText(" Edit Agent"),o.onclick=()=>this.navigate("edit-agent",s.name)}renderConfigRow(e,s,n){let a=e.createDiv({cls:"af-detail-row"});a.createSpan({cls:"af-detail-label",text:s}),a.createSpan({cls:"af-detail-value af-mono",text:n})}renderAgentRunsTab(e,s){if(s.length===0){this.renderEmptyState(e,"scroll-text","No runs yet","Run this agent to see history");return}for(let n of s){let a=e.createDiv({cls:"af-run-list-item"}),i=a.createDiv({cls:`af-tl-icon ${this.statusToTimelineClass(n.status)}`});(0,b.setIcon)(i,this.statusToIconName(n.status));let o=a.createDiv({cls:"af-tl-body"}),c=o.createDiv({cls:"af-tl-title"});c.createSpan({text:n.task}),c.createSpan({cls:`af-status-badge ${this.statusToBadgeClass(n.status)}`,text:this.statusToBadgeText(n.status)});let l=o.createDiv({cls:"af-tl-meta"});l.createSpan({text:`${this.formatStarted(n.started)} \xB7 ${this.formatDuration(n.durationSeconds)}`}),n.tokensUsed&&l.createSpan({text:`${n.tokensUsed.toLocaleString()} tokens`}),o.createDiv({cls:"af-tl-desc",text:Wt(n.output,120)}),a.onclick=()=>this.openSlideover(n)}}async renderAgentMemoryTab(e,s){if(!s.memory){this.renderEmptyState(e,"file-text","Memory disabled","Enable memory in agent config");return}let n=await this.plugin.repository.readWorkingMemory(s.name),a=e.createDiv({cls:"af-form-help"}),i=n?.tokenEstimate??0,o=s.reflection.enabled?`reflection on (${s.reflection.schedule})`:"reflection off";a.setText(`~${i} / ${s.memoryTokenBudget} tokens \xB7 ${o}`),!n||n.sections.length===0?this.renderEmptyState(e,"file-text","No memories yet","Agent will learn from runs"):e.createDiv({cls:"af-output-block"}).setText(Ke(n.sections));let c=e.createDiv({cls:"af-slideover-actions"}),l=c.createEl("button",{cls:"af-btn-sm"});R(l,"moon","af-btn-icon"),l.appendText(" Reflect now"),l.onclick=async()=>{l.disabled=!0,l.setText(" Reflecting\u2026");let d=await this.plugin.runtime.runReflectionNow(s.name);new b.Notice(`Agent Fleet: ${d.message}`),await this.renderAgentMemoryTab((e.empty(),e),s)};let h=c.createEl("button",{cls:"af-btn-sm"});R(h,"external-link","af-btn-icon"),h.appendText(" Open in Editor"),h.onclick=()=>void this.plugin.openPath(this.plugin.repository.getWorkingMemoryPath(s.name))}timeAgo(e){let s=Math.round((Date.now()-e.getTime())/1e3);if(s<60)return"just now";let n=Math.round(s/60);if(n<60)return`${n}m ago`;let a=Math.round(n/60);return a<24?`${a}h ago`:`${Math.round(a/24)}d ago`}timeUntil(e){let s=Math.round((e.getTime()-Date.now())/1e3);if(s<60)return"< 1m";let n=Math.round(s/60);if(n<60)return`in ${n}m`;let a=Math.round(n/60);return a<24?`in ${a}h`:`in ${Math.round(a/24)}d`}renderKanbanPage(e){let s=e.createDiv({cls:"af-kanban-page"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.runtime.getRecentRuns(),i=s.createDiv({cls:"af-kanban-toolbar"});i.createDiv({cls:"af-page-title",text:"Tasks Board"}),i.createDiv({cls:"af-toolbar-spacer"});let o=i.createEl("button",{cls:"af-btn-sm primary"});R(o,"plus","af-btn-icon"),o.appendText(" New Task"),o.onclick=()=>this.navigate("create-task");let c=s.createDiv({cls:"af-kanban-board"}),l=[],h=[],d=[],u=[],p=[],m=new Set;for(let v of n.agents){let k=this.plugin.runtime.getAgentState(v.name);k.status==="running"&&k.currentTaskId&&m.add(k.currentTaskId)}let f=this.toLocalDateStr(new Date);for(let v of a)v.status==="success"&&this.runToLocalDate(v.started)===f?u.push(v):(v.status==="failure"||v.status==="timeout"||v.status==="cancelled")&&this.runToLocalDate(v.started)===f&&p.push(v);let g=new Set(u.map(v=>v.task)),y=new Set(p.map(v=>v.task));for(let v of n.tasks){let k=g.has(v.taskId)||y.has(v.taskId)||v.lastRun&&this.runToLocalDate(v.lastRun)===f;if(m.has(v.taskId))d.push(v);else{if(k&&!v.schedule)continue;v.schedule&&v.enabled?h.push(v):l.push(v)}}this.renderKanbanColumn(c,"Backlog","inbox",l.length,v=>{for(let k of l)this.renderKanbanTaskCard(v,k,n,!0)},"backlog"),this.renderKanbanColumn(c,"Scheduled","clock",h.length,v=>{for(let k of h)this.renderKanbanTaskCard(v,k,n,!0)},"scheduled"),this.renderKanbanColumn(c,"Running","loader-2",d.length,v=>{for(let k of d)this.renderKanbanRunningCard(v,k)},"running",!1,"running"),this.renderKanbanColumn(c,"Done","check-circle-2",u.length,v=>{for(let k of u.slice(0,10))this.renderKanbanCompletedCard(v,k)},"completed"),this.renderKanbanColumn(c,"Failed","x-circle",p.length,v=>{for(let k of p)this.renderKanbanFailedCard(v,k)},"failed",!1,"failed")}renderKanbanColumn(e,s,n,a,i,o,c=!1,l){let h=e.createDiv({cls:`af-kanban-column${l?` af-kanban-${l}`:""}`});(o==="backlog"||o==="scheduled")&&Yo(h,m=>this.handleTaskDrop(m,o));let u=h.createDiv({cls:"af-kanban-col-header"}).createDiv({cls:"af-kanban-col-title"});R(u,n),u.appendText(` ${s} `),u.createSpan({cls:"af-kanban-col-count",text:String(a)});let p=h.createDiv({cls:"af-kanban-col-body"});if(i(p),a===0&&p.createDiv({cls:"af-kanban-empty",text:"No items"}),c){let f=h.createDiv({cls:"af-kanban-col-add"}).createEl("button");R(f,"plus","af-btn-icon"),f.appendText(" Add task"),f.onclick=()=>{this.navigate("create-task")}}}handleTaskDrop(e,s){let n=this.plugin.runtime.getSnapshot().tasks.find(a=>a.taskId===e);if(n){if(s==="backlog")this.setTaskEnabled(n,!1).then(()=>{new b.Notice(`Task "${e}" moved to backlog (disabled)`)});else if(s==="scheduled"){if(!n.schedule&&!n.runAt){new b.Notice(`Task "${e}" needs a schedule. Open task details to set one.`),this.navigate("task-detail",e);return}this.setTaskEnabled(n,!0).then(()=>{new b.Notice(`Task "${e}" moved to scheduled (enabled)`)})}}}async setTaskEnabled(e,s){let n=this.plugin.app.vault.getAbstractFileByPath(e.filePath);if(!n||!(n instanceof b.TFile))return;let a=await this.plugin.app.vault.cachedRead(n),{frontmatter:i,body:o}=J(a);i.enabled=s,await this.plugin.app.vault.modify(n,H(i,o)),await this.plugin.refreshFromVault()}renderKanbanTaskCard(e,s,n,a){let i=e.createDiv({cls:`af-kanban-card af-priority-${s.priority}`});if(a){Vo(i,s.taskId);let f=i.createDiv({cls:"af-kanban-card-grip"});(0,b.setIcon)(f,"grip-vertical")}let o=i.createDiv({cls:"af-kanban-card-header"});o.createDiv({cls:"af-kanban-card-title",text:s.taskId});let l=(n.agents.find(f=>f.name===s.agent)?.enabled??!1)&&s.enabled,h=o.createSpan({cls:`af-kanban-card-status ${l?"active":"inactive"}`});h.title=l?"Active":"Inactive";let d=i.createDiv({cls:"af-kanban-card-agent"}),u=d.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(u,"bot"),d.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let m=i.createDiv({cls:"af-kanban-card-footer"}).createSpan({cls:"af-kanban-card-schedule"});l?s.schedule?(R(m,"refresh-cw","af-meta-icon"),m.appendText(` ${this.humanizeCron(s.schedule)}`)):m.appendText(s.runAt??"Manual"):(R(m,"pause","af-meta-icon"),m.appendText(" Paused")),i.onclick=()=>this.navigate("task-detail",s.taskId)}renderKanbanRunningCard(e,s){let n=e.createDiv({cls:"af-kanban-card af-kanban-card-running"});n.createDiv({cls:"af-kanban-card-title",text:s.taskId});let a=n.createDiv({cls:"af-kanban-card-agent"}),i=a.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(i,"bot"),a.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let c=this.plugin.runtime.getSnapshot().agents.find(S=>S.name===s.agent)?.timeout??300,l=this.plugin.runtime.getAgentState(s.agent),h=l.runStarted?new Date(l.runStarted).getTime():Date.now(),p=n.createDiv({cls:"af-kanban-progress"}).createDiv({cls:"af-kanban-progress-track"}).createDiv({cls:"af-kanban-progress-bar af-kanban-progress-bar-real"}),m=(Date.now()-h)/1e3,f=Math.min(95,m/c*100);p.setCssStyles({width:`${f}%`});let g=n.createDiv({cls:"af-kanban-card-footer"}),y=g.createSpan({cls:"af-kanban-card-schedule"});R(y,"loader-2","af-meta-icon");let v=Math.round(m);y.appendText(` ${v}s / ${c}s`);let k=g.createEl("button",{cls:"af-kanban-stop-btn"});(0,b.setIcon)(k,"square"),k.title="Stop task",k.onclick=S=>{S.stopPropagation(),this.plugin.runtime.abortAgentRun(s.agent),new b.Notice(`Stopped task "${s.taskId}"`)};let w=window.setInterval(()=>{let S=(Date.now()-h)/1e3,T=Math.min(95,S/c*100);p.setCssStyles({width:`${T}%`});let _=Math.round(S);y.textContent="",(0,b.setIcon)(y,"loader-2"),y.appendText(` ${_}s / ${c}s`)},1e3);this.streamingUnsubscribes.push(()=>window.clearInterval(w))}renderKanbanCompletedCard(e,s){let n=e.createDiv({cls:"af-kanban-card"});n.createDiv({cls:"af-kanban-card-title",text:s.task});let a=n.createDiv({cls:"af-kanban-card-agent"}),i=a.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(i,"bot"),a.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let c=n.createDiv({cls:"af-kanban-card-footer"}).createSpan({cls:"af-kanban-card-schedule"});R(c,"check-circle-2","af-meta-icon"),c.appendText(` ${this.formatStarted(s.started)} \xB7 ${this.formatDuration(s.durationSeconds)}`),n.onclick=()=>this.openSlideover(s)}renderKanbanFailedCard(e,s){let n=s.status==="cancelled",a=e.createDiv({cls:`af-kanban-card ${n?"af-kanban-card-cancelled":"af-kanban-card-failed"}`});a.createDiv({cls:"af-kanban-card-title",text:s.task});let i=a.createDiv({cls:"af-kanban-card-agent"}),o=i.createSpan({cls:"af-kanban-card-agent-icon"});(0,b.setIcon)(o,"bot"),i.createSpan({cls:"af-kanban-card-agent-name",text:s.agent});let c=n?`Stopped after ${s.durationSeconds}s`:s.status==="timeout"?`Timeout after ${s.durationSeconds}s`:Wt(s.output,60),l=a.createDiv({cls:"af-kanban-card-error"});R(l,n?"square":"alert-triangle","af-meta-icon"),l.appendText(` ${c}`);let h=a.createDiv({cls:"af-kanban-card-footer"}),d=h.createSpan({cls:"af-kanban-card-schedule"});if(R(d,n?"square":"x-circle","af-meta-icon"),d.appendText(` ${this.formatStarted(s.started)}`),!n){let u=h.createEl("button",{cls:"af-btn-sm"});R(u,"refresh-cw","af-btn-icon"),u.appendText(" Retry"),u.onclick=p=>{p.stopPropagation(),this.plugin.runAgentPrompt(s.agent)}}a.onclick=()=>this.openSlideover(s)}renderRunsPage(e){let s=e.createDiv({cls:"af-runs-page"}),n=this.plugin.runtime.getRecentRuns(),a=s.createDiv({cls:"af-runs-toolbar"});a.createDiv({cls:"af-page-title",text:"Run History"}),a.createDiv({cls:"af-toolbar-spacer"});let i=s.createDiv({cls:"af-runs-table"});if(n.length===0){this.renderEmptyState(i,"scroll-text","No runs yet","Run an agent to see history here");return}let o=i.createEl("table"),l=o.createEl("thead").createEl("tr");for(let d of["Status","Agent","Task","Started","Duration","Tokens","Model"])l.createEl("th",{text:d});let h=o.createEl("tbody");for(let d of n.slice(0,50))this.renderRunRow(h,d)}renderRunRow(e,s){let n=e.createEl("tr"),i=n.createEl("td").createSpan({cls:`af-status-badge ${this.statusToBadgeClass(s.status)}`}),o=i.createSpan();(0,b.setIcon)(o,this.statusToIconName(s.status)),i.appendText(` ${this.statusToBadgeText(s.status)}`);let c=n.createEl("td",{cls:"af-agent-link"});c.setText(s.agent),c.onclick=l=>{l.stopPropagation(),this.navigate("agent-detail",s.agent)},n.createEl("td",{text:s.task}),n.createEl("td",{cls:"af-mono",text:this.formatStarted(s.started)}),n.createEl("td",{cls:"af-mono",text:this.formatDuration(s.durationSeconds)}),n.createEl("td",{cls:"af-mono",text:s.tokensUsed?s.tokensUsed.toLocaleString():"\u2014"}),n.createEl("td",{cls:"af-mono",text:s.model}),n.setCssStyles({cursor:"pointer"}),n.onclick=()=>this.openSlideover(s)}renderSkillsPage(e){let s=e.createDiv({cls:"af-skills-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Skills Library"}),a.createDiv({cls:"af-toolbar-spacer"});let i=a.createEl("button",{cls:"af-btn-sm primary"});R(i,"plus","af-btn-icon"),i.appendText(" New Skill"),i.onclick=()=>void this.plugin.createSkillTemplate();let o=s.createDiv({cls:"af-skills-grid"});if(n.skills.length===0){this.renderEmptyState(o,"puzzle","No skills yet","Create skills to give agents specialized abilities");return}for(let c of n.skills)this.renderSkillCard(o,c,n.agents)}renderSkillCard(e,s,n){let a=e.createDiv({cls:"af-skill-card"}),i=a.createDiv({cls:"af-skill-card-header"}),o=i.createDiv({cls:"af-skill-card-icon"});(0,b.setIcon)(o,this.getSkillIcon(s.name));let c=i.createEl("button",{cls:"af-btn-sm af-btn-xs"});R(c,"edit","af-btn-icon"),c.onclick=h=>{h.stopPropagation(),this.navigate("edit-skill",s.name)},a.createDiv({cls:"af-skill-card-name",text:s.name}),a.createDiv({cls:"af-skill-card-desc",text:s.description??"No description"});let l=n.filter(h=>h.skills.includes(s.name));if(l.length>0){let h=a.createDiv({cls:"af-skill-card-agents"});for(let d of l)h.createSpan({cls:"af-skill-card-agent-tag",text:d.name})}}renderChannelsPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Channels"}),a.createDiv({cls:"af-toolbar-spacer"});let i=a.createEl("button",{cls:"af-btn-sm primary"});R(i,"plus","af-btn-icon"),i.appendText(" New Channel"),i.onclick=()=>this.navigate("create-channel");let o=s.createDiv({cls:"af-agents-grid"});if(n.channels.length===0){this.renderEmptyState(o,"radio","No channels configured","Connect an agent to Slack or another chat platform");return}for(let c of n.channels)this.renderChannelCard(o,c,n.validationIssues)}async renderWikiKeepersPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Wiki Keepers"}),a.createDiv({cls:"af-toolbar-spacer"});let i=n.agents.filter(c=>c.wikiKeeper!==void 0);if(i.length===0){this.renderEmptyState(s,"library","No Wiki Keepers yet","Open Settings \u2192 Agent Fleet \u2192 Wiki Keepers \u2192 + Add to create one.");return}let o=s.createDiv({cls:"af-wk-list"});o.setCssStyles({display:"flex"}),o.setCssStyles({flexDirection:"column"}),o.setCssStyles({gap:"16px"});for(let c of i)await this.renderWikiKeeperCard(o,c)}async renderWikiKeeperCard(e,s){let n=s.wikiKeeper,a=e.createDiv({cls:"af-card"});a.setCssStyles({padding:"16px"}),a.setCssStyles({border:"1px solid var(--background-modifier-border)"}),a.setCssStyles({borderRadius:"8px"});let i=a.createDiv();i.setCssStyles({display:"flex"}),i.setCssStyles({alignItems:"center"}),i.setCssStyles({gap:"12px"}),i.setCssStyles({marginBottom:"12px"});let o=i.createDiv();o.setCssStyles({flex:"1"}),o.createEl("strong",{text:s.name});let c=n.scopeRoot||"(whole vault)";o.createEl("div",{text:`Scope: ${c} \xB7 topics: ${n.topicsRoot}/ \xB7 log: ${n.logPath}`,cls:"af-form-hint"});let l=i.createEl("button",{cls:"af-btn-sm"});l.appendText("Open log"),l.onclick=()=>{let g=n.scopeRoot?`${n.scopeRoot}/${n.logPath}`:n.logPath,y=this.plugin.app.vault.getAbstractFileByPath(g);y instanceof b.TFile?this.plugin.app.workspace.getLeaf().openFile(y):new b.Notice(`Log file not found: ${g}`)};let h=n.scopeRoot?`${n.scopeRoot}/${n.logPath}`:n.logPath,d=this.plugin.app.vault.getAbstractFileByPath(h),u=null;if(d instanceof b.TFile)try{let g=await this.plugin.app.vault.cachedRead(d);u=Ko(g)}catch{u=null}if(!u){a.createDiv({cls:"af-form-hint"}).setText("No lint report yet. Run wiki-lint manually or wait for the weekly task to fire.");return}let p=a.createDiv();if(p.setCssStyles({display:"flex"}),p.setCssStyles({alignItems:"baseline"}),p.setCssStyles({gap:"12px"}),p.setCssStyles({marginBottom:"8px"}),p.createEl("strong",{text:`Lint ${u.date}`}),p.createSpan({cls:"af-form-hint",text:`${u.summary.length} summary lines \xB7 ${u.autoApplied.length} auto-applied \xB7 ${u.needsReview.length} needs review`}),u.summary.length>0){let g=a.createEl("details");g.createEl("summary",{text:"Summary"});let y=g.createEl("ul");y.setCssStyles({marginTop:"4px"});for(let v of u.summary)y.createEl("li",{text:v})}if(u.autoApplied.length>0){let g=a.createEl("details");g.createEl("summary",{text:`Auto-applied (${u.autoApplied.length})`});let y=g.createEl("ul");y.setCssStyles({marginTop:"4px"});for(let v of u.autoApplied)y.createEl("li",{text:v})}if(u.refreshChained.length>0){let g=a.createEl("details");g.createEl("summary",{text:`Refresh chained (${u.refreshChained.length})`});let y=g.createEl("ul");y.setCssStyles({marginTop:"4px"});for(let v of u.refreshChained)y.createEl("li",{text:v})}let m=a.createDiv();if(m.setCssStyles({marginTop:"12px"}),m.createEl("strong",{text:`Needs review (${u.needsReview.length})`}),u.needsReview.length===0){m.createDiv({cls:"af-form-hint",text:"All clear. Nothing requires manual review from this lint pass."});return}let f=m.createDiv();f.setCssStyles({display:"flex"}),f.setCssStyles({flexDirection:"column"}),f.setCssStyles({gap:"6px"}),f.setCssStyles({marginTop:"8px"});for(let g of u.needsReview){let y=f.createDiv();y.setCssStyles({display:"flex"}),y.setCssStyles({alignItems:"flex-start"}),y.setCssStyles({gap:"8px"}),y.setCssStyles({padding:"8px 10px"}),y.setCssStyles({background:"var(--background-secondary)"}),y.setCssStyles({borderRadius:"4px"}),y.setCssStyles({fontSize:"13px"});let v=y.createDiv();v.setCssStyles({flex:"1"}),v.setText(g);let k=y.createEl("button",{cls:"af-btn-sm",text:"Dismiss"});k.title="Hide this item from the dashboard until the next lint pass (does not modify log.md).",k.onclick=()=>{y.remove()}}}renderChannelCard(e,s,n){let a=this.plugin.channelManager?.getChannelStatus(s.name)??"disabled",i=Qo(a),o=s.enabled&&a!=="disabled"?"af-agent-card":"af-agent-card disabled",c=e.createDiv({cls:o});c.setCssStyles({cursor:"default"});let l=c.createDiv({cls:"af-agent-card-header"}),h=l.createDiv({cls:`af-agent-card-avatar ${i}`});(0,b.setIcon)(h,"radio");let d=l.createDiv({cls:"af-agent-card-titleblock"});d.createDiv({cls:"af-agent-card-name",text:s.name}),d.createDiv({cls:"af-agent-card-desc",text:`Default: ${s.defaultAgent}`});let u=l.createSpan({cls:`af-pill ${Sh(a)}`});if(u.createSpan({cls:"af-dot"}),u.appendText(` ${a}`),s.allowedAgents.length>0){let T=c.createDiv({cls:"af-agent-card-skills"});for(let _ of s.allowedAgents){let D=T.createSpan({cls:"af-skill-tag",text:_});_===s.defaultAgent&&D.setCssStyles({fontWeight:"700"})}}let p=c.createDiv({cls:"af-agent-card-stats"}),m=this.plugin.channelManager?.getSessionCount(s.name)??0,f=this.plugin.channelManager?.getMetrics(s.name),g=s.allowedAgents.length>0?String(s.allowedAgents.length):"all";this.renderAgentStat(p,g,"Agents"),this.renderAgentStat(p,String(m),"Sessions"),this.renderAgentStat(p,String(f?.messagesReceived??0),"In"),this.renderAgentStat(p,String(f?.messagesSent??0),"Out");let y=c.createDiv({cls:"af-agent-card-footer"}),v=[s.type];s.enabled||v.push("disabled"),s.allowedUsers.length>0?v.push(`${s.allowedUsers.length} user(s)`):v.push("allowlist empty"),y.createSpan({cls:"af-agent-card-meta",text:v.join(" \xB7 ")});let w=y.createDiv({cls:"af-agent-card-actions"}).createEl("button",{cls:"af-btn-sm"});R(w,"edit","af-btn-icon"),w.appendText(" Edit"),w.onclick=T=>{T.stopPropagation(),this.navigate("edit-channel",s.name)};let S=n.filter(T=>T.path===s.filePath);if(S.length>0){let T=c.createDiv({cls:"af-channel-issues"});for(let _ of S)T.createDiv({cls:"af-channel-issue-row",text:_.message})}}renderCreateChannelPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.plugin.runtime.getSnapshot(),a=this.plugin.channelCredentials.list(),i=s.createDiv({cls:"af-detail-header"}),o=i.createDiv({cls:"af-detail-header-left"}),c=o.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(c,"plus");let l=o.createDiv();l.createDiv({cls:"af-detail-header-name",text:"Create New Channel"}),l.createDiv({cls:"af-detail-header-desc",text:"Connect an external chat transport to an agent"});let h=i.createDiv({cls:"af-detail-header-actions"}),d={name:"",type:"slack",defaultAgent:n.agents[0]?.name??"",allowedAgents:[],credentialRef:a[0]?.ref??"",allowedUsers:"",perUserSessions:!0,channelContext:"",enabled:!0,tags:"",body:"",transportJson:""},u=s.createDiv({cls:"af-create-form"}),p=u.createDiv({cls:"af-create-section"}),m=p.createDiv({cls:"af-create-section-header"}),f=m.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(f,"radio"),m.createSpan({text:"Channel Details"}),this.createFormField(p,"Name","my-slack","Unique identifier for this channel",j=>{d.name=j});let g=p.createDiv({cls:"af-form-row"});g.createDiv({cls:"af-form-label",text:"Type"});let y=g.createEl("select",{cls:"af-form-select"});y.createEl("option",{text:"slack",attr:{value:"slack"}}),y.createEl("option",{text:"telegram",attr:{value:"telegram"}}),y.createEl("option",{text:"discord",attr:{value:"discord"}}),y.addEventListener("change",()=>{d.type=y.value});let v=p.createDiv({cls:"af-form-row"}),k=v.createDiv({cls:"af-form-label"});k.setText("Credential"),this.addTooltip(k,"Configured in Settings \u2192 Agent Fleet \u2192 Channel Credentials");let w=v.createEl("select",{cls:"af-form-select"});a.length===0&&w.createEl("option",{text:"(no credentials configured)",attr:{value:""}});for(let j of a)w.createEl("option",{text:`${j.ref} (${j.entry.type})`,attr:{value:j.ref}});w.addEventListener("change",()=>{d.credentialRef=w.value});let S=p.createDiv({cls:"af-form-row af-form-row-toggle"});S.createDiv({cls:"af-form-label",text:"Enabled"});let T=S.createDiv({cls:"af-agent-card-toggle on"});T.onclick=()=>{let j=T.hasClass("on");T.toggleClass("on",!j),d.enabled=!j};let _=u.createDiv({cls:"af-create-section"}),D=_.createDiv({cls:"af-create-section-header"}),O=D.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(O,"bot"),D.createSpan({text:"Agent Routing"});let C=_.createDiv({cls:"af-form-row"}),E=C.createDiv({cls:"af-form-label"});E.setText("Default agent"),this.addTooltip(E,"Used when no @agent-name prefix is given in a message");let P=C.createEl("select",{cls:"af-form-select"});for(let j of n.agents)P.createEl("option",{text:j.name,attr:{value:j.name}});P.addEventListener("change",()=>{d.defaultAgent=P.value});let N=_.createDiv({cls:"af-form-row"}),M=N.createDiv({cls:"af-form-label"});M.setText("Allowed agents"),this.addTooltip(M,"Agents reachable via @prefix. Leave unchecked to allow all agents.");let U=N.createDiv({cls:"af-form-checkboxes"});for(let j of n.agents){let Ce=U.createEl("label",{cls:"af-form-checkbox-label"}),xe=Ce.createEl("input",{attr:{type:"checkbox",value:j.name}});Ce.appendText(` ${j.name}`),xe.addEventListener("change",()=>{xe.checked?d.allowedAgents.includes(j.name)||d.allowedAgents.push(j.name):d.allowedAgents=d.allowedAgents.filter(he=>he!==j.name)})}let $=_.createDiv({cls:"af-form-row af-form-row-toggle"}),Z=$.createDiv({cls:"af-form-label"});Z.setText("Per-user sessions"),this.addTooltip(Z,"Each external user gets their own isolated Claude session");let de=$.createDiv({cls:"af-agent-card-toggle on"});de.onclick=()=>{let j=de.hasClass("on");de.toggleClass("on",!j),d.perUserSessions=!j};let me=u.createDiv({cls:"af-create-section"}),ge=me.createDiv({cls:"af-create-section-header"}),X=ge.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(X,"shield-check"),ge.createSpan({text:"Access Control"});let W=me.createDiv({cls:"af-form-label"});W.setText("Allowed users"),this.addTooltip(W,"User IDs, one per line \u2014 Slack (U...), Telegram (numeric), or Discord (numeric snowflakes). Only listed users can reach the bot.");let K=me.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`U0AQW6P37N1
12040
11951
  U0BXYZ12345`,rows:"4"}});K.addEventListener("input",()=>{d.allowedUsers=K.value});let ne=u.createDiv({cls:"af-create-section"}),G=ne.createDiv({cls:"af-create-section-header"}),ee=G.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(ee,"message-square");let Q=G.createSpan({text:"Channel Context"});this.addTooltip(Q,"Extra instructions appended to the agent's system prompt when reached through this channel");let ae=ne.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"You are being contacted via Slack. Keep replies concise...",rows:"6"}});ae.addEventListener("input",()=>{d.channelContext=ae.value}),this.createFormField(p,"Tags","ops, internal","Comma-separated metadata",j=>{d.tags=j});let ve=u.createDiv({cls:"af-create-section"}),Pe=ve.createDiv({cls:"af-create-section-header"}),Ie=Pe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ie,"settings");let Me=Pe.createSpan({text:"Advanced"});this.addTooltip(Me,"Markdown body (shown in the channel detail page) and transport-specific overrides");let Fe=ve.createDiv({cls:"af-form-label"});Fe.setText("Body (markdown)"),this.addTooltip(Fe,"Free-form notes for this channel. Shown in the channel detail page; not sent to the agent.");let ke=ve.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Notes, runbook snippets, escalation contacts\u2026",rows:"4"}});ke.addEventListener("input",()=>{d.body=ke.value});let Oe=ve.createDiv({cls:"af-form-label"});Oe.setText("Transport config (JSON)"),this.addTooltip(Oe,"Optional JSON object for transport-specific overrides (e.g. Slack socket_mode, telegram webhook settings). Leave blank for defaults.");let Ue=ve.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`{
12041
11952
  "socket_mode": true
12042
- }`,rows:"4"}});Ue.addEventListener("input",()=>{d.transportJson=Ue.value});let Ne=s.createDiv({cls:"af-create-footer"}),q=Ne.createEl("button",{cls:"af-btn-sm",text:"Cancel"});q.onclick=()=>this.navigate("channels");let ie=Ne.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(ie,"plus","af-btn-icon"),ie.appendText(" Create Channel"),ie.onclick=async()=>{let j=d.name.trim();if(!j){new b.Notice("Channel name is required.");return}if(!d.credentialRef){new b.Notice("Select a credential.");return}let Ce;if(d.transportJson.trim())try{let re=JSON.parse(d.transportJson);if(re&&typeof re=="object"&&!Array.isArray(re))Ce=re;else{new b.Notice("Transport config must be a JSON object.");return}}catch(re){new b.Notice(`Transport JSON is invalid: ${re instanceof Error?re.message:String(re)}`);return}let xe=re=>re.split(/[\n,]+/).map(I=>I.trim()).filter(Boolean),he=re=>re.split(",").map(I=>I.trim()).filter(Boolean),$e={name:oe(j),type:d.type,default_agent:d.defaultAgent,allowed_agents:d.allowedAgents.length>0?d.allowedAgents:void 0,enabled:d.enabled,credential_ref:d.credentialRef,allowed_users:xe(d.allowedUsers),per_user_sessions:d.perUserSessions,channel_context:d.channelContext.trim()||void 0,tags:he(d.tags).length>0?he(d.tags):void 0,transport:Ce};try{let re=oe(j),I=await this.plugin.repository.getAvailablePath(this.plugin.repository.getSubfolder("channels"),re);await this.plugin.app.vault.create(I,H($e,d.body.trim())),new b.Notice(`Channel "${re}" created.`),await this.plugin.refreshFromVault(),this.navigate("edit-channel",re)}catch(re){let I=re instanceof Error?re.message:String(re);new b.Notice(`Failed to create channel: ${I}`)}}}renderEditChannelPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"radio","No channel selected","");return}let a=this.plugin.runtime.getSnapshot().channels.find(I=>I.name===n);if(!a){this.renderEmptyState(s,"radio","Channel not found",`Channel "${n}" was not found`);return}let r=this.plugin.runtime.getSnapshot(),o=this.plugin.channelCredentials.list(),c=this.plugin.channelManager?.getChannelStatus(a.name)??"disabled",l=s.createDiv({cls:"af-detail-header"}),h=l.createDiv({cls:"af-detail-header-left"}),d=h.createDiv({cls:`af-agent-card-avatar ${Xo(c)}`});(0,b.setIcon)(d,"radio");let u=h.createDiv();u.createDiv({cls:"af-detail-header-name",text:`Edit Channel: ${a.name}`}),u.createDiv({cls:"af-detail-header-desc",text:`Status: ${c}`});let p=l.createDiv({cls:"af-detail-header-actions"}),m={type:a.type,defaultAgent:a.defaultAgent,allowedAgents:[...a.allowedAgents],credentialRef:a.credentialRef,allowedUsers:a.allowedUsers.join(`
12043
- `),perUserSessions:a.perUserSessions,channelContext:a.channelContext,enabled:a.enabled,tags:a.tags.join(", "),body:a.body,transportJson:Object.keys(a.transport).length>0?JSON.stringify(a.transport,null,2):""},f=s.createDiv({cls:"af-create-form"}),g=f.createDiv({cls:"af-create-section"}),y=g.createDiv({cls:"af-create-section-header"}),v=y.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(v,"radio"),y.createSpan({text:"Channel Details"});let k=g.createDiv({cls:"af-form-row"});k.createDiv({cls:"af-form-label",text:"Name"}),k.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.name,disabled:"true"}}).setCssStyles({opacity:"0.6"});let S=g.createDiv({cls:"af-form-row"});S.createDiv({cls:"af-form-label",text:"Type"});let T=S.createEl("select",{cls:"af-form-select"});for(let I of["slack","telegram","discord"]){let se=T.createEl("option",{text:I,attr:{value:I}});I===a.type&&(se.selected=!0)}T.addEventListener("change",()=>{m.type=T.value});let _=g.createDiv({cls:"af-form-row"}),D=_.createDiv({cls:"af-form-label"});D.setText("Credential"),this.addTooltip(D,"Configured in Settings \u2192 Agent Fleet \u2192 Channel Credentials");let O=_.createEl("select",{cls:"af-form-select"});o.length===0&&O.createEl("option",{text:"(no credentials configured)",attr:{value:""}});for(let I of o){let se=O.createEl("option",{text:`${I.ref} (${I.entry.type})`,attr:{value:I.ref}});I.ref===a.credentialRef&&(se.selected=!0)}O.addEventListener("change",()=>{m.credentialRef=O.value});let C=g.createDiv({cls:"af-form-row af-form-row-toggle"});C.createDiv({cls:"af-form-label",text:"Enabled"});let E=C.createDiv({cls:`af-agent-card-toggle${a.enabled?" on":""}`});E.onclick=()=>{let I=E.hasClass("on");E.toggleClass("on",!I),m.enabled=!I};let P=f.createDiv({cls:"af-create-section"}),N=P.createDiv({cls:"af-create-section-header"}),M=N.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(M,"bot"),N.createSpan({text:"Agent Routing"});let U=P.createDiv({cls:"af-form-row"}),$=U.createDiv({cls:"af-form-label"});$.setText("Default agent"),this.addTooltip($,"Used when no @agent-name prefix is given in a message");let Z=U.createEl("select",{cls:"af-form-select"});for(let I of r.agents){let se=Z.createEl("option",{text:I.name,attr:{value:I.name}});I.name===a.defaultAgent&&(se.selected=!0)}Z.addEventListener("change",()=>{m.defaultAgent=Z.value});let de=P.createDiv({cls:"af-form-row"}),me=de.createDiv({cls:"af-form-label"});me.setText("Allowed agents"),this.addTooltip(me,"Agents reachable via @prefix. Leave unchecked to allow all agents.");let ge=de.createDiv({cls:"af-form-checkboxes"});for(let I of r.agents){let se=ge.createEl("label",{cls:"af-form-checkbox-label"}),ce=se.createEl("input",{attr:{type:"checkbox",value:I.name}});a.allowedAgents.includes(I.name)&&(ce.checked=!0),se.appendText(` ${I.name}`),ce.addEventListener("change",()=>{ce.checked?m.allowedAgents.includes(I.name)||m.allowedAgents.push(I.name):m.allowedAgents=m.allowedAgents.filter(te=>te!==I.name)})}let X=P.createDiv({cls:"af-form-row af-form-row-toggle"}),W=X.createDiv({cls:"af-form-label"});W.setText("Per-user sessions"),this.addTooltip(W,"Each external user gets their own isolated Claude session");let K=X.createDiv({cls:`af-agent-card-toggle${a.perUserSessions?" on":""}`});K.onclick=()=>{let I=K.hasClass("on");K.toggleClass("on",!I),m.perUserSessions=!I};let ne=f.createDiv({cls:"af-create-section"}),G=ne.createDiv({cls:"af-create-section-header"}),ee=G.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(ee,"shield-check"),G.createSpan({text:"Access Control"});let Q=ne.createDiv({cls:"af-form-label"});Q.setText("Allowed users"),this.addTooltip(Q,"Slack user IDs (U...), one per line. Only listed users can reach the bot.");let ae=ne.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`U0AQW6P37N1
11953
+ }`,rows:"4"}});Ue.addEventListener("input",()=>{d.transportJson=Ue.value});let Ne=s.createDiv({cls:"af-create-footer"}),q=Ne.createEl("button",{cls:"af-btn-sm",text:"Cancel"});q.onclick=()=>this.navigate("channels");let re=Ne.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(re,"plus","af-btn-icon"),re.appendText(" Create Channel"),re.onclick=async()=>{let j=d.name.trim();if(!j){new b.Notice("Channel name is required.");return}if(!d.credentialRef){new b.Notice("Select a credential.");return}let Ce;if(d.transportJson.trim())try{let ie=JSON.parse(d.transportJson);if(ie&&typeof ie=="object"&&!Array.isArray(ie))Ce=ie;else{new b.Notice("Transport config must be a JSON object.");return}}catch(ie){new b.Notice(`Transport JSON is invalid: ${ie instanceof Error?ie.message:String(ie)}`);return}let xe=ie=>ie.split(/[\n,]+/).map(I=>I.trim()).filter(Boolean),he=ie=>ie.split(",").map(I=>I.trim()).filter(Boolean),$e={name:oe(j),type:d.type,default_agent:d.defaultAgent,allowed_agents:d.allowedAgents.length>0?d.allowedAgents:void 0,enabled:d.enabled,credential_ref:d.credentialRef,allowed_users:xe(d.allowedUsers),per_user_sessions:d.perUserSessions,channel_context:d.channelContext.trim()||void 0,tags:he(d.tags).length>0?he(d.tags):void 0,transport:Ce};try{let ie=oe(j),I=await this.plugin.repository.getAvailablePath(this.plugin.repository.getSubfolder("channels"),ie);await this.plugin.app.vault.create(I,H($e,d.body.trim())),new b.Notice(`Channel "${ie}" created.`),await this.plugin.refreshFromVault(),this.navigate("edit-channel",ie)}catch(ie){let I=ie instanceof Error?ie.message:String(ie);new b.Notice(`Failed to create channel: ${I}`)}}}renderEditChannelPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"radio","No channel selected","");return}let a=this.plugin.runtime.getSnapshot().channels.find(I=>I.name===n);if(!a){this.renderEmptyState(s,"radio","Channel not found",`Channel "${n}" was not found`);return}let i=this.plugin.runtime.getSnapshot(),o=this.plugin.channelCredentials.list(),c=this.plugin.channelManager?.getChannelStatus(a.name)??"disabled",l=s.createDiv({cls:"af-detail-header"}),h=l.createDiv({cls:"af-detail-header-left"}),d=h.createDiv({cls:`af-agent-card-avatar ${Qo(c)}`});(0,b.setIcon)(d,"radio");let u=h.createDiv();u.createDiv({cls:"af-detail-header-name",text:`Edit Channel: ${a.name}`}),u.createDiv({cls:"af-detail-header-desc",text:`Status: ${c}`});let p=l.createDiv({cls:"af-detail-header-actions"}),m={type:a.type,defaultAgent:a.defaultAgent,allowedAgents:[...a.allowedAgents],credentialRef:a.credentialRef,allowedUsers:a.allowedUsers.join(`
11954
+ `),perUserSessions:a.perUserSessions,channelContext:a.channelContext,enabled:a.enabled,tags:a.tags.join(", "),body:a.body,transportJson:Object.keys(a.transport).length>0?JSON.stringify(a.transport,null,2):""},f=s.createDiv({cls:"af-create-form"}),g=f.createDiv({cls:"af-create-section"}),y=g.createDiv({cls:"af-create-section-header"}),v=y.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(v,"radio"),y.createSpan({text:"Channel Details"});let k=g.createDiv({cls:"af-form-row"});k.createDiv({cls:"af-form-label",text:"Name"}),k.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.name,disabled:"true"}}).setCssStyles({opacity:"0.6"});let S=g.createDiv({cls:"af-form-row"});S.createDiv({cls:"af-form-label",text:"Type"});let T=S.createEl("select",{cls:"af-form-select"});for(let I of["slack","telegram","discord"]){let se=T.createEl("option",{text:I,attr:{value:I}});I===a.type&&(se.selected=!0)}T.addEventListener("change",()=>{m.type=T.value});let _=g.createDiv({cls:"af-form-row"}),D=_.createDiv({cls:"af-form-label"});D.setText("Credential"),this.addTooltip(D,"Configured in Settings \u2192 Agent Fleet \u2192 Channel Credentials");let O=_.createEl("select",{cls:"af-form-select"});o.length===0&&O.createEl("option",{text:"(no credentials configured)",attr:{value:""}});for(let I of o){let se=O.createEl("option",{text:`${I.ref} (${I.entry.type})`,attr:{value:I.ref}});I.ref===a.credentialRef&&(se.selected=!0)}O.addEventListener("change",()=>{m.credentialRef=O.value});let C=g.createDiv({cls:"af-form-row af-form-row-toggle"});C.createDiv({cls:"af-form-label",text:"Enabled"});let E=C.createDiv({cls:`af-agent-card-toggle${a.enabled?" on":""}`});E.onclick=()=>{let I=E.hasClass("on");E.toggleClass("on",!I),m.enabled=!I};let P=f.createDiv({cls:"af-create-section"}),N=P.createDiv({cls:"af-create-section-header"}),M=N.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(M,"bot"),N.createSpan({text:"Agent Routing"});let U=P.createDiv({cls:"af-form-row"}),$=U.createDiv({cls:"af-form-label"});$.setText("Default agent"),this.addTooltip($,"Used when no @agent-name prefix is given in a message");let Z=U.createEl("select",{cls:"af-form-select"});for(let I of i.agents){let se=Z.createEl("option",{text:I.name,attr:{value:I.name}});I.name===a.defaultAgent&&(se.selected=!0)}Z.addEventListener("change",()=>{m.defaultAgent=Z.value});let de=P.createDiv({cls:"af-form-row"}),me=de.createDiv({cls:"af-form-label"});me.setText("Allowed agents"),this.addTooltip(me,"Agents reachable via @prefix. Leave unchecked to allow all agents.");let ge=de.createDiv({cls:"af-form-checkboxes"});for(let I of i.agents){let se=ge.createEl("label",{cls:"af-form-checkbox-label"}),ce=se.createEl("input",{attr:{type:"checkbox",value:I.name}});a.allowedAgents.includes(I.name)&&(ce.checked=!0),se.appendText(` ${I.name}`),ce.addEventListener("change",()=>{ce.checked?m.allowedAgents.includes(I.name)||m.allowedAgents.push(I.name):m.allowedAgents=m.allowedAgents.filter(te=>te!==I.name)})}let X=P.createDiv({cls:"af-form-row af-form-row-toggle"}),W=X.createDiv({cls:"af-form-label"});W.setText("Per-user sessions"),this.addTooltip(W,"Each external user gets their own isolated Claude session");let K=X.createDiv({cls:`af-agent-card-toggle${a.perUserSessions?" on":""}`});K.onclick=()=>{let I=K.hasClass("on");K.toggleClass("on",!I),m.perUserSessions=!I};let ne=f.createDiv({cls:"af-create-section"}),G=ne.createDiv({cls:"af-create-section-header"}),ee=G.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(ee,"shield-check"),G.createSpan({text:"Access Control"});let Q=ne.createDiv({cls:"af-form-label"});Q.setText("Allowed users"),this.addTooltip(Q,"Slack user IDs (U...), one per line. Only listed users can reach the bot.");let ae=ne.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`U0AQW6P37N1
12044
11955
  U0BXYZ12345`,rows:"4"}});ae.value=a.allowedUsers.join(`
12045
- `),ae.addEventListener("input",()=>{m.allowedUsers=ae.value});let ve=f.createDiv({cls:"af-create-section"}),Pe=ve.createDiv({cls:"af-create-section-header"}),Ie=Pe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ie,"message-square");let Me=Pe.createSpan({text:"Channel Context"});this.addTooltip(Me,"Extra instructions appended to the agent's system prompt when reached through this channel");let Fe=ve.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"You are being contacted via Slack. Keep replies concise...",rows:"6"}});Fe.value=a.channelContext,Fe.addEventListener("input",()=>{m.channelContext=Fe.value}),this.createFormField(g,"Tags","ops, internal","Comma-separated metadata",I=>{m.tags=I},a.tags.join(", "));let ke=f.createDiv({cls:"af-create-section"}),Oe=ke.createDiv({cls:"af-create-section-header"}),Ue=Oe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ue,"settings");let Ne=Oe.createSpan({text:"Advanced"});this.addTooltip(Ne,"Markdown body (shown in the channel detail page) and transport-specific overrides"),ke.createDiv({cls:"af-form-label"}).setText("Body (markdown)");let ie=ke.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Notes, runbook snippets, escalation contacts\u2026",rows:"4"}});ie.value=a.body,ie.addEventListener("input",()=>{m.body=ie.value});let j=ke.createDiv({cls:"af-form-label"});j.setText("Transport config (JSON)"),this.addTooltip(j,"Optional JSON object for transport-specific overrides (e.g. Slack socket_mode). Leave blank for defaults.");let Ce=ke.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`{
11956
+ `),ae.addEventListener("input",()=>{m.allowedUsers=ae.value});let ve=f.createDiv({cls:"af-create-section"}),Pe=ve.createDiv({cls:"af-create-section-header"}),Ie=Pe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ie,"message-square");let Me=Pe.createSpan({text:"Channel Context"});this.addTooltip(Me,"Extra instructions appended to the agent's system prompt when reached through this channel");let Fe=ve.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"You are being contacted via Slack. Keep replies concise...",rows:"6"}});Fe.value=a.channelContext,Fe.addEventListener("input",()=>{m.channelContext=Fe.value}),this.createFormField(g,"Tags","ops, internal","Comma-separated metadata",I=>{m.tags=I},a.tags.join(", "));let ke=f.createDiv({cls:"af-create-section"}),Oe=ke.createDiv({cls:"af-create-section-header"}),Ue=Oe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ue,"settings");let Ne=Oe.createSpan({text:"Advanced"});this.addTooltip(Ne,"Markdown body (shown in the channel detail page) and transport-specific overrides"),ke.createDiv({cls:"af-form-label"}).setText("Body (markdown)");let re=ke.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Notes, runbook snippets, escalation contacts\u2026",rows:"4"}});re.value=a.body,re.addEventListener("input",()=>{m.body=re.value});let j=ke.createDiv({cls:"af-form-label"});j.setText("Transport config (JSON)"),this.addTooltip(j,"Optional JSON object for transport-specific overrides (e.g. Slack socket_mode). Leave blank for defaults.");let Ce=ke.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`{
12046
11957
  "socket_mode": true
12047
- }`,rows:"4"}});Ce.value=m.transportJson,Ce.addEventListener("input",()=>{m.transportJson=Ce.value});let xe=s.createDiv({cls:"af-create-footer"}),he=xe.createEl("button",{cls:"af-btn-sm danger"});R(he,"trash-2","af-btn-icon"),he.appendText(" Delete"),he.onclick=async()=>{await this.plugin.repository.deleteChannel(a.name),new b.Notice(`Channel "${a.name}" deleted.`),await new Promise(I=>window.setTimeout(I,200)),await this.plugin.refreshFromVault(),this.navigate("channels")},xe.createDiv({cls:"af-toolbar-spacer"});let $e=xe.createEl("button",{cls:"af-btn-sm",text:"Cancel"});$e.onclick=()=>this.navigate("channels");let re=xe.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(re,"check","af-btn-icon"),re.appendText(" Save Changes"),re.onclick=async()=>{let I=te=>te.split(/[\n,]+/).map(st=>st.trim()).filter(Boolean),se=te=>te.split(",").map(st=>st.trim()).filter(Boolean),ce;if(m.transportJson.trim())try{let te=JSON.parse(m.transportJson);if(te&&typeof te=="object"&&!Array.isArray(te))ce=te;else{new b.Notice("Transport config must be a JSON object.");return}}catch(te){new b.Notice(`Transport JSON is invalid: ${te instanceof Error?te.message:String(te)}`);return}else ce={};try{await this.plugin.repository.updateChannel(a.name,{type:m.type,default_agent:m.defaultAgent,allowed_agents:m.allowedAgents.length>0?m.allowedAgents:[],enabled:m.enabled,credential_ref:m.credentialRef,allowed_users:I(m.allowedUsers),per_user_sessions:m.perUserSessions,channel_context:m.channelContext.trim(),tags:se(m.tags),body:m.body.trim(),transport:ce}),new b.Notice(`Channel "${a.name}" updated.`),await this.plugin.refreshFromVault(),this.navigate("channels")}catch(te){let st=te instanceof Error?te.message:String(te);new b.Notice(`Failed to update channel: ${st}`)}}}renderApprovalsPage(e){let s=e.createDiv({cls:"af-approvals-page"}),n=this.plugin.runtime.getRecentRuns(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Approvals"}),a.createDiv({cls:"af-toolbar-spacer"});let r=n.filter(c=>(c.approvals??[]).some(l=>l.status==="pending"));if(r.length>0){let c=s.createDiv({cls:"af-section-card"}),h=c.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(h,"alert-triangle"),h.appendText(` Pending (${r.length})`);let d=c.createDiv({cls:"af-approvals-list"});for(let u of r)this.renderApprovalItem(d,u,!0)}else{let c=s.createDiv({cls:"af-section-card"});this.renderEmptyState(c,"shield-check","No pending approvals","All clear!")}let o=n.filter(c=>(c.approvals??[]).some(l=>l.status!=="pending"));if(o.length>0){let c=s.createDiv({cls:"af-section-card"}),h=c.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(h,"check-circle-2"),h.appendText(" History");let d=c.createDiv({cls:"af-approvals-list"});for(let u of o.slice(0,20))this.renderApprovalItem(d,u,!1)}}renderApprovalItem(e,s,n){for(let a of s.approvals??[]){if(n&&a.status!=="pending"||!n&&a.status==="pending")continue;let r=e.createDiv({cls:"af-approval-item"}),o=r.createDiv({cls:"af-approval-item-icon"});a.status==="pending"?((0,b.setIcon)(o,"shield-check"),o.addClass("pending")):a.status==="approved"?((0,b.setIcon)(o,"check-circle-2"),o.addClass("approved")):((0,b.setIcon)(o,"x-circle"),o.addClass("rejected"));let c=r.createDiv({cls:"af-approval-item-body"});if(c.createDiv({cls:"af-approval-item-title",text:`${s.agent} \u2192 ${a.tool}`}),c.createDiv({cls:"af-approval-item-meta",text:`Task: ${s.task} \xB7 ${a.command??"no command"} \xB7 ${this.formatStarted(s.started)}`}),a.reason&&c.createDiv({cls:"af-approval-item-reason",text:`Reason: ${a.reason}`}),n&&a.status==="pending"){let l=r.createDiv({cls:"af-approval-item-actions"}),h=l.createEl("button",{cls:"af-btn-approve"});R(h,"check-circle-2","af-btn-icon"),h.appendText(" Approve"),h.onclick=()=>void this.plugin.runtime.resolveApproval(s,a.tool,"approved").then(()=>this.render());let d=l.createEl("button",{cls:"af-btn-reject"});R(d,"x-circle","af-btn-icon"),d.appendText(" Reject"),d.onclick=()=>void this.plugin.runtime.resolveApproval(s,a.tool,"rejected").then(()=>this.render())}}}renderTaskDetailPage(e){let s=e.createDiv({cls:"af-task-detail-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"circle-dot","No task selected","");return}let a=this.plugin.runtime.getSnapshot().tasks.find(P=>P.taskId===n);if(!a){this.renderEmptyState(s,"circle-dot","Task not found",`Task "${n}" was not found`);return}let r=this.plugin.runtime.getSnapshot(),o=this.plugin.runtime.getRecentRuns().filter(P=>P.task===n),c=r.agents.find(P=>P.name===a.agent),l=s.createDiv({cls:"af-detail-header"}),h=l.createDiv({cls:"af-detail-header-left"}),d=h.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(d,"circle-dot");let u=h.createDiv();u.createDiv({cls:"af-detail-header-name",text:a.taskId}),u.createDiv({cls:"af-detail-header-desc",text:`Agent: ${a.agent}`});let p=l.createDiv({cls:"af-detail-header-actions"}),m=p.createEl("button",{cls:"af-btn-sm"});R(m,"edit","af-btn-icon"),m.appendText(" Edit"),m.onclick=()=>this.navigate("edit-task",a.taskId);let f=p.createEl("button",{cls:"af-btn-sm primary"});R(f,"play","af-btn-icon"),f.appendText(" Run Now"),f.onclick=()=>void this.plugin.runtime.runTaskNow(a);let g=s.createDiv({cls:"af-section-card"}),v=g.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(v,"file-text"),v.appendText(" Details");let k=g.createDiv({cls:"af-config-form"});this.renderConfigRow(k,"Agent",a.agent),this.renderConfigRow(k,"Priority",a.priority.charAt(0).toUpperCase()+a.priority.slice(1)),this.renderConfigRow(k,"Status",a.enabled?"Enabled":"Disabled");let w=a.schedule?this.humanizeCron(a.schedule):a.runAt??"Manual (run on demand)";this.renderConfigRow(k,"Schedule",w),a.schedule&&this.renderConfigRow(k,"Catch up if missed",a.catchUp?"Yes":"No"),this.renderConfigRow(k,"Created",a.created),this.renderConfigRow(k,"Runs",String(a.runCount)),a.lastRun&&this.renderConfigRow(k,"Last Run",this.formatStarted(a.lastRun));let S=s.createDiv({cls:"af-section-card"}),_=S.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(_,"message-square"),_.appendText(" Instructions"),S.createDiv({cls:"af-output-block",text:a.body||"(empty)"});let D=s.createDiv({cls:"af-section-card"}),C=D.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(C,"scroll-text"),C.appendText(" Recent Runs");let E=D.createDiv({cls:"af-timeline"});if(o.length===0)this.renderEmptyState(E,"scroll-text","No runs yet","");else for(let P of o.slice(0,10))this.renderTimelineItem(E,P)}handleSearch(e,s){if(s.querySelector(".af-search-results")?.remove(),e.length<2)return;let n=e.toLowerCase(),a=this.plugin.runtime.getSnapshot(),r=this.plugin.runtime.getRecentRuns(),o=[];for(let l of a.agents)(l.name.toLowerCase().includes(n)||(l.description?.toLowerCase().includes(n)??!1))&&o.push({label:`Agent: ${l.name}`,icon:"bot",action:()=>this.navigate("agent-detail",l.name)});for(let l of a.tasks)(l.taskId.toLowerCase().includes(n)||l.body.toLowerCase().includes(n))&&o.push({label:`Task: ${l.taskId}`,icon:"circle-dot",action:()=>this.navigate("task-detail",l.taskId)});for(let l of a.skills)l.name.toLowerCase().includes(n)&&o.push({label:`Skill: ${l.name}`,icon:"puzzle",action:()=>this.navigate("edit-skill",l.name)});for(let l of r.slice(0,20))l.output.toLowerCase().includes(n)&&o.push({label:`Run: ${l.agent} / ${l.task}`,icon:"scroll-text",action:()=>this.openSlideover(l)});if(o.length===0)return;let c=s.createDiv({cls:"af-search-results"});for(let l of o.slice(0,10)){let h=c.createDiv({cls:"af-search-result-item"});R(h,l.icon,"af-search-result-icon"),h.createSpan({text:l.label}),h.onclick=()=>{c.remove(),l.action()}}}openChatSlideover(e){this.plugin.openChatView(e.name)}openSlideover(e){this.contentEl.querySelector(".af-slideover-overlay")?.remove();let s=this.contentEl.createDiv({cls:"af-slideover-overlay"}),n=s.createDiv({cls:"af-slideover"}),a=n.createDiv({cls:"af-slideover-header"});a.createDiv({cls:"af-slideover-title",text:"Run Details"});let r=a.createEl("button",{cls:"clickable-icon"});(0,b.setIcon)(r,"cross"),r.onclick=()=>s.remove();let o=n.createDiv({cls:"af-slideover-body"}),c=o.createDiv({cls:"af-slideover-section"});c.createDiv({cls:"af-slideover-section-title",text:"METADATA"}),this.renderDetailRow(c,"Run ID",e.runId.slice(0,8)),this.renderDetailRow(c,"Agent",e.agent),this.renderDetailRow(c,"Task",e.task);let l=c.createDiv({cls:"af-detail-row"});l.createSpan({cls:"af-detail-label",text:"Status"});let d=l.createSpan({cls:"af-detail-value"}).createSpan({cls:`af-status-badge ${this.statusToBadgeClass(e.status)}`}),u=d.createSpan();(0,b.setIcon)(u,this.statusToIconName(e.status)),d.appendText(` ${this.statusToBadgeText(e.status)}`),this.renderDetailRow(c,"Started",e.started),this.renderDetailRow(c,"Duration",this.formatDuration(e.durationSeconds)),this.renderDetailRow(c,"Tokens",e.tokensUsed?e.tokensUsed.toLocaleString():"\u2014");{let w={task:"from task override",agent:"from agent",settings:"from settings default","cli-default":"CLI default"},S=e.modelSource?` (${w[e.modelSource]??e.modelSource})`:"",T=e.concreteModel&&e.concreteModel!==e.model?` \u2192 ${e.concreteModel}`:"";this.renderDetailRow(c,"Model",`${e.model}${T}${S}`)}let p=5e4,m=w=>w.length>p?w.slice(0,p)+`
11958
+ }`,rows:"4"}});Ce.value=m.transportJson,Ce.addEventListener("input",()=>{m.transportJson=Ce.value});let xe=s.createDiv({cls:"af-create-footer"}),he=xe.createEl("button",{cls:"af-btn-sm danger"});R(he,"trash-2","af-btn-icon"),he.appendText(" Delete"),he.onclick=async()=>{await this.plugin.repository.deleteChannel(a.name),new b.Notice(`Channel "${a.name}" deleted.`),await new Promise(I=>window.setTimeout(I,200)),await this.plugin.refreshFromVault(),this.navigate("channels")},xe.createDiv({cls:"af-toolbar-spacer"});let $e=xe.createEl("button",{cls:"af-btn-sm",text:"Cancel"});$e.onclick=()=>this.navigate("channels");let ie=xe.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(ie,"check","af-btn-icon"),ie.appendText(" Save Changes"),ie.onclick=async()=>{let I=te=>te.split(/[\n,]+/).map(st=>st.trim()).filter(Boolean),se=te=>te.split(",").map(st=>st.trim()).filter(Boolean),ce;if(m.transportJson.trim())try{let te=JSON.parse(m.transportJson);if(te&&typeof te=="object"&&!Array.isArray(te))ce=te;else{new b.Notice("Transport config must be a JSON object.");return}}catch(te){new b.Notice(`Transport JSON is invalid: ${te instanceof Error?te.message:String(te)}`);return}else ce={};try{await this.plugin.repository.updateChannel(a.name,{type:m.type,default_agent:m.defaultAgent,allowed_agents:m.allowedAgents.length>0?m.allowedAgents:[],enabled:m.enabled,credential_ref:m.credentialRef,allowed_users:I(m.allowedUsers),per_user_sessions:m.perUserSessions,channel_context:m.channelContext.trim(),tags:se(m.tags),body:m.body.trim(),transport:ce}),new b.Notice(`Channel "${a.name}" updated.`),await this.plugin.refreshFromVault(),this.navigate("channels")}catch(te){let st=te instanceof Error?te.message:String(te);new b.Notice(`Failed to update channel: ${st}`)}}}renderApprovalsPage(e){let s=e.createDiv({cls:"af-approvals-page"}),n=this.plugin.runtime.getRecentRuns(),a=s.createDiv({cls:"af-agents-toolbar"});a.createDiv({cls:"af-page-title",text:"Approvals"}),a.createDiv({cls:"af-toolbar-spacer"});let i=n.filter(c=>(c.approvals??[]).some(l=>l.status==="pending"));if(i.length>0){let c=s.createDiv({cls:"af-section-card"}),h=c.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(h,"alert-triangle"),h.appendText(` Pending (${i.length})`);let d=c.createDiv({cls:"af-approvals-list"});for(let u of i)this.renderApprovalItem(d,u,!0)}else{let c=s.createDiv({cls:"af-section-card"});this.renderEmptyState(c,"shield-check","No pending approvals","All clear!")}let o=n.filter(c=>(c.approvals??[]).some(l=>l.status!=="pending"));if(o.length>0){let c=s.createDiv({cls:"af-section-card"}),h=c.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(h,"check-circle-2"),h.appendText(" History");let d=c.createDiv({cls:"af-approvals-list"});for(let u of o.slice(0,20))this.renderApprovalItem(d,u,!1)}}renderApprovalItem(e,s,n){for(let a of s.approvals??[]){if(n&&a.status!=="pending"||!n&&a.status==="pending")continue;let i=e.createDiv({cls:"af-approval-item"}),o=i.createDiv({cls:"af-approval-item-icon"});a.status==="pending"?((0,b.setIcon)(o,"shield-check"),o.addClass("pending")):a.status==="approved"?((0,b.setIcon)(o,"check-circle-2"),o.addClass("approved")):((0,b.setIcon)(o,"x-circle"),o.addClass("rejected"));let c=i.createDiv({cls:"af-approval-item-body"});if(c.createDiv({cls:"af-approval-item-title",text:`${s.agent} \u2192 ${a.tool}`}),c.createDiv({cls:"af-approval-item-meta",text:`Task: ${s.task} \xB7 ${a.command??"no command"} \xB7 ${this.formatStarted(s.started)}`}),a.reason&&c.createDiv({cls:"af-approval-item-reason",text:`Reason: ${a.reason}`}),n&&a.status==="pending"){let l=i.createDiv({cls:"af-approval-item-actions"}),h=l.createEl("button",{cls:"af-btn-approve"});R(h,"check-circle-2","af-btn-icon"),h.appendText(" Approve"),h.onclick=()=>void this.plugin.runtime.resolveApproval(s,a.tool,"approved").then(()=>this.render());let d=l.createEl("button",{cls:"af-btn-reject"});R(d,"x-circle","af-btn-icon"),d.appendText(" Reject"),d.onclick=()=>void this.plugin.runtime.resolveApproval(s,a.tool,"rejected").then(()=>this.render())}}}renderTaskDetailPage(e){let s=e.createDiv({cls:"af-task-detail-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"circle-dot","No task selected","");return}let a=this.plugin.runtime.getSnapshot().tasks.find(P=>P.taskId===n);if(!a){this.renderEmptyState(s,"circle-dot","Task not found",`Task "${n}" was not found`);return}let i=this.plugin.runtime.getSnapshot(),o=this.plugin.runtime.getRecentRuns().filter(P=>P.task===n),c=i.agents.find(P=>P.name===a.agent),l=s.createDiv({cls:"af-detail-header"}),h=l.createDiv({cls:"af-detail-header-left"}),d=h.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(d,"circle-dot");let u=h.createDiv();u.createDiv({cls:"af-detail-header-name",text:a.taskId}),u.createDiv({cls:"af-detail-header-desc",text:`Agent: ${a.agent}`});let p=l.createDiv({cls:"af-detail-header-actions"}),m=p.createEl("button",{cls:"af-btn-sm"});R(m,"edit","af-btn-icon"),m.appendText(" Edit"),m.onclick=()=>this.navigate("edit-task",a.taskId);let f=p.createEl("button",{cls:"af-btn-sm primary"});R(f,"play","af-btn-icon"),f.appendText(" Run Now"),f.onclick=()=>void this.plugin.runtime.runTaskNow(a);let g=s.createDiv({cls:"af-section-card"}),v=g.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(v,"file-text"),v.appendText(" Details");let k=g.createDiv({cls:"af-config-form"});this.renderConfigRow(k,"Agent",a.agent),this.renderConfigRow(k,"Priority",a.priority.charAt(0).toUpperCase()+a.priority.slice(1)),this.renderConfigRow(k,"Status",a.enabled?"Enabled":"Disabled");let w=a.schedule?this.humanizeCron(a.schedule):a.runAt??"Manual (run on demand)";this.renderConfigRow(k,"Schedule",w),a.schedule&&this.renderConfigRow(k,"Catch up if missed",a.catchUp?"Yes":"No"),this.renderConfigRow(k,"Created",a.created),this.renderConfigRow(k,"Runs",String(a.runCount)),a.lastRun&&this.renderConfigRow(k,"Last Run",this.formatStarted(a.lastRun));let S=s.createDiv({cls:"af-section-card"}),_=S.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(_,"message-square"),_.appendText(" Instructions"),S.createDiv({cls:"af-output-block",text:a.body||"(empty)"});let D=s.createDiv({cls:"af-section-card"}),C=D.createDiv({cls:"af-section-header"}).createDiv({cls:"af-section-title"});R(C,"scroll-text"),C.appendText(" Recent Runs");let E=D.createDiv({cls:"af-timeline"});if(o.length===0)this.renderEmptyState(E,"scroll-text","No runs yet","");else for(let P of o.slice(0,10))this.renderTimelineItem(E,P)}handleSearch(e,s){if(s.querySelector(".af-search-results")?.remove(),e.length<2)return;let n=e.toLowerCase(),a=this.plugin.runtime.getSnapshot(),i=this.plugin.runtime.getRecentRuns(),o=[];for(let l of a.agents)(l.name.toLowerCase().includes(n)||(l.description?.toLowerCase().includes(n)??!1))&&o.push({label:`Agent: ${l.name}`,icon:"bot",action:()=>this.navigate("agent-detail",l.name)});for(let l of a.tasks)(l.taskId.toLowerCase().includes(n)||l.body.toLowerCase().includes(n))&&o.push({label:`Task: ${l.taskId}`,icon:"circle-dot",action:()=>this.navigate("task-detail",l.taskId)});for(let l of a.skills)l.name.toLowerCase().includes(n)&&o.push({label:`Skill: ${l.name}`,icon:"puzzle",action:()=>this.navigate("edit-skill",l.name)});for(let l of i.slice(0,20))l.output.toLowerCase().includes(n)&&o.push({label:`Run: ${l.agent} / ${l.task}`,icon:"scroll-text",action:()=>this.openSlideover(l)});if(o.length===0)return;let c=s.createDiv({cls:"af-search-results"});for(let l of o.slice(0,10)){let h=c.createDiv({cls:"af-search-result-item"});R(h,l.icon,"af-search-result-icon"),h.createSpan({text:l.label}),h.onclick=()=>{c.remove(),l.action()}}}openChatSlideover(e){this.plugin.openChatView(e.name)}openSlideover(e){this.contentEl.querySelector(".af-slideover-overlay")?.remove();let s=this.contentEl.createDiv({cls:"af-slideover-overlay"}),n=s.createDiv({cls:"af-slideover"}),a=n.createDiv({cls:"af-slideover-header"});a.createDiv({cls:"af-slideover-title",text:"Run Details"});let i=a.createEl("button",{cls:"clickable-icon"});(0,b.setIcon)(i,"cross"),i.onclick=()=>s.remove();let o=n.createDiv({cls:"af-slideover-body"}),c=o.createDiv({cls:"af-slideover-section"});c.createDiv({cls:"af-slideover-section-title",text:"METADATA"}),this.renderDetailRow(c,"Run ID",e.runId.slice(0,8)),this.renderDetailRow(c,"Agent",e.agent),this.renderDetailRow(c,"Task",e.task);let l=c.createDiv({cls:"af-detail-row"});l.createSpan({cls:"af-detail-label",text:"Status"});let d=l.createSpan({cls:"af-detail-value"}).createSpan({cls:`af-status-badge ${this.statusToBadgeClass(e.status)}`}),u=d.createSpan();(0,b.setIcon)(u,this.statusToIconName(e.status)),d.appendText(` ${this.statusToBadgeText(e.status)}`),this.renderDetailRow(c,"Started",e.started),this.renderDetailRow(c,"Duration",this.formatDuration(e.durationSeconds)),this.renderDetailRow(c,"Tokens",e.tokensUsed?e.tokensUsed.toLocaleString():"\u2014");{let w={task:"from task override",agent:"from agent",settings:"from settings default","cli-default":"CLI default"},S=e.modelSource?` (${w[e.modelSource]??e.modelSource})`:"",T=e.concreteModel&&e.concreteModel!==e.model?` \u2192 ${e.concreteModel}`:"";this.renderDetailRow(c,"Model",`${e.model}${T}${S}`)}let p=5e4,m=w=>w.length>p?w.slice(0,p)+`
12048
11959
 
12049
11960
  ---
12050
11961
  *Truncated (${(w.length/1024).toFixed(0)} KB total). Open the run note for full content.*`:w,f=!!(e.finalResult&&e.finalResult.trim()),g=e.output?.trim()??"",y=f&&g.length>0&&g!==e.finalResult.trim();if(f){let w=o.createDiv({cls:"af-slideover-section"});w.createDiv({cls:"af-slideover-section-title",text:"OUTPUT"});let S=w.createDiv({cls:"af-output-block af-compact-md"});if(b.MarkdownRenderer.render(this.app,m(e.finalResult),S,"",this),y){let T=w.createEl("details",{cls:"af-run-transcript"}),_=T.createEl("summary");(0,b.setIcon)(_.createSpan({cls:"af-run-transcript-icon"}),"file-text"),_.createSpan({text:"Show full transcript"}),_.createSpan({cls:"af-run-transcript-meta"}).setText(`${(g.length/1024).toFixed(1)} KB`);let O=T.createDiv({cls:"af-output-block af-compact-md af-run-transcript-body"});b.MarkdownRenderer.render(this.app,m(g),O,"",this)}}else if(g){let w=o.createDiv({cls:"af-slideover-section"});w.createDiv({cls:"af-slideover-section-title",text:"OUTPUT"});let S=w.createDiv({cls:"af-output-block af-compact-md"});b.MarkdownRenderer.render(this.app,m(g),S,"",this)}if(e.toolsUsed.length>0){let w=o.createDiv({cls:"af-slideover-section"});w.createDiv({cls:"af-slideover-section-title",text:"TOOLS USED"}),w.createDiv({cls:"af-output-block",text:e.toolsUsed.join(`
12051
- `)})}let v=o.createDiv({cls:"af-slideover-actions"});if(e.filePath){let w=v.createEl("button",{cls:"af-btn-sm"});R(w,"external-link","af-btn-icon"),w.appendText(" Open Run Note"),w.onclick=()=>void this.plugin.openPath(e.filePath)}let k=v.createEl("button",{cls:"af-btn-sm primary"});R(k,"refresh-cw","af-btn-icon"),k.appendText(" Re-run Task"),k.onclick=()=>void this.plugin.runAgentPrompt(e.agent),s.onclick=w=>{w.target===s&&s.remove()}}renderDetailRow(e,s,n){let a=e.createDiv({cls:"af-detail-row"});a.createSpan({cls:"af-detail-label",text:s}),a.createSpan({cls:"af-detail-value af-mono",text:n})}renderEmptyState(e,s,n,a){let r=e.createDiv({cls:"af-empty-state"}),o=r.createDiv({cls:"af-empty-icon"});(0,b.setIcon)(o,s),r.createDiv({cls:"af-empty-label",text:n}),a&&r.createDiv({cls:"af-empty-sublabel",text:a})}healthToClass(e){switch(e){case"running":return"running";case"error":return"error";case"pending":return"pending";case"disabled":return"disabled";default:return"idle"}}statusToTimelineClass(e){switch(e){case"success":return"success";case"failure":case"timeout":return"error";case"cancelled":return"warning";case"pending_approval":return"pending";default:return"running"}}statusToIconName(e){switch(e){case"success":return"check-circle-2";case"failure":return"x-circle";case"timeout":return"clock";case"pending_approval":return"shield-check";case"cancelled":return"square";default:return"loader-2"}}statusToBadgeClass(e){switch(e){case"success":return"success";case"failure":return"failure";case"timeout":return"timeout";case"pending_approval":return"pending";case"cancelled":return"cancelled";default:return"running"}}statusToBadgeText(e){switch(e){case"success":return"Success";case"failure":return"Failed";case"timeout":return"Timeout";case"pending_approval":return"Pending";case"cancelled":return"Cancelled";case"interrupted":return"Interrupted";default:return e}}formatDuration(e){if(e<60)return`${e}s`;let s=Math.floor(e/60),n=e%60;return n>0?`${s}m ${n}s`:`${s}m`}formatStarted(e){try{let s=new Date(e),n=new Date;if(s.toDateString()===n.toDateString())return s.toLocaleTimeString([],{hour:"2-digit",minute:"2-digit"});let a=new Date(n);return a.setDate(a.getDate()-1),s.toDateString()===a.toDateString()?`Yesterday ${s.toLocaleTimeString([],{hour:"2-digit",minute:"2-digit"})}`:s.toLocaleDateString([],{month:"short",day:"numeric"})+` ${s.toLocaleTimeString([],{hour:"2-digit",minute:"2-digit"})}`}catch{return e}}formatNextRun(e){try{let s=new Date(e),n=new Date,a=s.getTime()-n.getTime();if(a<0)return"overdue";let r=Math.round(a/6e4);if(r<60)return`${r}m`;let o=Math.round(r/60);return o<24?`${o}h`:s.toLocaleDateString([],{month:"short",day:"numeric"})}catch{return e}}getNextTaskLabel(e){let n=e.map(r=>r.nextRun).filter(Boolean).sort()[0];return n?`${e.find(r=>r.nextRun===n)?.agent??"unknown"} in ${this.formatNextRun(n)}`:"none"}getInitials(e){return e.split("-").map(s=>s[0]?.toUpperCase()??"").slice(0,2).join("")}renderAgentAvatar(e,s){let n=s.avatar?.trim();if(!n){(0,b.setIcon)(e,"bot");return}/^[a-z][a-z0-9-]*$/.test(n)?(0,b.setIcon)(e,n):e.setText(n)}getSkillIcon(e){return e.includes("git")?"settings":e.includes("summarize")||e.includes("log")?"activity":e.includes("review")||e.includes("check")?"check-circle-2":e.includes("vault")||e.includes("note")?"file-text":"puzzle"}renderInlineSchedule(e,s){let n=this.parseCronComponents(s.schedule),a=e.createDiv({cls:"af-form-row"});a.createDiv({cls:"af-form-label",text:"Frequency"});let r=a.createEl("select",{cls:"af-form-select"}),o=[["every_5m","Every 5 minutes"],["every_15m","Every 15 minutes"],["every_30m","Every 30 minutes"],["every_hour","Every hour"],["every_2h","Every 2 hours"],["daily","Daily"],["weekdays","Weekdays"],["weekly","Weekly"],["monthly","Monthly"]];for(let[w,S]of o){let T=r.createEl("option",{text:S,attr:{value:w}});w===n.freq&&(T.selected=!0)}let c=e.createDiv({cls:"af-form-row af-schedule-time-row"});c.createDiv({cls:"af-form-label",text:"Time"});let l=c.createDiv({cls:"af-schedule-time-selects"}),h=l.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let w=0;w<24;w++){let S=w>=12?"PM":"AM",T=w===0?12:w>12?w-12:w,_=h.createEl("option",{text:`${T} ${S}`,attr:{value:String(w)}});w===n.hour&&(_.selected=!0)}l.createSpan({cls:"af-schedule-colon",text:":"});let d=l.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let w=0;w<60;w+=5){let S=d.createEl("option",{text:String(w).padStart(2,"0"),attr:{value:String(w)}});w===n.minute&&(S.selected=!0)}let u=e.createDiv({cls:"af-form-row af-schedule-day-row"});u.createDiv({cls:"af-form-label",text:"Day"});let p=u.createDiv({cls:"af-schedule-day-buttons"}),m=["Sun","Mon","Tue","Wed","Thu","Fri","Sat"],f=new Set(n.days);for(let w=0;w<7;w++){let S=p.createEl("button",{cls:`af-schedule-day-btn${f.has(w)?" active":""}`,text:m[w]});S.onclick=()=>{f.has(w)?f.delete(w):f.add(w),S.toggleClass("active",f.has(w)),k()}}let g=e.createDiv({cls:"af-form-row af-schedule-dom-row"});g.createDiv({cls:"af-form-label",text:"Day of month"});let y=g.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let w=1;w<=28;w++){let S=y.createEl("option",{text:String(w),attr:{value:String(w)}});w===n.dayOfMonth&&(S.selected=!0)}let v=()=>{let w=r.value,S=["daily","weekdays","weekly","monthly"].includes(w),T=w==="weekly",_=w==="monthly";c.setCssStyles({display:S?"":"none"}),u.setCssStyles({display:T?"":"none"}),g.setCssStyles({display:_?"":"none"})},k=()=>{let w=r.value,S=h.value,T=d.value,_="";switch(w){case"every_5m":_="*/5 * * * *";break;case"every_15m":_="*/15 * * * *";break;case"every_30m":_="*/30 * * * *";break;case"every_hour":_="0 * * * *";break;case"every_2h":_="0 */2 * * *";break;case"daily":_=`${T} ${S} * * *`;break;case"weekdays":_=`${T} ${S} * * 1-5`;break;case"weekly":{let D=Array.from(f).sort().join(",")||"1";_=`${T} ${S} * * ${D}`;break}case"monthly":_=`${T} ${S} ${y.value} * *`;break}s.schedule=_,s.type="recurring"};r.addEventListener("change",()=>{v(),k()}),h.addEventListener("change",k),d.addEventListener("change",k),y.addEventListener("change",k),v()}renderHeartbeatSchedule(e,s){let n=this.parseCronComponents(s.heartbeatSchedule),a=e.createDiv({cls:"af-form-row"});a.createDiv({cls:"af-form-label",text:"Frequency"});let r=a.createEl("select",{cls:"af-form-select"}),o=[["every_5m","Every 5 minutes"],["every_15m","Every 15 minutes"],["every_30m","Every 30 minutes"],["every_hour","Every hour"],["every_2h","Every 2 hours"],["every_4h","Every 4 hours"],["every_6h","Every 6 hours"],["every_12h","Every 12 hours"],["daily","Once a day"]],c="every_hour",l={"*/5 * * * *":"every_5m","*/15 * * * *":"every_15m","*/30 * * * *":"every_30m","0 * * * *":"every_hour","0 */2 * * *":"every_2h","0 */4 * * *":"every_4h","0 */6 * * *":"every_6h","0 */12 * * *":"every_12h"};l[s.heartbeatSchedule]?c=l[s.heartbeatSchedule]:(n.freq==="daily"||n.freq==="weekdays")&&(c="daily");for(let[g,y]of o){let v=r.createEl("option",{text:y,attr:{value:g}});g===c&&(v.selected=!0)}let h=e.createDiv({cls:"af-form-row af-schedule-time-row"});h.createDiv({cls:"af-form-label",text:"Time"});let d=h.createDiv({cls:"af-schedule-time-selects"}),u=d.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let g=0;g<24;g++){let y=g>=12?"PM":"AM",v=g===0?12:g>12?g-12:g,k=u.createEl("option",{text:`${v} ${y}`,attr:{value:String(g)}});g===n.hour&&(k.selected=!0)}d.createSpan({cls:"af-schedule-colon",text:":"});let p=d.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let g=0;g<60;g+=5){let y=p.createEl("option",{text:String(g).padStart(2,"0"),attr:{value:String(g)}});g===n.minute&&(y.selected=!0)}let m=()=>{h.setCssStyles({display:r.value==="daily"?"":"none"})},f=()=>{let g=r.value,y=u.value,v=p.value;switch(g){case"every_5m":s.heartbeatSchedule="*/5 * * * *";break;case"every_15m":s.heartbeatSchedule="*/15 * * * *";break;case"every_30m":s.heartbeatSchedule="*/30 * * * *";break;case"every_hour":s.heartbeatSchedule="0 * * * *";break;case"every_2h":s.heartbeatSchedule="0 */2 * * *";break;case"every_4h":s.heartbeatSchedule="0 */4 * * *";break;case"every_6h":s.heartbeatSchedule="0 */6 * * *";break;case"every_12h":s.heartbeatSchedule="0 */12 * * *";break;case"daily":s.heartbeatSchedule=`${v} ${y} * * *`;break}};r.addEventListener("change",()=>{m(),f()}),u.addEventListener("change",f),p.addEventListener("change",f),m()}parseCronComponents(e){let s={freq:"daily",hour:9,minute:0,days:[1],dayOfMonth:1};if(!e?.trim())return s;let n={"*/5 * * * *":"every_5m","*/15 * * * *":"every_15m","*/30 * * * *":"every_30m","0 * * * *":"every_hour","0 */2 * * *":"every_2h"};if(n[e])return{...s,freq:n[e]};let a=e.trim().split(/\s+/);if(a.length!==5)return s;let[r,o,c,,l]=a,h=parseInt(o??"9",10),d=parseInt(r??"0",10);if(c==="*"&&l==="*")return{...s,freq:"daily",hour:h,minute:d};if(c==="*"&&l==="1-5")return{...s,freq:"weekdays",hour:h,minute:d};if(c==="*"&&l!=="*"){let u=(l??"1").split(",").map(p=>parseInt(p,10));return{...s,freq:"weekly",hour:h,minute:d,days:u}}return l==="*"&&c!=="*"?{...s,freq:"monthly",hour:h,minute:d,dayOfMonth:parseInt(c??"1",10)}:{...s,hour:h,minute:d}}humanizeCron(e){let s={"*/5 * * * *":"Every 5 minutes","*/10 * * * *":"Every 10 minutes","*/15 * * * *":"Every 15 minutes","*/30 * * * *":"Every 30 minutes","0 * * * *":"Every hour","0 */2 * * *":"Every 2 hours"};if(s[e])return s[e];let n=e.toLowerCase().trim();if(n.startsWith("every ")||n.startsWith("daily ")||n==="daily")return e;if(n.startsWith("hourly"))return"Every hour";if(n.startsWith("weekdays")||n.startsWith("weekly")||n.startsWith("monthly"))return e;let a=e.trim().split(/\s+/);if(a.length!==5)return e;let[r,o,c,,l]=a,h=(p,m)=>{let f=parseInt(p??"0",10),g=parseInt(m??"0",10),y=f>=12?"PM":"AM",v=f===0?12:f>12?f-12:f;return g===0?`${v} ${y}`:`${v}:${String(g).padStart(2,"0")} ${y}`},d=["Sun","Mon","Tue","Wed","Thu","Fri","Sat"],u=p=>p==="1-5"?"weekdays":p==="0,6"?"weekends":p.split(",").map(f=>parseInt(f,10)).map(f=>d[f]??f).join(", ");return o==="*"&&c==="*"&&l==="*"?r==="*"?"Every minute":`Every hour at :${String(r).padStart(2,"0")}`:c==="*"&&l==="*"&&o!=="*"?`Daily at ${h(o??"0",r??"0")}`:c==="*"&&l==="1-5"&&o!=="*"?`Weekdays at ${h(o??"0",r??"0")}`:c==="*"&&l!=="*"&&o!=="*"?`${u(l??"1")} at ${h(o??"0",r??"0")}`:l==="*"&&c!=="*"&&o!=="*"?`Monthly on the ${c} at ${h(o??"0",r??"0")}`:e}getTagClass(e){return e==="monitoring"?"monitoring":e==="devops"?"devops":e==="sample"?"sample":"default"}renderCreateAgentPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=s.createDiv({cls:"af-detail-header"}),a=n.createDiv({cls:"af-detail-header-left"}),r=a.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(r,"plus");let o=a.createDiv();o.createDiv({cls:"af-detail-header-name",text:"Create New Agent"}),o.createDiv({cls:"af-detail-header-desc",text:"Configure a new agent for your fleet"});let c=n.createDiv({cls:"af-detail-header-actions"}),l={name:"",description:"",avatar:"",tags:"",systemPrompt:"",model:"default",adapter:"claude-code",cwd:"",timeout:300,permissionMode:"bypassPermissions",effort:"",selectedSkills:new Set,selectedMcpServers:new Set,skillsBody:"",contextBody:"",approvalRequired:"",memory:!0,enabled:!0,allowedCommands:"",blockedCommands:"",heartbeatEnabled:!1,heartbeatSchedule:"0 */6 * * *",heartbeatBody:"",heartbeatNotify:!0,heartbeatChannel:"",heartbeatChannelTarget:"",autoCompactThreshold:85,wikiReferences:[]},h={none:{label:"None",prompt:""},coding:{label:"Coding Agent",prompt:`You are a coding agent. Review code, write tests, fix bugs, and implement features.
11962
+ `)})}let v=o.createDiv({cls:"af-slideover-actions"});if(e.filePath){let w=v.createEl("button",{cls:"af-btn-sm"});R(w,"external-link","af-btn-icon"),w.appendText(" Open Run Note"),w.onclick=()=>void this.plugin.openPath(e.filePath)}let k=v.createEl("button",{cls:"af-btn-sm primary"});R(k,"refresh-cw","af-btn-icon"),k.appendText(" Re-run Task"),k.onclick=()=>void this.plugin.runAgentPrompt(e.agent),s.onclick=w=>{w.target===s&&s.remove()}}renderDetailRow(e,s,n){let a=e.createDiv({cls:"af-detail-row"});a.createSpan({cls:"af-detail-label",text:s}),a.createSpan({cls:"af-detail-value af-mono",text:n})}renderEmptyState(e,s,n,a){let i=e.createDiv({cls:"af-empty-state"}),o=i.createDiv({cls:"af-empty-icon"});(0,b.setIcon)(o,s),i.createDiv({cls:"af-empty-label",text:n}),a&&i.createDiv({cls:"af-empty-sublabel",text:a})}healthToClass(e){switch(e){case"running":return"running";case"error":return"error";case"pending":return"pending";case"disabled":return"disabled";default:return"idle"}}statusToTimelineClass(e){switch(e){case"success":return"success";case"failure":case"timeout":return"error";case"cancelled":return"warning";case"pending_approval":return"pending";default:return"running"}}statusToIconName(e){switch(e){case"success":return"check-circle-2";case"failure":return"x-circle";case"timeout":return"clock";case"pending_approval":return"shield-check";case"cancelled":return"square";default:return"loader-2"}}statusToBadgeClass(e){switch(e){case"success":return"success";case"failure":return"failure";case"timeout":return"timeout";case"pending_approval":return"pending";case"cancelled":return"cancelled";default:return"running"}}statusToBadgeText(e){switch(e){case"success":return"Success";case"failure":return"Failed";case"timeout":return"Timeout";case"pending_approval":return"Pending";case"cancelled":return"Cancelled";case"interrupted":return"Interrupted";default:return e}}formatDuration(e){if(e<60)return`${e}s`;let s=Math.floor(e/60),n=e%60;return n>0?`${s}m ${n}s`:`${s}m`}formatStarted(e){try{let s=new Date(e),n=new Date;if(s.toDateString()===n.toDateString())return s.toLocaleTimeString([],{hour:"2-digit",minute:"2-digit"});let a=new Date(n);return a.setDate(a.getDate()-1),s.toDateString()===a.toDateString()?`Yesterday ${s.toLocaleTimeString([],{hour:"2-digit",minute:"2-digit"})}`:s.toLocaleDateString([],{month:"short",day:"numeric"})+` ${s.toLocaleTimeString([],{hour:"2-digit",minute:"2-digit"})}`}catch{return e}}formatNextRun(e){try{let s=new Date(e),n=new Date,a=s.getTime()-n.getTime();if(a<0)return"overdue";let i=Math.round(a/6e4);if(i<60)return`${i}m`;let o=Math.round(i/60);return o<24?`${o}h`:s.toLocaleDateString([],{month:"short",day:"numeric"})}catch{return e}}getNextTaskLabel(e){let n=e.map(i=>i.nextRun).filter(Boolean).sort()[0];return n?`${e.find(i=>i.nextRun===n)?.agent??"unknown"} in ${this.formatNextRun(n)}`:"none"}getInitials(e){return e.split("-").map(s=>s[0]?.toUpperCase()??"").slice(0,2).join("")}renderAgentAvatar(e,s){let n=s.avatar?.trim();if(!n){(0,b.setIcon)(e,"bot");return}/^[a-z][a-z0-9-]*$/.test(n)?(0,b.setIcon)(e,n):e.setText(n)}getSkillIcon(e){return e.includes("git")?"settings":e.includes("summarize")||e.includes("log")?"activity":e.includes("review")||e.includes("check")?"check-circle-2":e.includes("vault")||e.includes("note")?"file-text":"puzzle"}renderInlineSchedule(e,s){let n=this.parseCronComponents(s.schedule),a=e.createDiv({cls:"af-form-row"});a.createDiv({cls:"af-form-label",text:"Frequency"});let i=a.createEl("select",{cls:"af-form-select"}),o=[["every_5m","Every 5 minutes"],["every_15m","Every 15 minutes"],["every_30m","Every 30 minutes"],["every_hour","Every hour"],["every_2h","Every 2 hours"],["daily","Daily"],["weekdays","Weekdays"],["weekly","Weekly"],["monthly","Monthly"]];for(let[w,S]of o){let T=i.createEl("option",{text:S,attr:{value:w}});w===n.freq&&(T.selected=!0)}let c=e.createDiv({cls:"af-form-row af-schedule-time-row"});c.createDiv({cls:"af-form-label",text:"Time"});let l=c.createDiv({cls:"af-schedule-time-selects"}),h=l.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let w=0;w<24;w++){let S=w>=12?"PM":"AM",T=w===0?12:w>12?w-12:w,_=h.createEl("option",{text:`${T} ${S}`,attr:{value:String(w)}});w===n.hour&&(_.selected=!0)}l.createSpan({cls:"af-schedule-colon",text:":"});let d=l.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let w=0;w<60;w+=5){let S=d.createEl("option",{text:String(w).padStart(2,"0"),attr:{value:String(w)}});w===n.minute&&(S.selected=!0)}let u=e.createDiv({cls:"af-form-row af-schedule-day-row"});u.createDiv({cls:"af-form-label",text:"Day"});let p=u.createDiv({cls:"af-schedule-day-buttons"}),m=["Sun","Mon","Tue","Wed","Thu","Fri","Sat"],f=new Set(n.days);for(let w=0;w<7;w++){let S=p.createEl("button",{cls:`af-schedule-day-btn${f.has(w)?" active":""}`,text:m[w]});S.onclick=()=>{f.has(w)?f.delete(w):f.add(w),S.toggleClass("active",f.has(w)),k()}}let g=e.createDiv({cls:"af-form-row af-schedule-dom-row"});g.createDiv({cls:"af-form-label",text:"Day of month"});let y=g.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let w=1;w<=28;w++){let S=y.createEl("option",{text:String(w),attr:{value:String(w)}});w===n.dayOfMonth&&(S.selected=!0)}let v=()=>{let w=i.value,S=["daily","weekdays","weekly","monthly"].includes(w),T=w==="weekly",_=w==="monthly";c.setCssStyles({display:S?"":"none"}),u.setCssStyles({display:T?"":"none"}),g.setCssStyles({display:_?"":"none"})},k=()=>{let w=i.value,S=h.value,T=d.value,_="";switch(w){case"every_5m":_="*/5 * * * *";break;case"every_15m":_="*/15 * * * *";break;case"every_30m":_="*/30 * * * *";break;case"every_hour":_="0 * * * *";break;case"every_2h":_="0 */2 * * *";break;case"daily":_=`${T} ${S} * * *`;break;case"weekdays":_=`${T} ${S} * * 1-5`;break;case"weekly":{let D=Array.from(f).sort().join(",")||"1";_=`${T} ${S} * * ${D}`;break}case"monthly":_=`${T} ${S} ${y.value} * *`;break}s.schedule=_,s.type="recurring"};i.addEventListener("change",()=>{v(),k()}),h.addEventListener("change",k),d.addEventListener("change",k),y.addEventListener("change",k),v()}renderHeartbeatSchedule(e,s){let n=this.parseCronComponents(s.heartbeatSchedule),a=e.createDiv({cls:"af-form-row"});a.createDiv({cls:"af-form-label",text:"Frequency"});let i=a.createEl("select",{cls:"af-form-select"}),o=[["every_5m","Every 5 minutes"],["every_15m","Every 15 minutes"],["every_30m","Every 30 minutes"],["every_hour","Every hour"],["every_2h","Every 2 hours"],["every_4h","Every 4 hours"],["every_6h","Every 6 hours"],["every_12h","Every 12 hours"],["daily","Once a day"]],c="every_hour",l={"*/5 * * * *":"every_5m","*/15 * * * *":"every_15m","*/30 * * * *":"every_30m","0 * * * *":"every_hour","0 */2 * * *":"every_2h","0 */4 * * *":"every_4h","0 */6 * * *":"every_6h","0 */12 * * *":"every_12h"};l[s.heartbeatSchedule]?c=l[s.heartbeatSchedule]:(n.freq==="daily"||n.freq==="weekdays")&&(c="daily");for(let[g,y]of o){let v=i.createEl("option",{text:y,attr:{value:g}});g===c&&(v.selected=!0)}let h=e.createDiv({cls:"af-form-row af-schedule-time-row"});h.createDiv({cls:"af-form-label",text:"Time"});let d=h.createDiv({cls:"af-schedule-time-selects"}),u=d.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let g=0;g<24;g++){let y=g>=12?"PM":"AM",v=g===0?12:g>12?g-12:g,k=u.createEl("option",{text:`${v} ${y}`,attr:{value:String(g)}});g===n.hour&&(k.selected=!0)}d.createSpan({cls:"af-schedule-colon",text:":"});let p=d.createEl("select",{cls:"af-form-select af-form-select-sm"});for(let g=0;g<60;g+=5){let y=p.createEl("option",{text:String(g).padStart(2,"0"),attr:{value:String(g)}});g===n.minute&&(y.selected=!0)}let m=()=>{h.setCssStyles({display:i.value==="daily"?"":"none"})},f=()=>{let g=i.value,y=u.value,v=p.value;switch(g){case"every_5m":s.heartbeatSchedule="*/5 * * * *";break;case"every_15m":s.heartbeatSchedule="*/15 * * * *";break;case"every_30m":s.heartbeatSchedule="*/30 * * * *";break;case"every_hour":s.heartbeatSchedule="0 * * * *";break;case"every_2h":s.heartbeatSchedule="0 */2 * * *";break;case"every_4h":s.heartbeatSchedule="0 */4 * * *";break;case"every_6h":s.heartbeatSchedule="0 */6 * * *";break;case"every_12h":s.heartbeatSchedule="0 */12 * * *";break;case"daily":s.heartbeatSchedule=`${v} ${y} * * *`;break}};i.addEventListener("change",()=>{m(),f()}),u.addEventListener("change",f),p.addEventListener("change",f),m()}parseCronComponents(e){let s={freq:"daily",hour:9,minute:0,days:[1],dayOfMonth:1};if(!e?.trim())return s;let n={"*/5 * * * *":"every_5m","*/15 * * * *":"every_15m","*/30 * * * *":"every_30m","0 * * * *":"every_hour","0 */2 * * *":"every_2h"};if(n[e])return{...s,freq:n[e]};let a=e.trim().split(/\s+/);if(a.length!==5)return s;let[i,o,c,,l]=a,h=parseInt(o??"9",10),d=parseInt(i??"0",10);if(c==="*"&&l==="*")return{...s,freq:"daily",hour:h,minute:d};if(c==="*"&&l==="1-5")return{...s,freq:"weekdays",hour:h,minute:d};if(c==="*"&&l!=="*"){let u=(l??"1").split(",").map(p=>parseInt(p,10));return{...s,freq:"weekly",hour:h,minute:d,days:u}}return l==="*"&&c!=="*"?{...s,freq:"monthly",hour:h,minute:d,dayOfMonth:parseInt(c??"1",10)}:{...s,hour:h,minute:d}}humanizeCron(e){let s={"*/5 * * * *":"Every 5 minutes","*/10 * * * *":"Every 10 minutes","*/15 * * * *":"Every 15 minutes","*/30 * * * *":"Every 30 minutes","0 * * * *":"Every hour","0 */2 * * *":"Every 2 hours"};if(s[e])return s[e];let n=e.toLowerCase().trim();if(n.startsWith("every ")||n.startsWith("daily ")||n==="daily")return e;if(n.startsWith("hourly"))return"Every hour";if(n.startsWith("weekdays")||n.startsWith("weekly")||n.startsWith("monthly"))return e;let a=e.trim().split(/\s+/);if(a.length!==5)return e;let[i,o,c,,l]=a,h=(p,m)=>{let f=parseInt(p??"0",10),g=parseInt(m??"0",10),y=f>=12?"PM":"AM",v=f===0?12:f>12?f-12:f;return g===0?`${v} ${y}`:`${v}:${String(g).padStart(2,"0")} ${y}`},d=["Sun","Mon","Tue","Wed","Thu","Fri","Sat"],u=p=>p==="1-5"?"weekdays":p==="0,6"?"weekends":p.split(",").map(f=>parseInt(f,10)).map(f=>d[f]??f).join(", ");return o==="*"&&c==="*"&&l==="*"?i==="*"?"Every minute":`Every hour at :${String(i).padStart(2,"0")}`:c==="*"&&l==="*"&&o!=="*"?`Daily at ${h(o??"0",i??"0")}`:c==="*"&&l==="1-5"&&o!=="*"?`Weekdays at ${h(o??"0",i??"0")}`:c==="*"&&l!=="*"&&o!=="*"?`${u(l??"1")} at ${h(o??"0",i??"0")}`:l==="*"&&c!=="*"&&o!=="*"?`Monthly on the ${c} at ${h(o??"0",i??"0")}`:e}getTagClass(e){return e==="monitoring"?"monitoring":e==="devops"?"devops":e==="sample"?"sample":"default"}renderCreateAgentPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=s.createDiv({cls:"af-detail-header"}),a=n.createDiv({cls:"af-detail-header-left"}),i=a.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(i,"plus");let o=a.createDiv();o.createDiv({cls:"af-detail-header-name",text:"Create New Agent"}),o.createDiv({cls:"af-detail-header-desc",text:"Configure a new agent for your fleet"});let c=n.createDiv({cls:"af-detail-header-actions"}),l={name:"",description:"",avatar:"",tags:"",systemPrompt:"",model:"default",adapter:"claude-code",cwd:"",timeout:300,permissionMode:"bypassPermissions",effort:"",selectedSkills:new Set,selectedMcpServers:new Set,skillsBody:"",contextBody:"",approvalRequired:"",memory:!0,enabled:!0,allowedCommands:"",blockedCommands:"",heartbeatEnabled:!1,heartbeatSchedule:"0 */6 * * *",heartbeatBody:"",heartbeatNotify:!0,heartbeatChannel:"",heartbeatChannelTarget:"",autoCompactThreshold:85,wikiReferences:[]},h={none:{label:"None",prompt:""},coding:{label:"Coding Agent",prompt:`You are a coding agent. Review code, write tests, fix bugs, and implement features.
12052
11963
  Follow existing code conventions. Write clean, well-tested code.
12053
11964
  If something is unclear, ask for clarification instead of guessing.`},monitor:{label:"Monitor",prompt:`You are a monitoring agent. Check system status, alert on failures, and report on health metrics.
12054
11965
  Be concise and factual. Highlight anomalies clearly.
@@ -12056,13 +11967,13 @@ Include timestamps and relevant context in all reports.`},briefing:{label:"Brief
12056
11967
  Prioritize recent and important changes. Keep summaries concise.
12057
11968
  End with explicit next actions if they exist.`},reviewer:{label:"Code Reviewer",prompt:`You are a code review agent. Analyze pull requests, suggest improvements, and flag potential issues.
12058
11969
  Focus on correctness, security, and maintainability.
12059
- Be specific \u2014 reference file names and line numbers.`}},d=s.createDiv({cls:"af-create-form"}),u=d.createDiv({cls:"af-create-section"}),p=u.createDiv({cls:"af-create-section-header"}),m=p.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(m,"user"),p.createSpan({text:"Identity"}),this.createFormField(u,"Name","deploy-watcher","Unique identifier (will be slugified)",B=>{l.name=B}),this.createFormField(u,"Description","Monitors deployments and reports status","",B=>{l.description=B});let f=u.createDiv({cls:"af-form-row"});f.createDiv({cls:"af-form-label",text:"Avatar"});let g=f.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"text",placeholder:"\u{1F6E1}\uFE0F"}});g.addEventListener("input",()=>{l.avatar=g.value}),this.createFormField(u,"Tags","devops, monitoring","Comma-separated",B=>{l.tags=B});let y=u.createDiv({cls:"af-form-row af-form-row-toggle"});y.createDiv({cls:"af-form-label",text:"Enabled"});let v=y.createDiv({cls:"af-agent-card-toggle on"});v.onclick=()=>{let B=v.hasClass("on");v.toggleClass("on",!B),l.enabled=!B};let k=d.createDiv({cls:"af-create-section"}),w=k.createDiv({cls:"af-create-section-header"}),S=w.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(S,"message-square"),w.createSpan({text:"System Prompt"});let T=k.createDiv({cls:"af-form-row"});T.createDiv({cls:"af-form-label",text:"Template"});let _=T.createEl("select",{cls:"af-form-select"});for(let[B,{label:V}]of Object.entries(h))_.createEl("option",{text:V,attr:{value:B}});let D=k.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"You are a deployment monitoring agent...",rows:"10"}});D.addEventListener("input",()=>{l.systemPrompt=D.value}),_.addEventListener("change",()=>{let B=h[_.value];B&&_.value!=="none"&&(l.systemPrompt=B.prompt,D.value=B.prompt)});let O=d.createDiv({cls:"af-create-section"}),C=O.createDiv({cls:"af-create-section-header"}),E=C.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(E,"settings"),C.createSpan({text:"Runtime Config"});let P=O.createDiv({cls:"af-create-config-grid"}),N=P.createDiv({cls:"af-form-row"});N.createDiv({cls:"af-form-label",text:"Adapter"});let M=N.createEl("select",{cls:"af-form-select"});for(let[B,V,Ee]of Jo){let we=M.createEl("option",{text:V,attr:{value:B,...Ee?{disabled:"true"}:{}}});B==="claude-code"&&(we.selected=!0)}let U=P.createDiv({cls:"af-form-row"}),$=U.createDiv({cls:"af-form-label",text:"Model"});this.addTooltip($,`Aliases (opus/sonnet/haiku/opusplan) work on any backend. Choose Custom\u2026 for a pinned ID or Bedrock/Vertex/Foundry. Blank = use Settings default (${this.plugin.settings.defaultModel||"CLI default"}).`);let Z=U.createDiv({cls:"af-form-field-wrap"}),de=()=>{Pt(Z,{value:l.model,adapter:l.adapter,onChange:B=>{l.model=B}})};de();let me=()=>{};M.addEventListener("change",()=>{l.adapter=M.value,(Ls(l.adapter)?Qs:Zs).some(V=>V.value===l.model.trim())&&(l.model=""),de(),l.permissionMode=Hn(l.permissionMode,l.adapter),me()});let ge=P.createDiv({cls:"af-form-row"});ge.createDiv({cls:"af-form-label",text:"Working Dir"});let X=ge.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Leave empty for vault root"}});X.addEventListener("input",()=>{l.cwd=X.value});let W=P.createDiv({cls:"af-form-row"});W.createDiv({cls:"af-form-label",text:"Timeout (sec)"});let K=W.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",value:"300"}});K.addEventListener("input",()=>{let B=parseInt(K.value,10);!isNaN(B)&&B>0&&(l.timeout=B)});let ne=P.createDiv({cls:"af-form-row"});ne.createDiv({cls:"af-form-label",text:"Permission Mode"});let G=ne.createEl("select",{cls:"af-form-select"}),ee=P.createDiv({cls:"af-form-hint",text:""});me=()=>{l.permissionMode=Hn(l.permissionMode,l.adapter);let B=Wn(l.adapter);G.empty();for(let[V,Ee]of B){let we=G.createEl("option",{text:Ee,attr:{value:V}});V===l.permissionMode&&(we.selected=!0)}ee.textContent=B.find(([V])=>V===G.value)?.[2]??""},me(),G.addEventListener("change",()=>{l.permissionMode=G.value,ee.textContent=Wn(l.adapter).find(([B])=>B===G.value)?.[2]??""});let Q=P.createDiv({cls:"af-form-row"});Q.createDiv({cls:"af-form-label",text:"Effort Level"});let ae=Q.createEl("select",{cls:"af-form-select"});for(let[B,V]of[["","Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]])ae.createEl("option",{text:V,attr:{value:B}});ae.addEventListener("change",()=>{l.effort=ae.value}),P.createDiv({cls:"af-form-hint",text:"Controls reasoning depth \u2014 low is fast, max is most thorough"});let ve=P.createDiv({cls:"af-form-row"}),Pe=ve.createDiv({cls:"af-form-label",text:"Auto-compact at"});this.addTooltip(Pe,"Percent of context window at which the chat auto-invokes /compact before the next message. 85% is a good default. Set 0 to disable.");let Ie=ve.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",min:"0",max:"100",value:String(l.autoCompactThreshold)}});Ie.addEventListener("input",()=>{let B=parseInt(Ie.value,10);!isNaN(B)&&B>=0&&B<=100&&(l.autoCompactThreshold=B)}),P.createDiv({cls:"af-form-hint",text:"0 disables auto-compact"});{let B=this.plugin.runtime.getSnapshot().agents.filter(V=>V.wikiKeeper!==void 0);if(B.length>0){let V=P.createDiv({cls:"af-form-row af-form-row-toggle"}),Ee=V.createDiv({cls:"af-form-label",text:"Wiki access"});this.addTooltip(Ee,"Lets this agent read + cite from the selected Wiki Keeper scopes (requires the wiki-query skill).");let we=V.createDiv({cls:"af-form-field-wrap"});for(let Ae of B){let nt=we.createEl("label",{cls:"af-form-checkbox-row"}),je=nt.createEl("input",{attr:{type:"checkbox"}});nt.createSpan({text:` ${Ae.name}`,cls:"af-form-checkbox-label"}),je.addEventListener("change",()=>{je.checked?l.wikiReferences.includes(Ae.name)||l.wikiReferences.push(Ae.name):l.wikiReferences=l.wikiReferences.filter(Tt=>Tt!==Ae.name)})}}}{let B=d.createDiv({cls:"af-create-section"}),V=B.createDiv({cls:"af-create-section-header"}),Ee=V.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ee,"heart-pulse");let we=V.createSpan({text:"Heartbeat"});this.addTooltip(we,"Autonomous periodic run \u2014 what the agent does when no one is asking");let Ae=B.createDiv({cls:"af-form-row af-form-row-toggle"});Ae.createDiv({cls:"af-form-label",text:"Enabled"});let nt=Ae.createDiv({cls:"af-agent-card-toggle"}),je=B.createDiv();je.setCssStyles({display:"none"}),nt.onclick=()=>{let ze=nt.hasClass("on");nt.toggleClass("on",!ze),l.heartbeatEnabled=!ze,je.setCssStyles({display:ze?"none":""})},this.renderHeartbeatSchedule(je,l);let Tt=je.createDiv({cls:"af-form-row af-form-row-toggle"}),Bt=Tt.createDiv({cls:"af-form-label"});Bt.setText("Notify"),this.addTooltip(Bt,"Show an Obsidian notice when the heartbeat completes");let bt=Tt.createDiv({cls:"af-agent-card-toggle on"});bt.onclick=()=>{let ze=bt.hasClass("on");bt.toggleClass("on",!ze),l.heartbeatNotify=!ze};let ds=this.plugin.runtime.getSnapshot(),Ns=je.createDiv({cls:"af-form-row"}),Ut=Ns.createDiv({cls:"af-form-label"});Ut.setText("Post to channel"),this.addTooltip(Ut,"Heartbeat results are posted to this Slack channel when the run completes");let L=Ns.createEl("select",{cls:"af-form-select"});L.createEl("option",{text:"(none)",attr:{value:""}});for(let ze of ds.channels)L.createEl("option",{text:ze.name,attr:{value:ze.name}});L.addEventListener("change",()=>{l.heartbeatChannel=L.value,qe()});let Y=je.createDiv({cls:"af-form-row"}),Te=Y.createDiv({cls:"af-form-label"});Te.setText("Target ID"),this.addTooltip(Te,"Specific channel id to post to (Discord/Slack channel id, Telegram chat id). Empty = DM the channel\u2019s first allowed user.");let Le=Y.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Channel/chat id \u2014 empty = DM"}});Le.value=l.heartbeatChannelTarget,Le.addEventListener("input",()=>{l.heartbeatChannelTarget=Le.value.trim()});let qe=()=>{Y.setCssStyles({display:l.heartbeatChannel?"":"none"})};qe();let at=je.createDiv({cls:"af-form-label"});at.setCssStyles({width:"auto"}),at.setCssStyles({marginTop:"12px"}),at.setText("Instruction"),this.addTooltip(at,'What the agent does on each heartbeat. Also used by the "Run Now" button.');let We=je.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Check status, scan for issues, report findings...",rows:"8"}});We.addEventListener("input",()=>{l.heartbeatBody=We.value})}let Me=d.createDiv({cls:"af-create-section"}),Fe=Me.createDiv({cls:"af-create-section-header"}),ke=Fe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(ke,"puzzle"),Fe.createSpan({text:"Skills"});let Oe=this.plugin.runtime.getSnapshot();if(Oe.skills.length>0){Me.createDiv({cls:"af-form-sublabel",text:"Shared Skills"});let B=Me.createDiv({cls:"af-create-skills-grid"});for(let V of Oe.skills){let Ee=B.createDiv({cls:"af-create-skill-item"}),we=Ee.createEl("input",{cls:"af-form-toggle",attr:{type:"checkbox"}});we.addEventListener("change",()=>{we.checked?l.selectedSkills.add(V.name):l.selectedSkills.delete(V.name)});let Ae=Ee.createDiv({cls:"af-create-skill-label"});Ae.createSpan({cls:"af-create-skill-name",text:V.name}),V.description&&Ae.createSpan({cls:"af-create-skill-desc",text:` \u2014 ${V.description}`})}}let Ue=Me.createDiv({cls:"af-form-sublabel"});Ue.setText("Agent-specific skills"),this.addTooltip(Ue,"Custom skills/instructions only for this agent, not shared with others");let Ne=Me.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Custom skills/instructions for this agent...",rows:"4"}});Ne.addEventListener("input",()=>{l.skillsBody=Ne.value});{let B=d.createDiv({cls:"af-create-section"}),V=B.createDiv({cls:"af-create-section-header"}),Ee=V.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ee,"plug");let we=V.createSpan({text:"MCP Servers"});this.addTooltip(we,"Grant agent access to MCP servers"),this.renderAgentMcpPicker(B,l.selectedMcpServers)}let q=d.createDiv({cls:"af-create-section"}),ie=q.createDiv({cls:"af-create-section-header"}),j=ie.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(j,"file-text");let Ce=ie.createSpan({text:"Context"});this.addTooltip(Ce,"Project-specific context included in every run");let xe=q.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Background info, repo structure, conventions...",rows:"4"}});xe.addEventListener("input",()=>{l.contextBody=xe.value});let he=d.createDiv({cls:"af-create-section"}),$e=he.createDiv({cls:"af-create-section-header"}),re=$e.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(re,"shield-check"),$e.createSpan({text:"Permissions"}),this.createFormField(he,"Approval required","git_push, file_delete","Comma-separated tool names",B=>{l.approvalRequired=B});let I=he.createDiv({cls:"af-form-row"});I.createDiv({cls:"af-form-label",text:"Allowed Commands"});let se=I.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:`Bash(curl *)
11970
+ Be specific \u2014 reference file names and line numbers.`}},d=s.createDiv({cls:"af-create-form"}),u=d.createDiv({cls:"af-create-section"}),p=u.createDiv({cls:"af-create-section-header"}),m=p.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(m,"user"),p.createSpan({text:"Identity"}),this.createFormField(u,"Name","deploy-watcher","Unique identifier (will be slugified)",B=>{l.name=B}),this.createFormField(u,"Description","Monitors deployments and reports status","",B=>{l.description=B});let f=u.createDiv({cls:"af-form-row"});f.createDiv({cls:"af-form-label",text:"Avatar"});let g=f.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"text",placeholder:"\u{1F6E1}\uFE0F"}});g.addEventListener("input",()=>{l.avatar=g.value}),this.createFormField(u,"Tags","devops, monitoring","Comma-separated",B=>{l.tags=B});let y=u.createDiv({cls:"af-form-row af-form-row-toggle"});y.createDiv({cls:"af-form-label",text:"Enabled"});let v=y.createDiv({cls:"af-agent-card-toggle on"});v.onclick=()=>{let B=v.hasClass("on");v.toggleClass("on",!B),l.enabled=!B};let k=d.createDiv({cls:"af-create-section"}),w=k.createDiv({cls:"af-create-section-header"}),S=w.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(S,"message-square"),w.createSpan({text:"System Prompt"});let T=k.createDiv({cls:"af-form-row"});T.createDiv({cls:"af-form-label",text:"Template"});let _=T.createEl("select",{cls:"af-form-select"});for(let[B,{label:V}]of Object.entries(h))_.createEl("option",{text:V,attr:{value:B}});let D=k.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"You are a deployment monitoring agent...",rows:"10"}});D.addEventListener("input",()=>{l.systemPrompt=D.value}),_.addEventListener("change",()=>{let B=h[_.value];B&&_.value!=="none"&&(l.systemPrompt=B.prompt,D.value=B.prompt)});let O=d.createDiv({cls:"af-create-section"}),C=O.createDiv({cls:"af-create-section-header"}),E=C.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(E,"settings"),C.createSpan({text:"Runtime Config"});let P=O.createDiv({cls:"af-create-config-grid"}),N=P.createDiv({cls:"af-form-row"});N.createDiv({cls:"af-form-label",text:"Adapter"});let M=N.createEl("select",{cls:"af-form-select"});for(let[B,V,Ee]of Xo){let we=M.createEl("option",{text:V,attr:{value:B,...Ee?{disabled:"true"}:{}}});B==="claude-code"&&(we.selected=!0)}let U=P.createDiv({cls:"af-form-row"}),$=U.createDiv({cls:"af-form-label",text:"Model"});this.addTooltip($,`Aliases (opus/sonnet/haiku/opusplan) work on any backend. Choose Custom\u2026 for a pinned ID or Bedrock/Vertex/Foundry. Blank = use Settings default (${this.plugin.settings.defaultModel||"CLI default"}).`);let Z=U.createDiv({cls:"af-form-field-wrap"}),de=()=>{Rt(Z,{value:l.model,adapter:l.adapter,onChange:B=>{l.model=B}})};de();let me=()=>{};M.addEventListener("change",()=>{l.adapter=M.value,(Ls(l.adapter)?Qs:Zs).some(V=>V.value===l.model.trim())&&(l.model=""),de(),l.permissionMode=Hn(l.permissionMode,l.adapter),me()});let ge=P.createDiv({cls:"af-form-row"});ge.createDiv({cls:"af-form-label",text:"Working Dir"});let X=ge.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Leave empty for vault root"}});X.addEventListener("input",()=>{l.cwd=X.value});let W=P.createDiv({cls:"af-form-row"});W.createDiv({cls:"af-form-label",text:"Timeout (sec)"});let K=W.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",value:"300"}});K.addEventListener("input",()=>{let B=parseInt(K.value,10);!isNaN(B)&&B>0&&(l.timeout=B)});let ne=P.createDiv({cls:"af-form-row"});ne.createDiv({cls:"af-form-label",text:"Permission Mode"});let G=ne.createEl("select",{cls:"af-form-select"}),ee=P.createDiv({cls:"af-form-hint",text:""});me=()=>{l.permissionMode=Hn(l.permissionMode,l.adapter);let B=Wn(l.adapter);G.empty();for(let[V,Ee]of B){let we=G.createEl("option",{text:Ee,attr:{value:V}});V===l.permissionMode&&(we.selected=!0)}ee.textContent=B.find(([V])=>V===G.value)?.[2]??""},me(),G.addEventListener("change",()=>{l.permissionMode=G.value,ee.textContent=Wn(l.adapter).find(([B])=>B===G.value)?.[2]??""});let Q=P.createDiv({cls:"af-form-row"});Q.createDiv({cls:"af-form-label",text:"Effort Level"});let ae=Q.createEl("select",{cls:"af-form-select"});for(let[B,V]of[["","Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]])ae.createEl("option",{text:V,attr:{value:B}});ae.addEventListener("change",()=>{l.effort=ae.value}),P.createDiv({cls:"af-form-hint",text:"Controls reasoning depth \u2014 low is fast, max is most thorough"});let ve=P.createDiv({cls:"af-form-row"}),Pe=ve.createDiv({cls:"af-form-label",text:"Auto-compact at"});this.addTooltip(Pe,"Percent of context window at which the chat auto-invokes /compact before the next message. 85% is a good default. Set 0 to disable.");let Ie=ve.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",min:"0",max:"100",value:String(l.autoCompactThreshold)}});Ie.addEventListener("input",()=>{let B=parseInt(Ie.value,10);!isNaN(B)&&B>=0&&B<=100&&(l.autoCompactThreshold=B)}),P.createDiv({cls:"af-form-hint",text:"0 disables auto-compact"});{let B=this.plugin.runtime.getSnapshot().agents.filter(V=>V.wikiKeeper!==void 0);if(B.length>0){let V=P.createDiv({cls:"af-form-row af-form-row-toggle"}),Ee=V.createDiv({cls:"af-form-label",text:"Wiki access"});this.addTooltip(Ee,"Lets this agent read + cite from the selected Wiki Keeper scopes (requires the wiki-query skill).");let we=V.createDiv({cls:"af-form-field-wrap"});for(let Ae of B){let nt=we.createEl("label",{cls:"af-form-checkbox-row"}),je=nt.createEl("input",{attr:{type:"checkbox"}});nt.createSpan({text:` ${Ae.name}`,cls:"af-form-checkbox-label"}),je.addEventListener("change",()=>{je.checked?l.wikiReferences.includes(Ae.name)||l.wikiReferences.push(Ae.name):l.wikiReferences=l.wikiReferences.filter(Tt=>Tt!==Ae.name)})}}}{let B=d.createDiv({cls:"af-create-section"}),V=B.createDiv({cls:"af-create-section-header"}),Ee=V.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ee,"heart-pulse");let we=V.createSpan({text:"Heartbeat"});this.addTooltip(we,"Autonomous periodic run \u2014 what the agent does when no one is asking");let Ae=B.createDiv({cls:"af-form-row af-form-row-toggle"});Ae.createDiv({cls:"af-form-label",text:"Enabled"});let nt=Ae.createDiv({cls:"af-agent-card-toggle"}),je=B.createDiv();je.setCssStyles({display:"none"}),nt.onclick=()=>{let ze=nt.hasClass("on");nt.toggleClass("on",!ze),l.heartbeatEnabled=!ze,je.setCssStyles({display:ze?"none":""})},this.renderHeartbeatSchedule(je,l);let Tt=je.createDiv({cls:"af-form-row af-form-row-toggle"}),Ut=Tt.createDiv({cls:"af-form-label"});Ut.setText("Notify"),this.addTooltip(Ut,"Show an Obsidian notice when the heartbeat completes");let bt=Tt.createDiv({cls:"af-agent-card-toggle on"});bt.onclick=()=>{let ze=bt.hasClass("on");bt.toggleClass("on",!ze),l.heartbeatNotify=!ze};let hs=this.plugin.runtime.getSnapshot(),Ns=je.createDiv({cls:"af-form-row"}),$t=Ns.createDiv({cls:"af-form-label"});$t.setText("Post to channel"),this.addTooltip($t,"Heartbeat results are posted to this Slack channel when the run completes");let L=Ns.createEl("select",{cls:"af-form-select"});L.createEl("option",{text:"(none)",attr:{value:""}});for(let ze of hs.channels)L.createEl("option",{text:ze.name,attr:{value:ze.name}});L.addEventListener("change",()=>{l.heartbeatChannel=L.value,qe()});let Y=je.createDiv({cls:"af-form-row"}),Te=Y.createDiv({cls:"af-form-label"});Te.setText("Target ID"),this.addTooltip(Te,"Specific channel id to post to (Discord/Slack channel id, Telegram chat id). Empty = DM the channel\u2019s first allowed user.");let Le=Y.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Channel/chat id \u2014 empty = DM"}});Le.value=l.heartbeatChannelTarget,Le.addEventListener("input",()=>{l.heartbeatChannelTarget=Le.value.trim()});let qe=()=>{Y.setCssStyles({display:l.heartbeatChannel?"":"none"})};qe();let at=je.createDiv({cls:"af-form-label"});at.setCssStyles({width:"auto"}),at.setCssStyles({marginTop:"12px"}),at.setText("Instruction"),this.addTooltip(at,'What the agent does on each heartbeat. Also used by the "Run Now" button.');let We=je.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Check status, scan for issues, report findings...",rows:"8"}});We.addEventListener("input",()=>{l.heartbeatBody=We.value})}let Me=d.createDiv({cls:"af-create-section"}),Fe=Me.createDiv({cls:"af-create-section-header"}),ke=Fe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(ke,"puzzle"),Fe.createSpan({text:"Skills"});let Oe=this.plugin.runtime.getSnapshot();if(Oe.skills.length>0){Me.createDiv({cls:"af-form-sublabel",text:"Shared Skills"});let B=Me.createDiv({cls:"af-create-skills-grid"});for(let V of Oe.skills){let Ee=B.createDiv({cls:"af-create-skill-item"}),we=Ee.createEl("input",{cls:"af-form-toggle",attr:{type:"checkbox"}});we.addEventListener("change",()=>{we.checked?l.selectedSkills.add(V.name):l.selectedSkills.delete(V.name)});let Ae=Ee.createDiv({cls:"af-create-skill-label"});Ae.createSpan({cls:"af-create-skill-name",text:V.name}),V.description&&Ae.createSpan({cls:"af-create-skill-desc",text:` \u2014 ${V.description}`})}}let Ue=Me.createDiv({cls:"af-form-sublabel"});Ue.setText("Agent-specific skills"),this.addTooltip(Ue,"Custom skills/instructions only for this agent, not shared with others");let Ne=Me.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Custom skills/instructions for this agent...",rows:"4"}});Ne.addEventListener("input",()=>{l.skillsBody=Ne.value});{let B=d.createDiv({cls:"af-create-section"}),V=B.createDiv({cls:"af-create-section-header"}),Ee=V.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ee,"plug");let we=V.createSpan({text:"MCP Servers"});this.addTooltip(we,"Grant agent access to MCP servers"),this.renderAgentMcpPicker(B,l.selectedMcpServers)}let q=d.createDiv({cls:"af-create-section"}),re=q.createDiv({cls:"af-create-section-header"}),j=re.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(j,"file-text");let Ce=re.createSpan({text:"Context"});this.addTooltip(Ce,"Project-specific context included in every run");let xe=q.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Background info, repo structure, conventions...",rows:"4"}});xe.addEventListener("input",()=>{l.contextBody=xe.value});let he=d.createDiv({cls:"af-create-section"}),$e=he.createDiv({cls:"af-create-section-header"}),ie=$e.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(ie,"shield-check"),$e.createSpan({text:"Permissions"}),this.createFormField(he,"Approval required","git_push, file_delete","Comma-separated tool names",B=>{l.approvalRequired=B});let I=he.createDiv({cls:"af-form-row"});I.createDiv({cls:"af-form-label",text:"Allowed Commands"});let se=I.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:`Bash(curl *)
12060
11971
  Bash(python3 *)
12061
11972
  Read
12062
11973
  Edit
12063
11974
  Write`,rows:"4"}});se.addEventListener("input",()=>{l.allowedCommands=se.value});let ce=he.createDiv({cls:"af-form-row"});ce.createDiv({cls:"af-form-label",text:"Blocked Commands"});let te=ce.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:`Bash(git push *)
12064
11975
  Bash(rm -rf *)
12065
- Bash(sudo *)`,rows:"4"}});te.addEventListener("input",()=>{l.blockedCommands=te.value}),he.createDiv({cls:"af-form-hint",text:"On Codex agents these become execpolicy command rules \u2014 only Bash(cmd args *) prefixes are enforced; tool-name rules (Read/Write) and mid-pattern wildcards are ignored, and file/network access is governed by Permission Mode (the sandbox)."});let st=he.createDiv({cls:"af-form-row"});st.createDiv({cls:"af-form-label",text:"Memory enabled"});let os=st.createDiv({cls:"af-agent-card-toggle on"});os.onclick=()=>{let B=os.hasClass("on");os.toggleClass("on",!B),l.memory=!B};let ls=s.createDiv({cls:"af-create-footer"}),cs=ls.createEl("button",{cls:"af-btn-sm",text:"Cancel"});cs.onclick=()=>this.navigate("agents");let Nt=ls.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(Nt,"plus","af-btn-icon"),Nt.appendText(" Create Agent"),Nt.onclick=async()=>{let B=l.name.trim();if(!B){new b.Notice("Agent name is required.");return}let V=oe(B);if(this.plugin.repository.getAgentByName(V)){new b.Notice(`Agent "${V}" already exists.`);return}let Ee=we=>we.split(",").map(Ae=>Ae.trim()).filter(Boolean);try{let we=Ae=>ye(Ae).map(nt=>nt.trim()).filter(Boolean);await this.plugin.repository.createAgentFolder({name:V,description:l.description.trim(),avatar:l.avatar.trim(),tags:Ee(l.tags),systemPrompt:l.systemPrompt.trim(),model:l.model.trim()||"default",adapter:l.adapter,cwd:l.cwd.trim(),timeout:l.timeout,permissionMode:l.permissionMode,effort:l.effort||void 0,approvalRequired:Ee(l.approvalRequired),memory:l.memory,memoryMaxEntries:100,skills:Array.from(l.selectedSkills),mcpServers:Array.from(l.selectedMcpServers),skillsBody:l.skillsBody.trim(),contextBody:l.contextBody.trim(),enabled:l.enabled,permissionRules:{allow:we(l.allowedCommands),deny:we(l.blockedCommands)},autoCompactThreshold:l.autoCompactThreshold,wikiReferences:l.wikiReferences}),l.heartbeatEnabled&&l.heartbeatBody.trim()&&await this.plugin.repository.updateHeartbeat(V,{enabled:l.heartbeatEnabled,schedule:l.heartbeatSchedule.trim(),notify:l.heartbeatNotify,channel:l.heartbeatChannel,channelTarget:l.heartbeatChannel?l.heartbeatChannelTarget:"",body:l.heartbeatBody.trim()}),new b.Notice(`Agent "${V}" created.`),await this.plugin.refreshFromVault(),this.navigate("agent-detail",V)}catch(we){let Ae=we instanceof Error?we.message:String(we);new b.Notice(`Failed to create agent: ${Ae}`)}}}renderCreateSkillPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=s.createDiv({cls:"af-detail-header"}),a=n.createDiv({cls:"af-detail-header-left"}),r=a.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(r,"plus");let o=a.createDiv();o.createDiv({cls:"af-detail-header-name",text:"Create New Skill"}),o.createDiv({cls:"af-detail-header-desc",text:"Define a reusable skill for your agents"});let c=n.createDiv({cls:"af-detail-header-actions"}),l={name:"",description:"",tags:"",body:"",toolsBody:"",referencesBody:"",examplesBody:""},h={none:{label:"None",prompt:""},cli:{label:"CLI Tool Wrapper",prompt:`You are using the {{tool}} CLI. All operations go through the wrapper script.
11976
+ Bash(sudo *)`,rows:"4"}});te.addEventListener("input",()=>{l.blockedCommands=te.value}),he.createDiv({cls:"af-form-hint",text:"On Codex agents these become execpolicy command rules \u2014 only Bash(cmd args *) prefixes are enforced; tool-name rules (Read/Write) and mid-pattern wildcards are ignored, and file/network access is governed by Permission Mode (the sandbox)."});let st=he.createDiv({cls:"af-form-row"});st.createDiv({cls:"af-form-label",text:"Memory enabled"});let ls=st.createDiv({cls:"af-agent-card-toggle on"});ls.onclick=()=>{let B=ls.hasClass("on");ls.toggleClass("on",!B),l.memory=!B};let cs=s.createDiv({cls:"af-create-footer"}),ds=cs.createEl("button",{cls:"af-btn-sm",text:"Cancel"});ds.onclick=()=>this.navigate("agents");let Bt=cs.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(Bt,"plus","af-btn-icon"),Bt.appendText(" Create Agent"),Bt.onclick=async()=>{let B=l.name.trim();if(!B){new b.Notice("Agent name is required.");return}let V=oe(B);if(this.plugin.repository.getAgentByName(V)){new b.Notice(`Agent "${V}" already exists.`);return}let Ee=we=>we.split(",").map(Ae=>Ae.trim()).filter(Boolean);try{let we=Ae=>ye(Ae).map(nt=>nt.trim()).filter(Boolean);await this.plugin.repository.createAgentFolder({name:V,description:l.description.trim(),avatar:l.avatar.trim(),tags:Ee(l.tags),systemPrompt:l.systemPrompt.trim(),model:l.model.trim()||"default",adapter:l.adapter,cwd:l.cwd.trim(),timeout:l.timeout,permissionMode:l.permissionMode,effort:l.effort||void 0,approvalRequired:Ee(l.approvalRequired),memory:l.memory,memoryMaxEntries:100,skills:Array.from(l.selectedSkills),mcpServers:Array.from(l.selectedMcpServers),skillsBody:l.skillsBody.trim(),contextBody:l.contextBody.trim(),enabled:l.enabled,permissionRules:{allow:we(l.allowedCommands),deny:we(l.blockedCommands)},autoCompactThreshold:l.autoCompactThreshold,wikiReferences:l.wikiReferences}),l.heartbeatEnabled&&l.heartbeatBody.trim()&&await this.plugin.repository.updateHeartbeat(V,{enabled:l.heartbeatEnabled,schedule:l.heartbeatSchedule.trim(),notify:l.heartbeatNotify,channel:l.heartbeatChannel,channelTarget:l.heartbeatChannel?l.heartbeatChannelTarget:"",body:l.heartbeatBody.trim()}),new b.Notice(`Agent "${V}" created.`),await this.plugin.refreshFromVault(),this.navigate("agent-detail",V)}catch(we){let Ae=we instanceof Error?we.message:String(we);new b.Notice(`Failed to create agent: ${Ae}`)}}}renderCreateSkillPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=s.createDiv({cls:"af-detail-header"}),a=n.createDiv({cls:"af-detail-header-left"}),i=a.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(i,"plus");let o=a.createDiv();o.createDiv({cls:"af-detail-header-name",text:"Create New Skill"}),o.createDiv({cls:"af-detail-header-desc",text:"Define a reusable skill for your agents"});let c=n.createDiv({cls:"af-detail-header-actions"}),l={name:"",description:"",tags:"",body:"",toolsBody:"",referencesBody:"",examplesBody:""},h={none:{label:"None",prompt:""},cli:{label:"CLI Tool Wrapper",prompt:`You are using the {{tool}} CLI. All operations go through the wrapper script.
12066
11977
 
12067
11978
  Requirements:
12068
11979
  - Ensure required environment variables are set
@@ -12107,28 +12018,28 @@ Usage: tool list [--filter <query>]
12107
12018
 
12108
12019
  User: Show me my tasks for today
12109
12020
 
12110
- Agent: ...`,rows:"6"}});me.addEventListener("input",()=>{l.examplesBody=me.value});let ge=s.createDiv({cls:"af-create-footer"}),X=ge.createEl("button",{cls:"af-btn-sm",text:"Cancel"});X.onclick=()=>this.navigate("skills");let W=ge.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(W,"plus","af-btn-icon"),W.appendText(" Create Skill"),W.onclick=async()=>{let K=l.name.trim();if(!K){new b.Notice("Skill name is required.");return}let ne=oe(K);if(this.plugin.repository.getSkillByName(ne)){new b.Notice(`Skill "${ne}" already exists.`);return}let G=ee=>ee.split(",").map(Q=>Q.trim()).filter(Boolean);try{await this.plugin.repository.createSkillFolder({name:ne,description:l.description.trim(),tags:G(l.tags),body:l.body.trim(),toolsBody:l.toolsBody.trim(),referencesBody:l.referencesBody.trim(),examplesBody:l.examplesBody.trim()}),new b.Notice(`Skill "${ne}" created.`),await this.plugin.refreshFromVault(),this.navigate("skills")}catch(ee){let Q=ee instanceof Error?ee.message:String(ee);new b.Notice(`Failed to create skill: ${Q}`)}}}renderEditAgentPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"bot","No agent selected","");return}let a=this.plugin.runtime.getSnapshot().agents.find(L=>L.name===n);if(!a){this.renderEmptyState(s,"bot","Agent not found",`Agent "${n}" was not found`);return}let r=s.createDiv({cls:"af-detail-header"}),o=r.createDiv({cls:"af-detail-header-left"}),c=o.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(c,"edit");let l=o.createDiv();l.createDiv({cls:"af-detail-header-name",text:`Edit Agent: ${a.name}`}),l.createDiv({cls:"af-detail-header-desc",text:"Modify agent configuration"});let h=r.createDiv({cls:"af-detail-header-actions"}),d={name:a.name,description:a.description??"",avatar:a.avatar,tags:a.tags.join(", "),systemPrompt:a.body,model:a.model,adapter:a.adapter,cwd:a.cwd??"",timeout:a.timeout,permissionMode:a.permissionMode,effort:a.effort??"",selectedSkills:new Set(a.skills),selectedMcpServers:new Set(a.mcpServers??[]),skillsBody:a.skillsBody,contextBody:a.contextBody,approvalRequired:a.approvalRequired.join(", "),memory:a.memory,memoryTokenBudget:a.memoryTokenBudget,reflectionEnabled:a.reflection.enabled,reflectionProposeSkills:a.reflection.proposeSkills,enabled:a.enabled,allowedCommands:a.permissionRules.allow.join(`
12021
+ Agent: ...`,rows:"6"}});me.addEventListener("input",()=>{l.examplesBody=me.value});let ge=s.createDiv({cls:"af-create-footer"}),X=ge.createEl("button",{cls:"af-btn-sm",text:"Cancel"});X.onclick=()=>this.navigate("skills");let W=ge.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(W,"plus","af-btn-icon"),W.appendText(" Create Skill"),W.onclick=async()=>{let K=l.name.trim();if(!K){new b.Notice("Skill name is required.");return}let ne=oe(K);if(this.plugin.repository.getSkillByName(ne)){new b.Notice(`Skill "${ne}" already exists.`);return}let G=ee=>ee.split(",").map(Q=>Q.trim()).filter(Boolean);try{await this.plugin.repository.createSkillFolder({name:ne,description:l.description.trim(),tags:G(l.tags),body:l.body.trim(),toolsBody:l.toolsBody.trim(),referencesBody:l.referencesBody.trim(),examplesBody:l.examplesBody.trim()}),new b.Notice(`Skill "${ne}" created.`),await this.plugin.refreshFromVault(),this.navigate("skills")}catch(ee){let Q=ee instanceof Error?ee.message:String(ee);new b.Notice(`Failed to create skill: ${Q}`)}}}renderEditAgentPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"bot","No agent selected","");return}let a=this.plugin.runtime.getSnapshot().agents.find(L=>L.name===n);if(!a){this.renderEmptyState(s,"bot","Agent not found",`Agent "${n}" was not found`);return}let i=s.createDiv({cls:"af-detail-header"}),o=i.createDiv({cls:"af-detail-header-left"}),c=o.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(c,"edit");let l=o.createDiv();l.createDiv({cls:"af-detail-header-name",text:`Edit Agent: ${a.name}`}),l.createDiv({cls:"af-detail-header-desc",text:"Modify agent configuration"});let h=i.createDiv({cls:"af-detail-header-actions"}),d={name:a.name,description:a.description??"",avatar:a.avatar,tags:a.tags.join(", "),systemPrompt:a.body,model:a.model,adapter:a.adapter,cwd:a.cwd??"",timeout:a.timeout,permissionMode:a.permissionMode,effort:a.effort??"",selectedSkills:new Set(a.skills),selectedMcpServers:new Set(a.mcpServers??[]),skillsBody:a.skillsBody,contextBody:a.contextBody,approvalRequired:a.approvalRequired.join(", "),memory:a.memory,memoryTokenBudget:a.memoryTokenBudget,reflectionEnabled:a.reflection.enabled,reflectionProposeSkills:a.reflection.proposeSkills,enabled:a.enabled,allowedCommands:a.permissionRules.allow.join(`
12111
12022
  `),blockedCommands:a.permissionRules.deny.join(`
12112
- `),heartbeatEnabled:a.heartbeatEnabled,heartbeatSchedule:a.heartbeatSchedule,heartbeatBody:a.heartbeatBody,heartbeatNotify:a.heartbeatNotify,heartbeatChannel:a.heartbeatChannel,heartbeatChannelTarget:a.heartbeatChannelTarget,autoCompactThreshold:a.autoCompactThreshold??85,wikiReferences:(a.wikiReferences??[]).map(L=>L.agent)},u=s.createDiv({cls:"af-create-form"}),p=u.createDiv({cls:"af-create-section"}),m=p.createDiv({cls:"af-create-section-header"}),f=m.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(f,"user"),m.createSpan({text:"Identity"});let g=p.createDiv({cls:"af-form-row"});g.createDiv({cls:"af-form-label",text:"Name"}),g.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.name,disabled:"true"}}).setCssStyles({opacity:"0.6"}),this.createFormField(p,"Description","Monitors deployments and reports status","",L=>{d.description=L},a.description??"");let v=p.createDiv({cls:"af-form-row"});v.createDiv({cls:"af-form-label",text:"Avatar"});let k=v.createEl("button",{cls:"af-avatar-picker-btn"}),w=k.createDiv({cls:"af-avatar-picker-preview"});this.renderAgentAvatar(w,{...a,avatar:d.avatar??a.avatar}),k.createSpan({cls:"af-avatar-picker-label",text:d.avatar||a.avatar||"Pick icon\u2026"}),k.addEventListener("click",()=>{new $n(this.app,d.avatar??a.avatar,L=>{d.avatar=L,w.empty(),(0,b.setIcon)(w,L),k.querySelector(".af-avatar-picker-label")?.setText(L)}).open()}),this.createFormField(p,"Tags","devops, monitoring","Comma-separated",L=>{d.tags=L},a.tags.join(", "));let S=p.createDiv({cls:"af-form-row"});S.createDiv({cls:"af-form-label",text:"Enabled"});let T=S.createDiv({cls:`af-agent-card-toggle${a.enabled?" on":""}`});T.onclick=()=>{let L=T.hasClass("on");T.toggleClass("on",!L),d.enabled=!L};let _=u.createDiv({cls:"af-create-section"}),D=_.createDiv({cls:"af-create-section-header"}),O=D.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(O,"message-square"),D.createSpan({text:"System Prompt"});let C=_.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"System prompt...",rows:"10"}});C.value=a.body,C.addEventListener("input",()=>{d.systemPrompt=C.value});let E=u.createDiv({cls:"af-create-section"}),P=E.createDiv({cls:"af-create-section-header"}),N=P.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(N,"settings"),P.createSpan({text:"Runtime Config"});let M=E.createDiv({cls:"af-create-config-grid"}),U=M.createDiv({cls:"af-form-row"});U.createDiv({cls:"af-form-label",text:"Adapter"});let $=U.createEl("select",{cls:"af-form-select"});for(let[L,Y,Te]of Jo){let Le=$.createEl("option",{text:Y,attr:{value:L,...Te?{disabled:"true"}:{}}});(L===a.adapter||Ls(a.adapter)&&L==="codex")&&(Le.selected=!0)}let Z=M.createDiv({cls:"af-form-row"}),de=Z.createDiv({cls:"af-form-label",text:"Model"});this.addTooltip(de,`Aliases (opus/sonnet/haiku/opusplan) work on any backend. Choose Custom\u2026 for a pinned ID or Bedrock/Vertex/Foundry. Blank = use Settings default (${this.plugin.settings.defaultModel||"CLI default"}).`);let me=Z.createDiv({cls:"af-form-field-wrap"}),ge=()=>{Pt(me,{value:d.model,adapter:d.adapter,onChange:L=>{d.model=L}})};ge();let X=()=>{};$.addEventListener("change",()=>{d.adapter=$.value,(Ls(d.adapter)?Qs:Zs).some(Y=>Y.value===d.model.trim())&&(d.model=""),ge(),d.permissionMode=Hn(d.permissionMode,d.adapter),X()});let W=M.createDiv({cls:"af-form-row"});W.createDiv({cls:"af-form-label",text:"Working Dir"});let K=W.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Leave empty for vault root",value:a.cwd??""}});K.addEventListener("input",()=>{d.cwd=K.value});let ne=M.createDiv({cls:"af-form-row"});ne.createDiv({cls:"af-form-label",text:"Timeout (sec)"});let G=ne.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",value:String(a.timeout)}});G.addEventListener("input",()=>{let L=parseInt(G.value,10);!isNaN(L)&&L>0&&(d.timeout=L)});let ee=M.createDiv({cls:"af-form-row"});ee.createDiv({cls:"af-form-label",text:"Permission Mode"});let Q=ee.createEl("select",{cls:"af-form-select"}),ae=M.createDiv({cls:"af-form-hint",text:""});X=()=>{d.permissionMode=Hn(d.permissionMode,d.adapter);let L=Wn(d.adapter);Q.empty();for(let[Y,Te]of L){let Le=Q.createEl("option",{text:Te,attr:{value:Y}});Y===d.permissionMode&&(Le.selected=!0)}ae.textContent=L.find(([Y])=>Y===Q.value)?.[2]??""},X(),Q.addEventListener("change",()=>{d.permissionMode=Q.value,ae.textContent=Wn(d.adapter).find(([L])=>L===Q.value)?.[2]??""});let ve=M.createDiv({cls:"af-form-row"});ve.createDiv({cls:"af-form-label",text:"Effort Level"});let Pe=ve.createEl("select",{cls:"af-form-select"});for(let[L,Y]of[["","Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]]){let Te=Pe.createEl("option",{text:Y,attr:{value:L}});L===(a.effort??"")&&(Te.selected=!0)}Pe.addEventListener("change",()=>{d.effort=Pe.value}),M.createDiv({cls:"af-form-hint",text:"Controls reasoning depth \u2014 low is fast, max is most thorough"});let Ie=M.createDiv({cls:"af-form-row"}),Me=Ie.createDiv({cls:"af-form-label",text:"Auto-compact at"});this.addTooltip(Me,"Percent of context window at which the chat auto-invokes /compact before the next message. 85% is a good default. Set 0 to disable.");let Fe=Ie.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",min:"0",max:"100",value:String(d.autoCompactThreshold)}});Fe.addEventListener("input",()=>{let L=parseInt(Fe.value,10);!isNaN(L)&&L>=0&&L<=100&&(d.autoCompactThreshold=L)}),M.createDiv({cls:"af-form-hint",text:"0 disables auto-compact"});{let L=this.plugin.runtime.getSnapshot().agents.filter(Y=>Y.wikiKeeper!==void 0);if(L.length>0){let Y=M.createDiv({cls:"af-form-row af-form-row-toggle"}),Te=Y.createDiv({cls:"af-form-label",text:"Wiki access"});this.addTooltip(Te,"Lets this agent read + cite from the selected Wiki Keeper scopes (requires the wiki-query skill).");let Le=Y.createDiv({cls:"af-form-field-wrap"});for(let qe of L){let at=Le.createEl("label",{cls:"af-form-checkbox-row"}),We=at.createEl("input",{attr:{type:"checkbox"}});d.wikiReferences.includes(qe.name)&&(We.checked=!0),at.createSpan({text:` ${qe.name}`,cls:"af-form-checkbox-label"}),We.addEventListener("change",()=>{We.checked?d.wikiReferences.includes(qe.name)||d.wikiReferences.push(qe.name):d.wikiReferences=d.wikiReferences.filter(ze=>ze!==qe.name)})}}}if(a.isFolder){let L=u.createDiv({cls:"af-create-section"}),Y=L.createDiv({cls:"af-create-section-header"}),Te=Y.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Te,"heart-pulse");let Le=Y.createSpan({text:"Heartbeat"});this.addTooltip(Le,"Autonomous periodic run \u2014 what the agent does when no one is asking");let qe=L.createDiv({cls:"af-form-row af-form-row-toggle"});qe.createDiv({cls:"af-form-label",text:"Enabled"});let at=qe.createDiv({cls:`af-agent-card-toggle${d.heartbeatEnabled?" on":""}`}),We=L.createDiv();We.setCssStyles({display:d.heartbeatEnabled?"":"none"}),at.onclick=()=>{let ct=at.hasClass("on");at.toggleClass("on",!ct),d.heartbeatEnabled=!ct,We.setCssStyles({display:ct?"none":""})},this.renderHeartbeatSchedule(We,d);let ze=We.createDiv({cls:"af-form-row af-form-row-toggle"}),Za=ze.createDiv({cls:"af-form-label"});Za.setText("Notify"),this.addTooltip(Za,"Show an Obsidian notice when the heartbeat completes");let Gn=ze.createDiv({cls:`af-agent-card-toggle${d.heartbeatNotify?" on":""}`});Gn.onclick=()=>{let ct=Gn.hasClass("on");Gn.toggleClass("on",!ct),d.heartbeatNotify=!ct};let sl=this.plugin.runtime.getSnapshot(),ei=We.createDiv({cls:"af-form-row"}),ti=ei.createDiv({cls:"af-form-label"});ti.setText("Post to channel"),this.addTooltip(ti,"Heartbeat results are posted to this Slack channel when the run completes");let Bs=ei.createEl("select",{cls:"af-form-select"});Bs.createEl("option",{text:"(none)",attr:{value:""}});for(let ct of sl.channels){let nl=Bs.createEl("option",{text:ct.name,attr:{value:ct.name}});ct.name===d.heartbeatChannel&&(nl.selected=!0)}Bs.addEventListener("change",()=>{d.heartbeatChannel=Bs.value,ni()});let Vn=We.createDiv({cls:"af-form-row"}),si=Vn.createDiv({cls:"af-form-label"});si.setText("Target ID"),this.addTooltip(si,"Specific channel id to post to (Discord/Slack channel id, Telegram chat id). Empty = DM the channel\u2019s first allowed user.");let Yn=Vn.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Channel/chat id \u2014 empty = DM"}});Yn.value=d.heartbeatChannelTarget,Yn.addEventListener("input",()=>{d.heartbeatChannelTarget=Yn.value.trim()});let ni=()=>{Vn.setCssStyles({display:d.heartbeatChannel?"":"none"})};ni();let Us=We.createDiv({cls:"af-form-label"});Us.setCssStyles({width:"auto"}),Us.setCssStyles({marginTop:"12px"}),Us.setText("Instruction"),this.addTooltip(Us,'What the agent does on each heartbeat. Also used by the "Run Now" button.');let Kn=We.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Check status, scan for issues, report findings...",rows:"8"}});Kn.value=d.heartbeatBody,Kn.addEventListener("input",()=>{d.heartbeatBody=Kn.value})}let ke=u.createDiv({cls:"af-create-section"}),Oe=ke.createDiv({cls:"af-create-section-header"}),Ue=Oe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ue,"puzzle"),Oe.createSpan({text:"Skills"});let Ne=this.plugin.runtime.getSnapshot();if(Ne.skills.length>0){ke.createDiv({cls:"af-form-sublabel",text:"Shared Skills"});let L=ke.createDiv({cls:"af-create-skills-grid"});for(let Y of Ne.skills){let Te=L.createDiv({cls:"af-create-skill-item"}),Le=Te.createEl("input",{cls:"af-form-toggle",attr:{type:"checkbox"}});Le.checked=d.selectedSkills.has(Y.name),Le.addEventListener("change",()=>{Le.checked?d.selectedSkills.add(Y.name):d.selectedSkills.delete(Y.name)});let qe=Te.createDiv({cls:"af-create-skill-label"});qe.createSpan({cls:"af-create-skill-name",text:Y.name}),Y.description&&qe.createSpan({cls:"af-create-skill-desc",text:` \u2014 ${Y.description}`})}}let q=ke.createDiv({cls:"af-form-sublabel"});q.setText("Agent-specific skills"),this.addTooltip(q,"Custom skills/instructions only for this agent, not shared with others");let ie=ke.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Custom skills/instructions for this agent...",rows:"4"}});ie.value=a.skillsBody,ie.addEventListener("input",()=>{d.skillsBody=ie.value});let j=u.createDiv({cls:"af-create-section"}),Ce=j.createDiv({cls:"af-create-section-header"}),xe=Ce.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(xe,"plug");let he=Ce.createSpan({text:"MCP Servers"});this.addTooltip(he,"Grant agent access to MCP servers"),this.renderAgentMcpPicker(j,d.selectedMcpServers);let $e=u.createDiv({cls:"af-create-section"}),re=$e.createDiv({cls:"af-create-section-header"}),I=re.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(I,"file-text");let se=re.createSpan({text:"Context"});this.addTooltip(se,"Project-specific context included in every run");let ce=$e.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Background info, repo structure, conventions...",rows:"4"}});ce.value=a.contextBody,ce.addEventListener("input",()=>{d.contextBody=ce.value});let te=u.createDiv({cls:"af-create-section"}),st=te.createDiv({cls:"af-create-section-header"}),os=st.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(os,"shield-check"),st.createSpan({text:"Permissions"}),this.createFormField(te,"Approval required","git_push, file_delete","Comma-separated tool names",L=>{d.approvalRequired=L},a.approvalRequired.join(", "));let ls=te.createDiv({cls:"af-form-row"});ls.createDiv({cls:"af-form-label",text:"Allowed Commands"});let cs=ls.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:`Bash(curl *)
12023
+ `),heartbeatEnabled:a.heartbeatEnabled,heartbeatSchedule:a.heartbeatSchedule,heartbeatBody:a.heartbeatBody,heartbeatNotify:a.heartbeatNotify,heartbeatChannel:a.heartbeatChannel,heartbeatChannelTarget:a.heartbeatChannelTarget,autoCompactThreshold:a.autoCompactThreshold??85,wikiReferences:(a.wikiReferences??[]).map(L=>L.agent)},u=s.createDiv({cls:"af-create-form"}),p=u.createDiv({cls:"af-create-section"}),m=p.createDiv({cls:"af-create-section-header"}),f=m.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(f,"user"),m.createSpan({text:"Identity"});let g=p.createDiv({cls:"af-form-row"});g.createDiv({cls:"af-form-label",text:"Name"}),g.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.name,disabled:"true"}}).setCssStyles({opacity:"0.6"}),this.createFormField(p,"Description","Monitors deployments and reports status","",L=>{d.description=L},a.description??"");let v=p.createDiv({cls:"af-form-row"});v.createDiv({cls:"af-form-label",text:"Avatar"});let k=v.createEl("button",{cls:"af-avatar-picker-btn"}),w=k.createDiv({cls:"af-avatar-picker-preview"});this.renderAgentAvatar(w,{...a,avatar:d.avatar??a.avatar}),k.createSpan({cls:"af-avatar-picker-label",text:d.avatar||a.avatar||"Pick icon\u2026"}),k.addEventListener("click",()=>{new $n(this.app,d.avatar??a.avatar,L=>{d.avatar=L,w.empty(),(0,b.setIcon)(w,L),k.querySelector(".af-avatar-picker-label")?.setText(L)}).open()}),this.createFormField(p,"Tags","devops, monitoring","Comma-separated",L=>{d.tags=L},a.tags.join(", "));let S=p.createDiv({cls:"af-form-row"});S.createDiv({cls:"af-form-label",text:"Enabled"});let T=S.createDiv({cls:`af-agent-card-toggle${a.enabled?" on":""}`});T.onclick=()=>{let L=T.hasClass("on");T.toggleClass("on",!L),d.enabled=!L};let _=u.createDiv({cls:"af-create-section"}),D=_.createDiv({cls:"af-create-section-header"}),O=D.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(O,"message-square"),D.createSpan({text:"System Prompt"});let C=_.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"System prompt...",rows:"10"}});C.value=a.body,C.addEventListener("input",()=>{d.systemPrompt=C.value});let E=u.createDiv({cls:"af-create-section"}),P=E.createDiv({cls:"af-create-section-header"}),N=P.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(N,"settings"),P.createSpan({text:"Runtime Config"});let M=E.createDiv({cls:"af-create-config-grid"}),U=M.createDiv({cls:"af-form-row"});U.createDiv({cls:"af-form-label",text:"Adapter"});let $=U.createEl("select",{cls:"af-form-select"});for(let[L,Y,Te]of Xo){let Le=$.createEl("option",{text:Y,attr:{value:L,...Te?{disabled:"true"}:{}}});(L===a.adapter||Ls(a.adapter)&&L==="codex")&&(Le.selected=!0)}let Z=M.createDiv({cls:"af-form-row"}),de=Z.createDiv({cls:"af-form-label",text:"Model"});this.addTooltip(de,`Aliases (opus/sonnet/haiku/opusplan) work on any backend. Choose Custom\u2026 for a pinned ID or Bedrock/Vertex/Foundry. Blank = use Settings default (${this.plugin.settings.defaultModel||"CLI default"}).`);let me=Z.createDiv({cls:"af-form-field-wrap"}),ge=()=>{Rt(me,{value:d.model,adapter:d.adapter,onChange:L=>{d.model=L}})};ge();let X=()=>{};$.addEventListener("change",()=>{d.adapter=$.value,(Ls(d.adapter)?Qs:Zs).some(Y=>Y.value===d.model.trim())&&(d.model=""),ge(),d.permissionMode=Hn(d.permissionMode,d.adapter),X()});let W=M.createDiv({cls:"af-form-row"});W.createDiv({cls:"af-form-label",text:"Working Dir"});let K=W.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Leave empty for vault root",value:a.cwd??""}});K.addEventListener("input",()=>{d.cwd=K.value});let ne=M.createDiv({cls:"af-form-row"});ne.createDiv({cls:"af-form-label",text:"Timeout (sec)"});let G=ne.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",value:String(a.timeout)}});G.addEventListener("input",()=>{let L=parseInt(G.value,10);!isNaN(L)&&L>0&&(d.timeout=L)});let ee=M.createDiv({cls:"af-form-row"});ee.createDiv({cls:"af-form-label",text:"Permission Mode"});let Q=ee.createEl("select",{cls:"af-form-select"}),ae=M.createDiv({cls:"af-form-hint",text:""});X=()=>{d.permissionMode=Hn(d.permissionMode,d.adapter);let L=Wn(d.adapter);Q.empty();for(let[Y,Te]of L){let Le=Q.createEl("option",{text:Te,attr:{value:Y}});Y===d.permissionMode&&(Le.selected=!0)}ae.textContent=L.find(([Y])=>Y===Q.value)?.[2]??""},X(),Q.addEventListener("change",()=>{d.permissionMode=Q.value,ae.textContent=Wn(d.adapter).find(([L])=>L===Q.value)?.[2]??""});let ve=M.createDiv({cls:"af-form-row"});ve.createDiv({cls:"af-form-label",text:"Effort Level"});let Pe=ve.createEl("select",{cls:"af-form-select"});for(let[L,Y]of[["","Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]]){let Te=Pe.createEl("option",{text:Y,attr:{value:L}});L===(a.effort??"")&&(Te.selected=!0)}Pe.addEventListener("change",()=>{d.effort=Pe.value}),M.createDiv({cls:"af-form-hint",text:"Controls reasoning depth \u2014 low is fast, max is most thorough"});let Ie=M.createDiv({cls:"af-form-row"}),Me=Ie.createDiv({cls:"af-form-label",text:"Auto-compact at"});this.addTooltip(Me,"Percent of context window at which the chat auto-invokes /compact before the next message. 85% is a good default. Set 0 to disable.");let Fe=Ie.createEl("input",{cls:"af-form-input af-form-input-sm",attr:{type:"number",min:"0",max:"100",value:String(d.autoCompactThreshold)}});Fe.addEventListener("input",()=>{let L=parseInt(Fe.value,10);!isNaN(L)&&L>=0&&L<=100&&(d.autoCompactThreshold=L)}),M.createDiv({cls:"af-form-hint",text:"0 disables auto-compact"});{let L=this.plugin.runtime.getSnapshot().agents.filter(Y=>Y.wikiKeeper!==void 0);if(L.length>0){let Y=M.createDiv({cls:"af-form-row af-form-row-toggle"}),Te=Y.createDiv({cls:"af-form-label",text:"Wiki access"});this.addTooltip(Te,"Lets this agent read + cite from the selected Wiki Keeper scopes (requires the wiki-query skill).");let Le=Y.createDiv({cls:"af-form-field-wrap"});for(let qe of L){let at=Le.createEl("label",{cls:"af-form-checkbox-row"}),We=at.createEl("input",{attr:{type:"checkbox"}});d.wikiReferences.includes(qe.name)&&(We.checked=!0),at.createSpan({text:` ${qe.name}`,cls:"af-form-checkbox-label"}),We.addEventListener("change",()=>{We.checked?d.wikiReferences.includes(qe.name)||d.wikiReferences.push(qe.name):d.wikiReferences=d.wikiReferences.filter(ze=>ze!==qe.name)})}}}if(a.isFolder){let L=u.createDiv({cls:"af-create-section"}),Y=L.createDiv({cls:"af-create-section-header"}),Te=Y.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Te,"heart-pulse");let Le=Y.createSpan({text:"Heartbeat"});this.addTooltip(Le,"Autonomous periodic run \u2014 what the agent does when no one is asking");let qe=L.createDiv({cls:"af-form-row af-form-row-toggle"});qe.createDiv({cls:"af-form-label",text:"Enabled"});let at=qe.createDiv({cls:`af-agent-card-toggle${d.heartbeatEnabled?" on":""}`}),We=L.createDiv();We.setCssStyles({display:d.heartbeatEnabled?"":"none"}),at.onclick=()=>{let ct=at.hasClass("on");at.toggleClass("on",!ct),d.heartbeatEnabled=!ct,We.setCssStyles({display:ct?"none":""})},this.renderHeartbeatSchedule(We,d);let ze=We.createDiv({cls:"af-form-row af-form-row-toggle"}),Za=ze.createDiv({cls:"af-form-label"});Za.setText("Notify"),this.addTooltip(Za,"Show an Obsidian notice when the heartbeat completes");let Gn=ze.createDiv({cls:`af-agent-card-toggle${d.heartbeatNotify?" on":""}`});Gn.onclick=()=>{let ct=Gn.hasClass("on");Gn.toggleClass("on",!ct),d.heartbeatNotify=!ct};let nl=this.plugin.runtime.getSnapshot(),er=We.createDiv({cls:"af-form-row"}),tr=er.createDiv({cls:"af-form-label"});tr.setText("Post to channel"),this.addTooltip(tr,"Heartbeat results are posted to this Slack channel when the run completes");let Bs=er.createEl("select",{cls:"af-form-select"});Bs.createEl("option",{text:"(none)",attr:{value:""}});for(let ct of nl.channels){let al=Bs.createEl("option",{text:ct.name,attr:{value:ct.name}});ct.name===d.heartbeatChannel&&(al.selected=!0)}Bs.addEventListener("change",()=>{d.heartbeatChannel=Bs.value,nr()});let Vn=We.createDiv({cls:"af-form-row"}),sr=Vn.createDiv({cls:"af-form-label"});sr.setText("Target ID"),this.addTooltip(sr,"Specific channel id to post to (Discord/Slack channel id, Telegram chat id). Empty = DM the channel\u2019s first allowed user.");let Yn=Vn.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Channel/chat id \u2014 empty = DM"}});Yn.value=d.heartbeatChannelTarget,Yn.addEventListener("input",()=>{d.heartbeatChannelTarget=Yn.value.trim()});let nr=()=>{Vn.setCssStyles({display:d.heartbeatChannel?"":"none"})};nr();let Us=We.createDiv({cls:"af-form-label"});Us.setCssStyles({width:"auto"}),Us.setCssStyles({marginTop:"12px"}),Us.setText("Instruction"),this.addTooltip(Us,'What the agent does on each heartbeat. Also used by the "Run Now" button.');let Kn=We.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Check status, scan for issues, report findings...",rows:"8"}});Kn.value=d.heartbeatBody,Kn.addEventListener("input",()=>{d.heartbeatBody=Kn.value})}let ke=u.createDiv({cls:"af-create-section"}),Oe=ke.createDiv({cls:"af-create-section-header"}),Ue=Oe.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Ue,"puzzle"),Oe.createSpan({text:"Skills"});let Ne=this.plugin.runtime.getSnapshot();if(Ne.skills.length>0){ke.createDiv({cls:"af-form-sublabel",text:"Shared Skills"});let L=ke.createDiv({cls:"af-create-skills-grid"});for(let Y of Ne.skills){let Te=L.createDiv({cls:"af-create-skill-item"}),Le=Te.createEl("input",{cls:"af-form-toggle",attr:{type:"checkbox"}});Le.checked=d.selectedSkills.has(Y.name),Le.addEventListener("change",()=>{Le.checked?d.selectedSkills.add(Y.name):d.selectedSkills.delete(Y.name)});let qe=Te.createDiv({cls:"af-create-skill-label"});qe.createSpan({cls:"af-create-skill-name",text:Y.name}),Y.description&&qe.createSpan({cls:"af-create-skill-desc",text:` \u2014 ${Y.description}`})}}let q=ke.createDiv({cls:"af-form-sublabel"});q.setText("Agent-specific skills"),this.addTooltip(q,"Custom skills/instructions only for this agent, not shared with others");let re=ke.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Custom skills/instructions for this agent...",rows:"4"}});re.value=a.skillsBody,re.addEventListener("input",()=>{d.skillsBody=re.value});let j=u.createDiv({cls:"af-create-section"}),Ce=j.createDiv({cls:"af-create-section-header"}),xe=Ce.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(xe,"plug");let he=Ce.createSpan({text:"MCP Servers"});this.addTooltip(he,"Grant agent access to MCP servers"),this.renderAgentMcpPicker(j,d.selectedMcpServers);let $e=u.createDiv({cls:"af-create-section"}),ie=$e.createDiv({cls:"af-create-section-header"}),I=ie.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(I,"file-text");let se=ie.createSpan({text:"Context"});this.addTooltip(se,"Project-specific context included in every run");let ce=$e.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:"Background info, repo structure, conventions...",rows:"4"}});ce.value=a.contextBody,ce.addEventListener("input",()=>{d.contextBody=ce.value});let te=u.createDiv({cls:"af-create-section"}),st=te.createDiv({cls:"af-create-section-header"}),ls=st.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(ls,"shield-check"),st.createSpan({text:"Permissions"}),this.createFormField(te,"Approval required","git_push, file_delete","Comma-separated tool names",L=>{d.approvalRequired=L},a.approvalRequired.join(", "));let cs=te.createDiv({cls:"af-form-row"});cs.createDiv({cls:"af-form-label",text:"Allowed Commands"});let ds=cs.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:`Bash(curl *)
12113
12024
  Bash(python3 *)
12114
12025
  Read
12115
12026
  Edit
12116
- Write`,rows:"4"}});cs.value=a.permissionRules.allow.join(`
12117
- `),cs.addEventListener("input",()=>{d.allowedCommands=cs.value});let Nt=te.createDiv({cls:"af-form-row"});Nt.createDiv({cls:"af-form-label",text:"Blocked Commands"});let B=Nt.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:`Bash(git push *)
12027
+ Write`,rows:"4"}});ds.value=a.permissionRules.allow.join(`
12028
+ `),ds.addEventListener("input",()=>{d.allowedCommands=ds.value});let Bt=te.createDiv({cls:"af-form-row"});Bt.createDiv({cls:"af-form-label",text:"Blocked Commands"});let B=Bt.createEl("textarea",{cls:"af-create-textarea",attr:{placeholder:`Bash(git push *)
12118
12029
  Bash(rm -rf *)
12119
12030
  Bash(sudo *)`,rows:"4"}});B.value=a.permissionRules.deny.join(`
12120
- `),B.addEventListener("input",()=>{d.blockedCommands=B.value}),te.createDiv({cls:"af-form-hint",text:"On Codex agents these become execpolicy command rules \u2014 only Bash(cmd args *) prefixes are enforced; tool-name rules (Read/Write) and mid-pattern wildcards are ignored, and file/network access is governed by Permission Mode (the sandbox)."});let V=te.createDiv({cls:"af-form-row"});V.createDiv({cls:"af-form-label",text:"Memory enabled"});let Ee=V.createDiv({cls:`af-agent-card-toggle${a.memory?" on":""}`});Ee.onclick=()=>{let L=Ee.hasClass("on");Ee.toggleClass("on",!L),d.memory=!L};let we=te.createDiv({cls:"af-form-row"});we.createDiv({cls:"af-form-label",text:"Memory token budget"});let Ae=we.createEl("input",{cls:"af-create-input",attr:{type:"number",min:"200",step:"100"}});Ae.value=String(d.memoryTokenBudget),Ae.addEventListener("input",()=>{let L=parseInt(Ae.value,10);Number.isFinite(L)&&(d.memoryTokenBudget=L)});let nt=te.createDiv({cls:"af-form-row"});nt.createDiv({cls:"af-form-label",text:"Nightly reflection"});let je=nt.createDiv({cls:`af-agent-card-toggle${a.reflection.enabled?" on":""}`});je.onclick=()=>{let L=je.hasClass("on");je.toggleClass("on",!L),d.reflectionEnabled=!L};let Tt=te.createDiv({cls:"af-form-row"});Tt.createDiv({cls:"af-form-label",text:"Propose skills from memory"});let Bt=Tt.createDiv({cls:`af-agent-card-toggle${a.reflection.proposeSkills?" on":""}`});Bt.onclick=()=>{let L=Bt.hasClass("on");Bt.toggleClass("on",!L),d.reflectionProposeSkills=!L};let bt=s.createDiv({cls:"af-create-footer"}),ds=bt.createEl("button",{cls:"af-btn-sm danger"});R(ds,"trash-2","af-btn-icon"),ds.appendText(" Delete"),ds.onclick=()=>void this.plugin.deleteAgent(a.name),bt.createDiv({cls:"af-toolbar-spacer"});let Ns=bt.createEl("button",{cls:"af-btn-sm",text:"Cancel"});Ns.onclick=()=>this.navigate("agent-detail",a.name);let Ut=bt.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(Ut,"check","af-btn-icon"),Ut.appendText(" Save Changes"),Ut.onclick=async()=>{let L=Y=>Y.split(",").map(Te=>Te.trim()).filter(Boolean);try{let Y=Te=>ye(Te).map(Le=>Le.trim()).filter(Boolean);await this.plugin.repository.updateAgent(a.name,{description:d.description.trim(),avatar:d.avatar.trim(),tags:L(d.tags),systemPrompt:d.systemPrompt.trim(),model:d.model.trim()||"default",adapter:d.adapter,cwd:d.cwd.trim(),timeout:d.timeout,permissionMode:d.permissionMode,effort:d.effort||void 0,approvalRequired:L(d.approvalRequired),memory:d.memory,memoryTokenBudget:d.memoryTokenBudget,reflectionEnabled:d.reflectionEnabled,reflectionProposeSkills:d.reflectionProposeSkills,skills:Array.from(d.selectedSkills),mcpServers:Array.from(d.selectedMcpServers),skillsBody:d.skillsBody.trim(),contextBody:d.contextBody.trim(),enabled:d.enabled,permissionRules:{allow:Y(d.allowedCommands),deny:Y(d.blockedCommands)},autoCompactThreshold:d.autoCompactThreshold,wikiReferences:d.wikiReferences}),a.isFolder&&await this.plugin.repository.updateHeartbeat(a.name,{enabled:d.heartbeatEnabled,schedule:d.heartbeatSchedule.trim(),notify:d.heartbeatNotify,channel:d.heartbeatChannel,channelTarget:d.heartbeatChannel?d.heartbeatChannelTarget:"",body:d.heartbeatBody.trim()}),new b.Notice(`Agent "${a.name}" updated.`),await this.plugin.refreshFromVault(),this.navigate("agent-detail",a.name)}catch(Y){let Te=Y instanceof Error?Y.message:String(Y);new b.Notice(`Failed to update agent: ${Te}`)}}}renderTaskChannelDelivery(e,s,n){let a=e.createDiv({cls:"af-form-row"}),r=a.createDiv({cls:"af-form-label",text:"Channel"}),o=a.createEl("select",{cls:"af-form-select"});o.createEl("option",{text:"\u2014 none (run log only) \u2014",attr:{value:""}});for(let u of s){let p=o.createEl("option",{text:u.name,attr:{value:u.name}});u.name===n.channel&&(p.selected=!0)}this.addTooltip(r,"Post this task\u2019s full output to a channel when it finishes. Leave as none to only write the run log (the default batched behavior).");let c=e.createDiv({cls:"af-form-row"}),l=c.createDiv({cls:"af-form-label",text:"Target ID"}),h=c.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Discord/Slack channel ID or Telegram chat ID \u2014 empty = DM you"}});h.value=n.channelTarget,h.addEventListener("input",()=>{n.channelTarget=h.value.trim()}),this.addTooltip(l,"Where in the channel to post. For Discord, enable Developer Mode and right-click the channel \u2192 Copy Channel ID. Leave empty to DM the first allowed user instead.");let d=()=>{c.setCssStyles({display:n.channel?"":"none"})};d(),o.addEventListener("change",()=>{n.channel=o.value,d()})}renderCreateTaskPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-detail-header"}),r=a.createDiv({cls:"af-detail-header-left"}),o=r.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(o,"plus");let c=r.createDiv();c.createDiv({cls:"af-detail-header-name",text:"Create New Task"}),c.createDiv({cls:"af-detail-header-desc",text:"Configure a new task for your fleet"});let l=a.createDiv({cls:"af-detail-header-actions"}),h={title:"",agent:n.agents[0]?.name??"",priority:"medium",tags:"",body:"",scheduleEnabled:!1,scheduleMode:"recurring",schedule:"0 9 * * *",runAt:"",type:"immediate",enabled:!0,catchUp:!0,effort:"",model:"",channel:"",channelTarget:""},d=s.createDiv({cls:"af-create-form"}),u=d.createDiv({cls:"af-create-section"}),p=u.createDiv({cls:"af-create-section-header"}),m=p.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(m,"file-text"),p.createSpan({text:"Task Details"}),this.createFormField(u,"Title","Daily status report","Used as the task identifier",q=>{h.title=q});let f=u.createDiv({cls:"af-form-row"});f.createDiv({cls:"af-form-label",text:"Agent"});let g=f.createEl("select",{cls:"af-form-select"});for(let q of n.agents)g.createEl("option",{text:q.name,attr:{value:q.name}});g.addEventListener("change",()=>{h.agent=g.value});let y=u.createDiv({cls:"af-form-row"});y.createDiv({cls:"af-form-label",text:"Priority"});let v=y.createEl("select",{cls:"af-form-select"}),k=[["low","Low"],["medium","Medium"],["high","High"],["critical","Critical"]];for(let[q,ie]of k){let j=v.createEl("option",{text:ie,attr:{value:q}});q==="medium"&&(j.selected=!0)}v.addEventListener("change",()=>{h.priority=v.value}),this.createFormField(u,"Tags","monitoring, devops","Comma-separated",q=>{h.tags=q});let w=d.createDiv({cls:"af-create-section"}),S=w.createDiv({cls:"af-create-section-header"}),T=S.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(T,"message-square"),S.createSpan({text:"Instructions"});let _=w.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Describe what the agent should do...",rows:"10"}});_.addEventListener("input",()=>{h.body=_.value});let D=d.createDiv({cls:"af-create-section"}),O=D.createDiv({cls:"af-create-section-header"}),C=O.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(C,"clock"),O.createSpan({text:"Schedule"});let E=D.createDiv({cls:"af-form-row af-form-row-toggle"});E.createDiv({cls:"af-form-label",text:"Enable schedule"});let P=E.createDiv({cls:"af-agent-card-toggle"}),N=D.createDiv({cls:"af-schedule-body"});N.setCssStyles({display:"none"}),P.onclick=()=>{let q=P.hasClass("on");P.toggleClass("on",!q),h.scheduleEnabled=!q,N.setCssStyles({display:q?"none":""}),q?h.type="immediate":h.type=h.scheduleMode==="once"?"once":"recurring"};let M=N.createDiv({cls:"af-form-row"});M.createDiv({cls:"af-form-label",text:"Mode"});let U=M.createEl("select",{cls:"af-form-select"});for(let[q,ie]of[["recurring","Recurring"],["once","One-time"]])U.createEl("option",{text:ie,attr:{value:q}});let $=N.createDiv(),Z=N.createDiv();Z.setCssStyles({display:"none"}),this.renderInlineSchedule($,h);let de=Z.createDiv({cls:"af-form-row"});de.createDiv({cls:"af-form-label",text:"Run at"});let me=de.createEl("input",{cls:"af-form-input",attr:{type:"datetime-local",value:this.toDatetimeLocal(new Date(Date.now()+36e5))}});h.runAt=new Date(me.value).toISOString(),me.addEventListener("input",()=>{h.runAt=me.value?new Date(me.value).toISOString():""}),U.addEventListener("change",()=>{h.scheduleMode=U.value,$.setCssStyles({display:h.scheduleMode==="recurring"?"":"none"}),Z.setCssStyles({display:h.scheduleMode==="once"?"":"none"}),h.scheduleEnabled&&(h.type=h.scheduleMode==="once"?"once":"recurring")});let ge=N.createDiv({cls:"af-form-row af-form-row-toggle"});ge.createDiv({cls:"af-form-label",text:"Enabled"});let X=ge.createDiv({cls:"af-agent-card-toggle on"});X.onclick=()=>{let q=X.hasClass("on");X.toggleClass("on",!q),h.enabled=!q};let W=N.createDiv({cls:"af-form-row af-form-row-toggle"});W.createDiv({cls:"af-form-label"}).setText("Catch up if missed");let ne=W.createDiv({cls:`af-agent-card-toggle${h.catchUp?" on":""}`});ne.onclick=()=>{let q=ne.hasClass("on");ne.toggleClass("on",!q),h.catchUp=!q};let G=d.createDiv({cls:"af-create-section"}),ee=G.createDiv({cls:"af-create-section-header"}),Q=ee.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Q,"gauge"),ee.createSpan({text:"Execution"});let ae=G.createDiv({cls:"af-form-row"}),ve=ae.createDiv({cls:"af-form-label",text:"Model"}),Pe=ae.createDiv({cls:"af-form-field-wrap"}),Ie=q=>{Pe.empty();let ie=n.agents.find(j=>j.name===q);Pt(Pe,{value:h.model,adapter:ie?.adapter,onChange:j=>{h.model=j},allowInherit:!0,inheritPlaceholder:ie?`Inherit from ${ie.name}${ie.model?` (${ie.model})`:""}`:"Inherit from agent"})};Ie(h.agent),g.addEventListener("change",()=>Ie(g.value)),this.addTooltip(ve,"Override the agent\u2019s model for this task only. Useful for routing simple runs to haiku while the agent stays on opus for heavier work.");let Me=G.createDiv({cls:"af-form-row"}),Fe=Me.createDiv({cls:"af-form-label",text:"Effort"}),ke=Me.createEl("select",{cls:"af-form-select"});for(let[q,ie]of[["","Agent Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]]){let j=ke.createEl("option",{text:ie,attr:{value:q}});q===h.effort&&(j.selected=!0)}ke.addEventListener("change",()=>{h.effort=ke.value}),this.renderTaskChannelDelivery(G,n.channels,h),this.addTooltip(Fe,"Overrides the agent\u2019s effort level for this task. Higher effort = more thinking tokens spent.");let Oe=s.createDiv({cls:"af-create-footer"}),Ue=Oe.createEl("button",{cls:"af-btn-sm",text:"Cancel"});Ue.onclick=()=>this.navigate("kanban");let Ne=Oe.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(Ne,"plus","af-btn-icon"),Ne.appendText(" Create Task"),Ne.onclick=async()=>{let q=h.title.trim();if(!q){new b.Notice("Task title is required.");return}let ie=oe(q),j=he=>he.split(",").map($e=>$e.trim()).filter(Boolean),Ce=h.scheduleEnabled?h.scheduleMode==="once"?"once":"recurring":"immediate",xe={task_id:ie,agent:h.agent,type:Ce,priority:h.priority,enabled:h.enabled,created:this.toLocalISO(new Date),run_count:0,catch_up:h.catchUp,effort:h.effort||void 0,model:h.model||void 0,channel:h.channel||void 0,channel_target:h.channel&&h.channelTarget?h.channelTarget:void 0,tags:j(h.tags)};if(Ce==="recurring")xe.schedule=h.schedule.trim()||"0 9 * * *";else if(Ce==="once"){if(!h.runAt){new b.Notice("Pick a date/time for the one-time run.");return}xe.run_at=h.runAt}try{let he=await this.plugin.repository.getAvailablePath(this.plugin.repository.getSubfolder("tasks"),ie);await this.plugin.app.vault.create(he,H(xe,h.body.trim()||"Describe the task here.")),new b.Notice(`Task "${ie}" created.`),await this.plugin.refreshFromVault(),this.navigate("task-detail",ie)}catch(he){let $e=he instanceof Error?he.message:String(he);new b.Notice(`Failed to create task: ${$e}`)}}}toLocalISO(e){let s=n=>String(n).padStart(2,"0");return`${e.getFullYear()}-${s(e.getMonth()+1)}-${s(e.getDate())}T${s(e.getHours())}:${s(e.getMinutes())}:${s(e.getSeconds())}`}toDatetimeLocal(e){let s=n=>String(n).padStart(2,"0");return`${e.getFullYear()}-${s(e.getMonth()+1)}-${s(e.getDate())}T${s(e.getHours())}:${s(e.getMinutes())}`}renderEditTaskPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"circle-dot","No task selected","");return}let a=this.plugin.runtime.getSnapshot().tasks.find(I=>I.taskId===n);if(!a){this.renderEmptyState(s,"circle-dot","Task not found",`Task "${n}" was not found`);return}let r=this.plugin.runtime.getSnapshot(),o=s.createDiv({cls:"af-detail-header"}),c=o.createDiv({cls:"af-detail-header-left"}),l=c.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(l,"edit");let h=c.createDiv();h.createDiv({cls:"af-detail-header-name",text:`Edit Task: ${a.taskId}`}),h.createDiv({cls:"af-detail-header-desc",text:"Modify task configuration"});let d=o.createDiv({cls:"af-detail-header-actions"}),u=!!(a.schedule||a.runAt),p={agent:a.agent,type:a.type,priority:a.priority,schedule:a.schedule??"0 9 * * *",runAt:a.runAt??"",scheduleEnabled:u,scheduleMode:a.type==="once"?"once":"recurring",enabled:a.enabled,catchUp:a.catchUp,effort:a.effort??"",model:a.model??"",channel:a.channel??"",channelTarget:a.channelTarget??"",tags:a.tags.join(", "),body:a.body},m=s.createDiv({cls:"af-create-form"}),f=m.createDiv({cls:"af-create-section"}),g=f.createDiv({cls:"af-create-section-header"}),y=g.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(y,"file-text"),g.createSpan({text:"Task Details"});let v=f.createDiv({cls:"af-form-row"});v.createDiv({cls:"af-form-label",text:"Title"}),v.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.taskId,disabled:"true"}}).setCssStyles({opacity:"0.6"});let w=f.createDiv({cls:"af-form-row"});w.createDiv({cls:"af-form-label",text:"Agent"});let S=w.createEl("select",{cls:"af-form-select"});for(let I of r.agents){let se=S.createEl("option",{text:I.name,attr:{value:I.name}});I.name===a.agent&&(se.selected=!0)}S.addEventListener("change",()=>{p.agent=S.value});let T=f.createDiv({cls:"af-form-row"});T.createDiv({cls:"af-form-label",text:"Priority"});let _=T.createEl("select",{cls:"af-form-select"}),D=[["low","Low"],["medium","Medium"],["high","High"],["critical","Critical"]];for(let[I,se]of D){let ce=_.createEl("option",{text:se,attr:{value:I}});I===a.priority&&(ce.selected=!0)}_.addEventListener("change",()=>{p.priority=_.value}),this.createFormField(f,"Tags","monitoring, critical","Comma-separated",I=>{p.tags=I},a.tags.join(", "));let O=m.createDiv({cls:"af-create-section"}),C=O.createDiv({cls:"af-create-section-header"}),E=C.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(E,"message-square"),C.createSpan({text:"Instructions"});let P=O.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Describe what the agent should do...",rows:"10"}});P.value=a.body,P.addEventListener("input",()=>{p.body=P.value});let N=m.createDiv({cls:"af-create-section"}),M=N.createDiv({cls:"af-create-section-header"}),U=M.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(U,"clock"),M.createSpan({text:"Schedule"});let $=N.createDiv({cls:"af-form-row af-form-row-toggle"});$.createDiv({cls:"af-form-label",text:"Enable schedule"});let Z=$.createDiv({cls:`af-agent-card-toggle${u?" on":""}`}),de=N.createDiv({cls:"af-schedule-body"});de.setCssStyles({display:u?"":"none"}),Z.onclick=()=>{let I=Z.hasClass("on");Z.toggleClass("on",!I),p.scheduleEnabled=!I,de.setCssStyles({display:I?"none":""}),I?p.type="immediate":p.type=p.scheduleMode==="once"?"once":"recurring"};let me=de.createDiv({cls:"af-form-row"});me.createDiv({cls:"af-form-label",text:"Mode"});let ge=me.createEl("select",{cls:"af-form-select"});for(let[I,se]of[["recurring","Recurring"],["once","One-time"]]){let ce=ge.createEl("option",{text:se,attr:{value:I}});I===p.scheduleMode&&(ce.selected=!0)}let X=de.createDiv(),W=de.createDiv();X.setCssStyles({display:p.scheduleMode==="recurring"?"":"none"}),W.setCssStyles({display:p.scheduleMode==="once"?"":"none"}),this.renderInlineSchedule(X,p);let K=W.createDiv({cls:"af-form-row"});K.createDiv({cls:"af-form-label",text:"Run at"});let ne=p.runAt?this.toDatetimeLocal(new Date(p.runAt)):this.toDatetimeLocal(new Date(Date.now()+36e5)),G=K.createEl("input",{cls:"af-form-input",attr:{type:"datetime-local",value:ne}});p.runAt||(p.runAt=new Date(G.value).toISOString()),G.addEventListener("input",()=>{p.runAt=G.value?new Date(G.value).toISOString():""}),ge.addEventListener("change",()=>{p.scheduleMode=ge.value,X.setCssStyles({display:p.scheduleMode==="recurring"?"":"none"}),W.setCssStyles({display:p.scheduleMode==="once"?"":"none"}),p.scheduleEnabled&&(p.type=p.scheduleMode==="once"?"once":"recurring")});let ee=de.createDiv({cls:"af-form-row af-form-row-toggle"});ee.createDiv({cls:"af-form-label",text:"Enabled"});let Q=ee.createDiv({cls:`af-agent-card-toggle${a.enabled?" on":""}`});Q.onclick=()=>{let I=Q.hasClass("on");Q.toggleClass("on",!I),p.enabled=!I};let ae=de.createDiv({cls:"af-form-row af-form-row-toggle"});ae.createDiv({cls:"af-form-label"}).setText("Catch up if missed");let Pe=ae.createDiv({cls:`af-agent-card-toggle${p.catchUp?" on":""}`});Pe.onclick=()=>{let I=Pe.hasClass("on");Pe.toggleClass("on",!I),p.catchUp=!I};let Ie=m.createDiv({cls:"af-create-section"}),Me=Ie.createDiv({cls:"af-create-section-header"}),Fe=Me.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Fe,"gauge"),Me.createSpan({text:"Execution"});let ke=Ie.createDiv({cls:"af-form-row"}),Oe=ke.createDiv({cls:"af-form-label",text:"Effort"}),Ue=ke.createEl("select",{cls:"af-form-select"}),Ne=[["","Agent Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]];for(let[I,se]of Ne){let ce=Ue.createEl("option",{text:se,attr:{value:I}});I===p.effort&&(ce.selected=!0)}Ue.addEventListener("change",()=>{p.effort=Ue.value}),this.addTooltip(Oe,"Overrides the agent\u2019s effort level for this task. Higher effort = more thinking tokens spent.");let q=Ie.createDiv({cls:"af-form-row"}),ie=q.createDiv({cls:"af-form-label",text:"Model"}),j=q.createDiv({cls:"af-form-field-wrap"}),Ce=I=>{j.empty();let se=r.agents.find(ce=>ce.name===I);Pt(j,{value:p.model,adapter:se?.adapter,onChange:ce=>{p.model=ce},allowInherit:!0,inheritPlaceholder:se?`Inherit from ${se.name}${se.model?` (${se.model})`:""}`:"Inherit from agent"})};Ce(p.agent),S.addEventListener("change",()=>Ce(S.value)),this.renderTaskChannelDelivery(Ie,r.channels,p),this.addTooltip(ie,"Override the agent\u2019s model for this task only. Useful for routing simple runs to haiku while the agent stays on opus for heavier work.");let xe=s.createDiv({cls:"af-create-footer"}),he=xe.createEl("button",{cls:"af-btn-sm danger"});R(he,"trash-2","af-btn-icon"),he.appendText(" Delete"),he.onclick=async()=>{await this.plugin.repository.deleteTask(a.taskId),new b.Notice(`Task "${a.taskId}" deleted.`),await new Promise(I=>window.setTimeout(I,200)),await this.plugin.refreshFromVault(),this.navigate("kanban")},xe.createDiv({cls:"af-toolbar-spacer"});let $e=xe.createEl("button",{cls:"af-btn-sm",text:"Cancel"});$e.onclick=()=>this.navigate("task-detail",a.taskId);let re=xe.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(re,"check","af-btn-icon"),re.appendText(" Save Changes"),re.onclick=async()=>{let I=ce=>ce.split(",").map(te=>te.trim()).filter(Boolean),se=p.scheduleEnabled?p.scheduleMode==="once"?"once":"recurring":"immediate";if(se==="once"&&!p.runAt){new b.Notice("Pick a date/time for the one-time run.");return}try{await this.plugin.repository.updateTask(a.taskId,{agent:p.agent,type:se,priority:p.priority,schedule:se==="recurring"?p.schedule.trim():"",runAt:se==="once"?p.runAt:"",enabled:p.enabled,catch_up:p.catchUp,effort:p.effort||void 0,model:p.model||"",channel:p.channel||"",channelTarget:p.channel&&p.channelTarget?p.channelTarget:"",tags:I(p.tags),body:p.body.trim()}),new b.Notice(`Task "${a.taskId}" updated.`),await this.plugin.refreshFromVault(),this.navigate("task-detail",a.taskId)}catch(ce){let te=ce instanceof Error?ce.message:String(ce);new b.Notice(`Failed to update task: ${te}`)}}}renderEditSkillPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"puzzle","No skill selected","");return}let a=this.plugin.runtime.getSnapshot().skills.find(G=>G.name===n);if(!a){this.renderEmptyState(s,"puzzle","Skill not found",`Skill "${n}" was not found`);return}let r=s.createDiv({cls:"af-detail-header"}),o=r.createDiv({cls:"af-detail-header-left"}),c=o.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(c,"edit");let l=o.createDiv();l.createDiv({cls:"af-detail-header-name",text:`Edit Skill: ${a.name}`}),l.createDiv({cls:"af-detail-header-desc",text:"Modify skill definition"});let h=r.createDiv({cls:"af-detail-header-actions"}),d={description:a.description??"",tags:a.tags.join(", "),body:a.body,toolsBody:a.toolsBody,referencesBody:a.referencesBody,examplesBody:a.examplesBody},u=s.createDiv({cls:"af-create-form"}),p=u.createDiv({cls:"af-create-section"}),m=p.createDiv({cls:"af-create-section-header"}),f=m.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(f,"puzzle"),m.createSpan({text:"Identity"});let g=p.createDiv({cls:"af-form-row"});g.createDiv({cls:"af-form-label",text:"Name"}),g.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.name,disabled:"true"}}).setCssStyles({opacity:"0.6"}),this.createFormField(p,"Description","Manage tasks and projects via CLI","",G=>{d.description=G},a.description??""),this.createFormField(p,"Tags","productivity, tasks","Comma-separated",G=>{d.tags=G},a.tags.join(", "));let v=u.createDiv({cls:"af-create-section"}),k=v.createDiv({cls:"af-create-section-header"}),w=k.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(w,"file-text"),k.createSpan({text:"Core Instructions"});let S=v.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Skill instructions...",rows:"10"}});S.value=a.body,S.addEventListener("input",()=>{d.body=S.value});let T=u.createDiv({cls:"af-create-section"}),_=T.createDiv({cls:"af-create-section-header"}),D=_.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(D,"wrench");let O=_.createSpan({text:"Tools"});this.addTooltip(O,"CLI commands, API endpoints, and tool definitions available to agents using this skill");let C=T.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`## Commands
12031
+ `),B.addEventListener("input",()=>{d.blockedCommands=B.value}),te.createDiv({cls:"af-form-hint",text:"On Codex agents these become execpolicy command rules \u2014 only Bash(cmd args *) prefixes are enforced; tool-name rules (Read/Write) and mid-pattern wildcards are ignored, and file/network access is governed by Permission Mode (the sandbox)."});let V=te.createDiv({cls:"af-form-row"});V.createDiv({cls:"af-form-label",text:"Memory enabled"});let Ee=V.createDiv({cls:`af-agent-card-toggle${a.memory?" on":""}`});Ee.onclick=()=>{let L=Ee.hasClass("on");Ee.toggleClass("on",!L),d.memory=!L};let we=te.createDiv({cls:"af-form-row"});we.createDiv({cls:"af-form-label",text:"Memory token budget"});let Ae=we.createEl("input",{cls:"af-create-input",attr:{type:"number",min:"200",step:"100"}});Ae.value=String(d.memoryTokenBudget),Ae.addEventListener("input",()=>{let L=parseInt(Ae.value,10);Number.isFinite(L)&&(d.memoryTokenBudget=L)});let nt=te.createDiv({cls:"af-form-row"});nt.createDiv({cls:"af-form-label",text:"Nightly reflection"});let je=nt.createDiv({cls:`af-agent-card-toggle${a.reflection.enabled?" on":""}`});je.onclick=()=>{let L=je.hasClass("on");je.toggleClass("on",!L),d.reflectionEnabled=!L};let Tt=te.createDiv({cls:"af-form-row"});Tt.createDiv({cls:"af-form-label",text:"Propose skills from memory"});let Ut=Tt.createDiv({cls:`af-agent-card-toggle${a.reflection.proposeSkills?" on":""}`});Ut.onclick=()=>{let L=Ut.hasClass("on");Ut.toggleClass("on",!L),d.reflectionProposeSkills=!L};let bt=s.createDiv({cls:"af-create-footer"}),hs=bt.createEl("button",{cls:"af-btn-sm danger"});R(hs,"trash-2","af-btn-icon"),hs.appendText(" Delete"),hs.onclick=()=>void this.plugin.deleteAgent(a.name),bt.createDiv({cls:"af-toolbar-spacer"});let Ns=bt.createEl("button",{cls:"af-btn-sm",text:"Cancel"});Ns.onclick=()=>this.navigate("agent-detail",a.name);let $t=bt.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R($t,"check","af-btn-icon"),$t.appendText(" Save Changes"),$t.onclick=async()=>{let L=Y=>Y.split(",").map(Te=>Te.trim()).filter(Boolean);try{let Y=Te=>ye(Te).map(Le=>Le.trim()).filter(Boolean);await this.plugin.repository.updateAgent(a.name,{description:d.description.trim(),avatar:d.avatar.trim(),tags:L(d.tags),systemPrompt:d.systemPrompt.trim(),model:d.model.trim()||"default",adapter:d.adapter,cwd:d.cwd.trim(),timeout:d.timeout,permissionMode:d.permissionMode,effort:d.effort||void 0,approvalRequired:L(d.approvalRequired),memory:d.memory,memoryTokenBudget:d.memoryTokenBudget,reflectionEnabled:d.reflectionEnabled,reflectionProposeSkills:d.reflectionProposeSkills,skills:Array.from(d.selectedSkills),mcpServers:Array.from(d.selectedMcpServers),skillsBody:d.skillsBody.trim(),contextBody:d.contextBody.trim(),enabled:d.enabled,permissionRules:{allow:Y(d.allowedCommands),deny:Y(d.blockedCommands)},autoCompactThreshold:d.autoCompactThreshold,wikiReferences:d.wikiReferences}),a.isFolder&&await this.plugin.repository.updateHeartbeat(a.name,{enabled:d.heartbeatEnabled,schedule:d.heartbeatSchedule.trim(),notify:d.heartbeatNotify,channel:d.heartbeatChannel,channelTarget:d.heartbeatChannel?d.heartbeatChannelTarget:"",body:d.heartbeatBody.trim()}),new b.Notice(`Agent "${a.name}" updated.`),await this.plugin.refreshFromVault(),this.navigate("agent-detail",a.name)}catch(Y){let Te=Y instanceof Error?Y.message:String(Y);new b.Notice(`Failed to update agent: ${Te}`)}}}renderTaskChannelDelivery(e,s,n){let a=e.createDiv({cls:"af-form-row"}),i=a.createDiv({cls:"af-form-label",text:"Channel"}),o=a.createEl("select",{cls:"af-form-select"});o.createEl("option",{text:"\u2014 none (run log only) \u2014",attr:{value:""}});for(let u of s){let p=o.createEl("option",{text:u.name,attr:{value:u.name}});u.name===n.channel&&(p.selected=!0)}this.addTooltip(i,"Post this task\u2019s full output to a channel when it finishes. Leave as none to only write the run log (the default batched behavior).");let c=e.createDiv({cls:"af-form-row"}),l=c.createDiv({cls:"af-form-label",text:"Target ID"}),h=c.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:"Discord/Slack channel ID or Telegram chat ID \u2014 empty = DM you"}});h.value=n.channelTarget,h.addEventListener("input",()=>{n.channelTarget=h.value.trim()}),this.addTooltip(l,"Where in the channel to post. For Discord, enable Developer Mode and right-click the channel \u2192 Copy Channel ID. Leave empty to DM the first allowed user instead.");let d=()=>{c.setCssStyles({display:n.channel?"":"none"})};d(),o.addEventListener("change",()=>{n.channel=o.value,d()})}renderCreateTaskPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.plugin.runtime.getSnapshot(),a=s.createDiv({cls:"af-detail-header"}),i=a.createDiv({cls:"af-detail-header-left"}),o=i.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(o,"plus");let c=i.createDiv();c.createDiv({cls:"af-detail-header-name",text:"Create New Task"}),c.createDiv({cls:"af-detail-header-desc",text:"Configure a new task for your fleet"});let l=a.createDiv({cls:"af-detail-header-actions"}),h={title:"",agent:n.agents[0]?.name??"",priority:"medium",tags:"",body:"",scheduleEnabled:!1,scheduleMode:"recurring",schedule:"0 9 * * *",runAt:"",type:"immediate",enabled:!0,catchUp:!0,effort:"",model:"",channel:"",channelTarget:""},d=s.createDiv({cls:"af-create-form"}),u=d.createDiv({cls:"af-create-section"}),p=u.createDiv({cls:"af-create-section-header"}),m=p.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(m,"file-text"),p.createSpan({text:"Task Details"}),this.createFormField(u,"Title","Daily status report","Used as the task identifier",q=>{h.title=q});let f=u.createDiv({cls:"af-form-row"});f.createDiv({cls:"af-form-label",text:"Agent"});let g=f.createEl("select",{cls:"af-form-select"});for(let q of n.agents)g.createEl("option",{text:q.name,attr:{value:q.name}});g.addEventListener("change",()=>{h.agent=g.value});let y=u.createDiv({cls:"af-form-row"});y.createDiv({cls:"af-form-label",text:"Priority"});let v=y.createEl("select",{cls:"af-form-select"}),k=[["low","Low"],["medium","Medium"],["high","High"],["critical","Critical"]];for(let[q,re]of k){let j=v.createEl("option",{text:re,attr:{value:q}});q==="medium"&&(j.selected=!0)}v.addEventListener("change",()=>{h.priority=v.value}),this.createFormField(u,"Tags","monitoring, devops","Comma-separated",q=>{h.tags=q});let w=d.createDiv({cls:"af-create-section"}),S=w.createDiv({cls:"af-create-section-header"}),T=S.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(T,"message-square"),S.createSpan({text:"Instructions"});let _=w.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Describe what the agent should do...",rows:"10"}});_.addEventListener("input",()=>{h.body=_.value});let D=d.createDiv({cls:"af-create-section"}),O=D.createDiv({cls:"af-create-section-header"}),C=O.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(C,"clock"),O.createSpan({text:"Schedule"});let E=D.createDiv({cls:"af-form-row af-form-row-toggle"});E.createDiv({cls:"af-form-label",text:"Enable schedule"});let P=E.createDiv({cls:"af-agent-card-toggle"}),N=D.createDiv({cls:"af-schedule-body"});N.setCssStyles({display:"none"}),P.onclick=()=>{let q=P.hasClass("on");P.toggleClass("on",!q),h.scheduleEnabled=!q,N.setCssStyles({display:q?"none":""}),q?h.type="immediate":h.type=h.scheduleMode==="once"?"once":"recurring"};let M=N.createDiv({cls:"af-form-row"});M.createDiv({cls:"af-form-label",text:"Mode"});let U=M.createEl("select",{cls:"af-form-select"});for(let[q,re]of[["recurring","Recurring"],["once","One-time"]])U.createEl("option",{text:re,attr:{value:q}});let $=N.createDiv(),Z=N.createDiv();Z.setCssStyles({display:"none"}),this.renderInlineSchedule($,h);let de=Z.createDiv({cls:"af-form-row"});de.createDiv({cls:"af-form-label",text:"Run at"});let me=de.createEl("input",{cls:"af-form-input",attr:{type:"datetime-local",value:this.toDatetimeLocal(new Date(Date.now()+36e5))}});h.runAt=new Date(me.value).toISOString(),me.addEventListener("input",()=>{h.runAt=me.value?new Date(me.value).toISOString():""}),U.addEventListener("change",()=>{h.scheduleMode=U.value,$.setCssStyles({display:h.scheduleMode==="recurring"?"":"none"}),Z.setCssStyles({display:h.scheduleMode==="once"?"":"none"}),h.scheduleEnabled&&(h.type=h.scheduleMode==="once"?"once":"recurring")});let ge=N.createDiv({cls:"af-form-row af-form-row-toggle"});ge.createDiv({cls:"af-form-label",text:"Enabled"});let X=ge.createDiv({cls:"af-agent-card-toggle on"});X.onclick=()=>{let q=X.hasClass("on");X.toggleClass("on",!q),h.enabled=!q};let W=N.createDiv({cls:"af-form-row af-form-row-toggle"});W.createDiv({cls:"af-form-label"}).setText("Catch up if missed");let ne=W.createDiv({cls:`af-agent-card-toggle${h.catchUp?" on":""}`});ne.onclick=()=>{let q=ne.hasClass("on");ne.toggleClass("on",!q),h.catchUp=!q};let G=d.createDiv({cls:"af-create-section"}),ee=G.createDiv({cls:"af-create-section-header"}),Q=ee.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Q,"gauge"),ee.createSpan({text:"Execution"});let ae=G.createDiv({cls:"af-form-row"}),ve=ae.createDiv({cls:"af-form-label",text:"Model"}),Pe=ae.createDiv({cls:"af-form-field-wrap"}),Ie=q=>{Pe.empty();let re=n.agents.find(j=>j.name===q);Rt(Pe,{value:h.model,adapter:re?.adapter,onChange:j=>{h.model=j},allowInherit:!0,inheritPlaceholder:re?`Inherit from ${re.name}${re.model?` (${re.model})`:""}`:"Inherit from agent"})};Ie(h.agent),g.addEventListener("change",()=>Ie(g.value)),this.addTooltip(ve,"Override the agent\u2019s model for this task only. Useful for routing simple runs to haiku while the agent stays on opus for heavier work.");let Me=G.createDiv({cls:"af-form-row"}),Fe=Me.createDiv({cls:"af-form-label",text:"Effort"}),ke=Me.createEl("select",{cls:"af-form-select"});for(let[q,re]of[["","Agent Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]]){let j=ke.createEl("option",{text:re,attr:{value:q}});q===h.effort&&(j.selected=!0)}ke.addEventListener("change",()=>{h.effort=ke.value}),this.renderTaskChannelDelivery(G,n.channels,h),this.addTooltip(Fe,"Overrides the agent\u2019s effort level for this task. Higher effort = more thinking tokens spent.");let Oe=s.createDiv({cls:"af-create-footer"}),Ue=Oe.createEl("button",{cls:"af-btn-sm",text:"Cancel"});Ue.onclick=()=>this.navigate("kanban");let Ne=Oe.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(Ne,"plus","af-btn-icon"),Ne.appendText(" Create Task"),Ne.onclick=async()=>{let q=h.title.trim();if(!q){new b.Notice("Task title is required.");return}let re=oe(q),j=he=>he.split(",").map($e=>$e.trim()).filter(Boolean),Ce=h.scheduleEnabled?h.scheduleMode==="once"?"once":"recurring":"immediate",xe={task_id:re,agent:h.agent,type:Ce,priority:h.priority,enabled:h.enabled,created:this.toLocalISO(new Date),run_count:0,catch_up:h.catchUp,effort:h.effort||void 0,model:h.model||void 0,channel:h.channel||void 0,channel_target:h.channel&&h.channelTarget?h.channelTarget:void 0,tags:j(h.tags)};if(Ce==="recurring")xe.schedule=h.schedule.trim()||"0 9 * * *";else if(Ce==="once"){if(!h.runAt){new b.Notice("Pick a date/time for the one-time run.");return}xe.run_at=h.runAt}try{let he=await this.plugin.repository.getAvailablePath(this.plugin.repository.getSubfolder("tasks"),re);await this.plugin.app.vault.create(he,H(xe,h.body.trim()||"Describe the task here.")),new b.Notice(`Task "${re}" created.`),await this.plugin.refreshFromVault(),this.navigate("task-detail",re)}catch(he){let $e=he instanceof Error?he.message:String(he);new b.Notice(`Failed to create task: ${$e}`)}}}toLocalISO(e){let s=n=>String(n).padStart(2,"0");return`${e.getFullYear()}-${s(e.getMonth()+1)}-${s(e.getDate())}T${s(e.getHours())}:${s(e.getMinutes())}:${s(e.getSeconds())}`}toDatetimeLocal(e){let s=n=>String(n).padStart(2,"0");return`${e.getFullYear()}-${s(e.getMonth()+1)}-${s(e.getDate())}T${s(e.getHours())}:${s(e.getMinutes())}`}renderEditTaskPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"circle-dot","No task selected","");return}let a=this.plugin.runtime.getSnapshot().tasks.find(I=>I.taskId===n);if(!a){this.renderEmptyState(s,"circle-dot","Task not found",`Task "${n}" was not found`);return}let i=this.plugin.runtime.getSnapshot(),o=s.createDiv({cls:"af-detail-header"}),c=o.createDiv({cls:"af-detail-header-left"}),l=c.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(l,"edit");let h=c.createDiv();h.createDiv({cls:"af-detail-header-name",text:`Edit Task: ${a.taskId}`}),h.createDiv({cls:"af-detail-header-desc",text:"Modify task configuration"});let d=o.createDiv({cls:"af-detail-header-actions"}),u=!!(a.schedule||a.runAt),p={agent:a.agent,type:a.type,priority:a.priority,schedule:a.schedule??"0 9 * * *",runAt:a.runAt??"",scheduleEnabled:u,scheduleMode:a.type==="once"?"once":"recurring",enabled:a.enabled,catchUp:a.catchUp,effort:a.effort??"",model:a.model??"",channel:a.channel??"",channelTarget:a.channelTarget??"",tags:a.tags.join(", "),body:a.body},m=s.createDiv({cls:"af-create-form"}),f=m.createDiv({cls:"af-create-section"}),g=f.createDiv({cls:"af-create-section-header"}),y=g.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(y,"file-text"),g.createSpan({text:"Task Details"});let v=f.createDiv({cls:"af-form-row"});v.createDiv({cls:"af-form-label",text:"Title"}),v.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.taskId,disabled:"true"}}).setCssStyles({opacity:"0.6"});let w=f.createDiv({cls:"af-form-row"});w.createDiv({cls:"af-form-label",text:"Agent"});let S=w.createEl("select",{cls:"af-form-select"});for(let I of i.agents){let se=S.createEl("option",{text:I.name,attr:{value:I.name}});I.name===a.agent&&(se.selected=!0)}S.addEventListener("change",()=>{p.agent=S.value});let T=f.createDiv({cls:"af-form-row"});T.createDiv({cls:"af-form-label",text:"Priority"});let _=T.createEl("select",{cls:"af-form-select"}),D=[["low","Low"],["medium","Medium"],["high","High"],["critical","Critical"]];for(let[I,se]of D){let ce=_.createEl("option",{text:se,attr:{value:I}});I===a.priority&&(ce.selected=!0)}_.addEventListener("change",()=>{p.priority=_.value}),this.createFormField(f,"Tags","monitoring, critical","Comma-separated",I=>{p.tags=I},a.tags.join(", "));let O=m.createDiv({cls:"af-create-section"}),C=O.createDiv({cls:"af-create-section-header"}),E=C.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(E,"message-square"),C.createSpan({text:"Instructions"});let P=O.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Describe what the agent should do...",rows:"10"}});P.value=a.body,P.addEventListener("input",()=>{p.body=P.value});let N=m.createDiv({cls:"af-create-section"}),M=N.createDiv({cls:"af-create-section-header"}),U=M.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(U,"clock"),M.createSpan({text:"Schedule"});let $=N.createDiv({cls:"af-form-row af-form-row-toggle"});$.createDiv({cls:"af-form-label",text:"Enable schedule"});let Z=$.createDiv({cls:`af-agent-card-toggle${u?" on":""}`}),de=N.createDiv({cls:"af-schedule-body"});de.setCssStyles({display:u?"":"none"}),Z.onclick=()=>{let I=Z.hasClass("on");Z.toggleClass("on",!I),p.scheduleEnabled=!I,de.setCssStyles({display:I?"none":""}),I?p.type="immediate":p.type=p.scheduleMode==="once"?"once":"recurring"};let me=de.createDiv({cls:"af-form-row"});me.createDiv({cls:"af-form-label",text:"Mode"});let ge=me.createEl("select",{cls:"af-form-select"});for(let[I,se]of[["recurring","Recurring"],["once","One-time"]]){let ce=ge.createEl("option",{text:se,attr:{value:I}});I===p.scheduleMode&&(ce.selected=!0)}let X=de.createDiv(),W=de.createDiv();X.setCssStyles({display:p.scheduleMode==="recurring"?"":"none"}),W.setCssStyles({display:p.scheduleMode==="once"?"":"none"}),this.renderInlineSchedule(X,p);let K=W.createDiv({cls:"af-form-row"});K.createDiv({cls:"af-form-label",text:"Run at"});let ne=p.runAt?this.toDatetimeLocal(new Date(p.runAt)):this.toDatetimeLocal(new Date(Date.now()+36e5)),G=K.createEl("input",{cls:"af-form-input",attr:{type:"datetime-local",value:ne}});p.runAt||(p.runAt=new Date(G.value).toISOString()),G.addEventListener("input",()=>{p.runAt=G.value?new Date(G.value).toISOString():""}),ge.addEventListener("change",()=>{p.scheduleMode=ge.value,X.setCssStyles({display:p.scheduleMode==="recurring"?"":"none"}),W.setCssStyles({display:p.scheduleMode==="once"?"":"none"}),p.scheduleEnabled&&(p.type=p.scheduleMode==="once"?"once":"recurring")});let ee=de.createDiv({cls:"af-form-row af-form-row-toggle"});ee.createDiv({cls:"af-form-label",text:"Enabled"});let Q=ee.createDiv({cls:`af-agent-card-toggle${a.enabled?" on":""}`});Q.onclick=()=>{let I=Q.hasClass("on");Q.toggleClass("on",!I),p.enabled=!I};let ae=de.createDiv({cls:"af-form-row af-form-row-toggle"});ae.createDiv({cls:"af-form-label"}).setText("Catch up if missed");let Pe=ae.createDiv({cls:`af-agent-card-toggle${p.catchUp?" on":""}`});Pe.onclick=()=>{let I=Pe.hasClass("on");Pe.toggleClass("on",!I),p.catchUp=!I};let Ie=m.createDiv({cls:"af-create-section"}),Me=Ie.createDiv({cls:"af-create-section-header"}),Fe=Me.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(Fe,"gauge"),Me.createSpan({text:"Execution"});let ke=Ie.createDiv({cls:"af-form-row"}),Oe=ke.createDiv({cls:"af-form-label",text:"Effort"}),Ue=ke.createEl("select",{cls:"af-form-select"}),Ne=[["","Agent Default"],["low","Low"],["medium","Medium"],["high","High"],["max","Max"]];for(let[I,se]of Ne){let ce=Ue.createEl("option",{text:se,attr:{value:I}});I===p.effort&&(ce.selected=!0)}Ue.addEventListener("change",()=>{p.effort=Ue.value}),this.addTooltip(Oe,"Overrides the agent\u2019s effort level for this task. Higher effort = more thinking tokens spent.");let q=Ie.createDiv({cls:"af-form-row"}),re=q.createDiv({cls:"af-form-label",text:"Model"}),j=q.createDiv({cls:"af-form-field-wrap"}),Ce=I=>{j.empty();let se=i.agents.find(ce=>ce.name===I);Rt(j,{value:p.model,adapter:se?.adapter,onChange:ce=>{p.model=ce},allowInherit:!0,inheritPlaceholder:se?`Inherit from ${se.name}${se.model?` (${se.model})`:""}`:"Inherit from agent"})};Ce(p.agent),S.addEventListener("change",()=>Ce(S.value)),this.renderTaskChannelDelivery(Ie,i.channels,p),this.addTooltip(re,"Override the agent\u2019s model for this task only. Useful for routing simple runs to haiku while the agent stays on opus for heavier work.");let xe=s.createDiv({cls:"af-create-footer"}),he=xe.createEl("button",{cls:"af-btn-sm danger"});R(he,"trash-2","af-btn-icon"),he.appendText(" Delete"),he.onclick=async()=>{await this.plugin.repository.deleteTask(a.taskId),new b.Notice(`Task "${a.taskId}" deleted.`),await new Promise(I=>window.setTimeout(I,200)),await this.plugin.refreshFromVault(),this.navigate("kanban")},xe.createDiv({cls:"af-toolbar-spacer"});let $e=xe.createEl("button",{cls:"af-btn-sm",text:"Cancel"});$e.onclick=()=>this.navigate("task-detail",a.taskId);let ie=xe.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(ie,"check","af-btn-icon"),ie.appendText(" Save Changes"),ie.onclick=async()=>{let I=ce=>ce.split(",").map(te=>te.trim()).filter(Boolean),se=p.scheduleEnabled?p.scheduleMode==="once"?"once":"recurring":"immediate";if(se==="once"&&!p.runAt){new b.Notice("Pick a date/time for the one-time run.");return}try{await this.plugin.repository.updateTask(a.taskId,{agent:p.agent,type:se,priority:p.priority,schedule:se==="recurring"?p.schedule.trim():"",runAt:se==="once"?p.runAt:"",enabled:p.enabled,catch_up:p.catchUp,effort:p.effort||void 0,model:p.model||"",channel:p.channel||"",channelTarget:p.channel&&p.channelTarget?p.channelTarget:"",tags:I(p.tags),body:p.body.trim()}),new b.Notice(`Task "${a.taskId}" updated.`),await this.plugin.refreshFromVault(),this.navigate("task-detail",a.taskId)}catch(ce){let te=ce instanceof Error?ce.message:String(ce);new b.Notice(`Failed to update task: ${te}`)}}}renderEditSkillPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=this.detailContext;if(!n){this.renderEmptyState(s,"puzzle","No skill selected","");return}let a=this.plugin.runtime.getSnapshot().skills.find(G=>G.name===n);if(!a){this.renderEmptyState(s,"puzzle","Skill not found",`Skill "${n}" was not found`);return}let i=s.createDiv({cls:"af-detail-header"}),o=i.createDiv({cls:"af-detail-header-left"}),c=o.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(c,"edit");let l=o.createDiv();l.createDiv({cls:"af-detail-header-name",text:`Edit Skill: ${a.name}`}),l.createDiv({cls:"af-detail-header-desc",text:"Modify skill definition"});let h=i.createDiv({cls:"af-detail-header-actions"}),d={description:a.description??"",tags:a.tags.join(", "),body:a.body,toolsBody:a.toolsBody,referencesBody:a.referencesBody,examplesBody:a.examplesBody},u=s.createDiv({cls:"af-create-form"}),p=u.createDiv({cls:"af-create-section"}),m=p.createDiv({cls:"af-create-section-header"}),f=m.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(f,"puzzle"),m.createSpan({text:"Identity"});let g=p.createDiv({cls:"af-form-row"});g.createDiv({cls:"af-form-label",text:"Name"}),g.createEl("input",{cls:"af-form-input",attr:{type:"text",value:a.name,disabled:"true"}}).setCssStyles({opacity:"0.6"}),this.createFormField(p,"Description","Manage tasks and projects via CLI","",G=>{d.description=G},a.description??""),this.createFormField(p,"Tags","productivity, tasks","Comma-separated",G=>{d.tags=G},a.tags.join(", "));let v=u.createDiv({cls:"af-create-section"}),k=v.createDiv({cls:"af-create-section-header"}),w=k.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(w,"file-text"),k.createSpan({text:"Core Instructions"});let S=v.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"Skill instructions...",rows:"10"}});S.value=a.body,S.addEventListener("input",()=>{d.body=S.value});let T=u.createDiv({cls:"af-create-section"}),_=T.createDiv({cls:"af-create-section-header"}),D=_.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(D,"wrench");let O=_.createSpan({text:"Tools"});this.addTooltip(O,"CLI commands, API endpoints, and tool definitions available to agents using this skill");let C=T.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`## Commands
12121
12032
 
12122
12033
  ### list
12123
12034
  ...`,rows:"8"}});C.value=a.toolsBody,C.addEventListener("input",()=>{d.toolsBody=C.value});let E=u.createDiv({cls:"af-create-section"}),P=E.createDiv({cls:"af-create-section-header"}),N=P.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(N,"book-open");let M=P.createSpan({text:"References"});this.addTooltip(M,"Background docs, conventions, cheat sheets");let U=E.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"API docs, filter syntax, conventions...",rows:"6"}});U.value=a.referencesBody,U.addEventListener("input",()=>{d.referencesBody=U.value});let $=u.createDiv({cls:"af-create-section"}),Z=$.createDiv({cls:"af-create-section-header"}),de=Z.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(de,"message-circle");let me=Z.createSpan({text:"Examples"});this.addTooltip(me,"Example prompts and ideal outputs showing how to use this skill");let ge=$.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`## Example: List all tasks
12124
- ...`,rows:"6"}});ge.value=a.examplesBody,ge.addEventListener("input",()=>{d.examplesBody=ge.value});let X=s.createDiv({cls:"af-create-footer"}),W=X.createEl("button",{cls:"af-btn-sm danger"});R(W,"trash-2","af-btn-icon"),W.appendText(" Delete"),W.onclick=async()=>{await this.plugin.repository.deleteSkill(a.name),new b.Notice(`Skill "${a.name}" deleted.`),await new Promise(G=>window.setTimeout(G,200)),await this.plugin.refreshFromVault(),this.navigate("skills")},X.createDiv({cls:"af-toolbar-spacer"});let K=X.createEl("button",{cls:"af-btn-sm",text:"Cancel"});K.onclick=()=>this.navigate("skills");let ne=X.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(ne,"check","af-btn-icon"),ne.appendText(" Save Changes"),ne.onclick=async()=>{let G=ee=>ee.split(",").map(Q=>Q.trim()).filter(Boolean);try{await this.plugin.repository.updateSkill(a.name,{description:d.description.trim(),tags:G(d.tags),body:d.body.trim(),toolsBody:d.toolsBody.trim(),referencesBody:d.referencesBody.trim(),examplesBody:d.examplesBody.trim()}),new b.Notice(`Skill "${a.name}" updated.`),await this.plugin.refreshFromVault(),this.navigate("skills")}catch(ee){let Q=ee instanceof Error?ee.message:String(ee);new b.Notice(`Failed to update skill: ${Q}`)}}}renderAgentMcpPicker(e,s){e.createDiv({cls:"af-form-hint",text:"Servers from the MCP Servers tab. Checked servers are available to this agent on any adapter (Claude or Codex). Leave all unchecked to grant every enabled server."});let n=this.plugin.repository.getMcpServers();if(n.length===0){let r=e.createDiv({cls:"af-form-hint"});r.appendText("No MCP servers registered yet. ");let o=r.createEl("a",{cls:"af-link",text:"Add one in the MCP Servers tab."});o.onclick=c=>{c.preventDefault(),this.navigate("mcp")};return}let a=e.createDiv({cls:"af-create-skills-grid"});for(let r of n){let o=a.createDiv({cls:"af-mcp-agent-item"}),c=o.createEl("input",{cls:"af-form-toggle",attr:{type:"checkbox"}});c.checked=s.has(r.name),c.addEventListener("change",()=>{c.checked?s.add(r.name):s.delete(r.name)});let h=o.createDiv({cls:"af-mcp-agent-label"}).createDiv({cls:"af-mcp-agent-name-row"}),d=h.createSpan({cls:`af-mcp-status-dot ${r.enabled?"idle":"disabled"}`});d.title=r.enabled?"enabled":"disabled",h.createSpan({cls:"af-create-skill-name",text:r.name}),h.createSpan({cls:"af-mcp-agent-tool-count",text:r.type}),r.enabled||h.createSpan({cls:"af-mcp-agent-tool-count af-muted",text:"disabled"})}}renderMcpPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=s.createDiv({cls:"af-agents-toolbar"});n.createDiv({cls:"af-page-title",text:"MCP Servers"}),n.createDiv({cls:"af-toolbar-spacer"});let a=n.createEl("button",{cls:"af-btn-sm primary"});R(a,"plus","af-btn-icon"),a.appendText(" Add Server"),a.onclick=()=>this.navigate("add-mcp-server");let r=this.plugin.repository.getMcpServers();if(r.length===0){this.renderEmptyState(s,"plug","No MCP servers registered","Click 'Add Server' above to register one.");return}let o=s.createDiv({cls:"af-agents-grid"});for(let c of r)this.renderMcpCard(o,c)}mcpProbeCache=new Map;mcpHasToken(e){return this.plugin.mcpAuth.hasToken(e.name)}mcpNeedsAuth(e){return e.type!=="stdio"&&(e.auth==="oauth"||e.auth==="bearer")&&!this.mcpHasToken(e)}renderMcpCard(e,s){let n=e.createDiv({cls:`af-mcp-card${s.enabled?"":" af-mcp-card-disabled"}`}),a=this.mcpNeedsAuth(s),r=n.createDiv({cls:"af-agent-card-header"}),o=s.enabled?a?"pending":"idle":"disabled",c=r.createDiv({cls:`af-agent-card-avatar ${o}`});(0,b.setIcon)(c,"plug");let l=r.createDiv({cls:"af-agent-card-titleblock"});l.createDiv({cls:"af-agent-card-name",text:s.name});let h=l.createDiv({cls:"af-agent-card-desc af-mcp-meta"});h.createSpan({cls:"af-mcp-type-badge",text:s.type}),s.source==="imported"&&h.createSpan({cls:"af-badge",text:"imported"});let d=r.createDiv({cls:`af-agent-card-toggle${s.enabled?" on":""}`});d.onclick=g=>{g.stopPropagation(),this.plugin.repository.setMcpServerEnabled(s.name,!s.enabled).then(async()=>{await this.plugin.refreshFromVault(),this.render()})};let u=n.createDiv({cls:`af-mcp-status-badge ${s.enabled?a?"needs-auth":"connected":"disabled"}`}),p=u.createSpan();if(s.enabled?a?((0,b.setIcon)(p,"alert-circle"),u.createSpan({text:" Needs auth"})):((0,b.setIcon)(p,"check-circle"),u.createSpan({text:s.type==="stdio"?" Enabled":" Authenticated"})):((0,b.setIcon)(p,"pause"),u.createSpan({text:" Disabled"})),s.description){let g=this.truncateDescription(s.description,120);n.createDiv({cls:"af-mcp-description",text:g})}let m=s.url??s.command??"";m&&n.createDiv({cls:"af-mcp-command",text:jt(m,60)});let f=this.mcpProbeCache.get(s.name);if(f&&f.length>0){let g=n.createDiv({cls:"af-mcp-tool-footer"}),y=g.createDiv({cls:"af-mcp-tool-count"}),v=y.createSpan();(0,b.setIcon)(v,"wrench"),y.createSpan({text:` ${f.length} tools`});let k=g.createDiv({cls:"af-mcp-tool-chips"});for(let w of f.slice(0,4))k.createSpan({cls:"af-mcp-tool-chip",text:w.name});f.length>4&&k.createSpan({cls:"af-mcp-tool-chip af-mcp-tool-chip-more",text:`+${f.length-4}`})}if(this.authenticatingServers.has(s.name)){let y=n.createDiv({cls:"af-mcp-auth-row"}).createEl("button",{cls:"af-btn-sm primary",attr:{disabled:"true"}}),v=y.createSpan({cls:"af-spin"});(0,b.setIcon)(v,"loader-2"),y.appendText(" Authenticating\u2026")}else if(s.enabled&&a&&s.auth==="oauth"){let y=n.createDiv({cls:"af-mcp-auth-row"}).createEl("button",{cls:"af-btn-sm primary"}),v=y.createSpan();(0,b.setIcon)(v,"key"),y.appendText(" Authenticate"),y.onclick=k=>{k.stopPropagation(),this.authenticateMcpServer(s)}}n.onclick=()=>this.openMcpDetailSlideover(s)}async authenticateMcpServer(e){if(!e.url){new b.Notice("No URL found for this server \u2014 can't authenticate.");return}this.authenticatingServers.add(e.name),this.render(),new b.Notice(`Authenticating ${e.name}\u2026 Complete authorization in your browser.`,1e4);try{let s=e.type==="sse"?"sse":"http";await this.plugin.mcpManager.authenticateServer(e.name,e.url,s),new b.Notice(`${e.name} authenticated successfully!`)}catch(s){let n=s instanceof Error?s.message:String(s);new b.Notice(`Authentication failed: ${n}`,8e3)}finally{this.authenticatingServers.delete(e.name),this.render()}}async probeMcpServer(e){try{let s=await this.plugin.mcpManager.probeServer(e);this.mcpProbeCache.set(e.name,s),s.length===0&&new b.Notice(`No tools discovered for ${e.name}.`)}catch(s){let n=s instanceof Error?s.message:String(s);new b.Notice(`Probe failed: ${n}`)}}truncateDescription(e,s){let n=ye(e)[0]??e,a=n.split(/(?<=[.!?])\s/)[0]??n,r=a.length<n.length?a:n;return r.length<=s?r:r.slice(0,s-1)+"\u2026"}openMcpDetailSlideover(e){this.contentEl.querySelector(".af-slideover-overlay")?.remove();let s=this.contentEl.createDiv({cls:"af-slideover-overlay"}),n=s.createDiv({cls:"af-slideover"}),a=n.createDiv({cls:"af-slideover-header"});a.createDiv({cls:"af-slideover-title",text:e.name});let r=a.createEl("button",{cls:"clickable-icon"});(0,b.setIcon)(r,"cross"),r.onclick=()=>s.remove(),s.onclick=y=>{y.target===s&&s.remove()};let o=n.createDiv({cls:"af-slideover-body"});if(e.description){let y=o.createDiv({cls:"af-slideover-section"});y.createDiv({cls:"af-slideover-section-title",text:"DESCRIPTION"}),y.createDiv({cls:"af-mcp-detail-description",text:e.description})}let c=o.createDiv({cls:"af-slideover-section"});c.createDiv({cls:"af-slideover-section-title",text:"SERVER INFO"}),this.renderDetailRow(c,"Name",e.name),this.renderDetailRow(c,"Transport",e.type),this.renderDetailRow(c,"Enabled",e.enabled?"yes":"no"),e.type!=="stdio"&&(this.renderDetailRow(c,"Auth",e.auth??"none"),this.renderDetailRow(c,"Authenticated",this.mcpHasToken(e)?"yes":"no")),e.source&&this.renderDetailRow(c,"Source",e.source),e.url&&this.renderDetailRow(c,"URL",e.url),e.command&&this.renderDetailRow(c,"Command",e.command),e.args&&e.args.length>0&&this.renderDetailRow(c,"Args",e.args.join(" "));let l=this.mcpProbeCache.get(e.name)??[],h=o.createDiv({cls:"af-slideover-section"});h.createDiv({cls:"af-slideover-section-title"}).setText(`TOOLS (${l.length})`);let u=h.createEl("button",{cls:"af-btn-sm"}),p=u.createSpan();if((0,b.setIcon)(p,"wrench"),u.appendText(" Probe tools"),u.onclick=async()=>{u.disabled=!0,u.setText(" Probing\u2026"),await this.probeMcpServer(e),s.remove(),this.openMcpDetailSlideover(e)},l.length>0)for(let y of l){let v=h.createDiv({cls:"af-mcp-tool-detail"}),k=v.createDiv({cls:"af-mcp-tool-detail-header"}),w=k.createSpan({cls:"af-mcp-tool-detail-name"}),S=w.createSpan();if((0,b.setIcon)(S,"wrench"),w.createSpan({text:` ${y.name}`}),y.inputSchema){let T=y.inputSchema.required??[];T.length>0&&k.createSpan({cls:"af-mcp-tool-param-count",text:`${T.length} param${T.length!==1?"s":""}`})}if(y.description){let T=ye(y.description).filter(O=>O.trim()),_=T.slice(0,2).join(" ").trim();if(T.length>2){let O=v.createEl("details",{cls:"af-mcp-tool-detail-desc"});O.createEl("summary",{text:this.truncateDescription(_,200)}),O.createDiv({cls:"af-mcp-tool-detail-full",text:y.description})}else v.createDiv({cls:"af-mcp-tool-detail-desc",text:_})}if(y.inputSchema){let T=y.inputSchema.properties,_=new Set(y.inputSchema.required??[]);if(T&&Object.keys(T).length>0){let D=v.createDiv({cls:"af-mcp-tool-params"});for(let[O,C]of Object.entries(T)){let E=D.createDiv({cls:"af-mcp-tool-param"});E.createSpan({cls:"af-mcp-tool-param-name",text:O}),C.type&&E.createSpan({cls:"af-mcp-tool-param-type",text:C.type}),_.has(O)&&E.createSpan({cls:"af-mcp-tool-param-required",text:"required"}),C.description&&E.createSpan({cls:"af-mcp-tool-param-desc",text:jt(C.description,80)})}}}}else h.createDiv({cls:"af-form-hint",text:'Click "Probe tools" to discover the tools this server exposes.'});let m=o.createDiv({cls:"af-slideover-section"});if(m.createDiv({cls:"af-slideover-section-title",text:"ACTIONS"}),e.enabled&&e.url&&e.auth==="oauth"&&!this.mcpHasToken(e)){let y=m.createEl("button",{cls:"af-btn-sm primary"}),v=y.createSpan();(0,b.setIcon)(v,"key"),y.appendText(" Authenticate"),y.onclick=()=>{s.remove(),this.authenticateMcpServer(e)}}let f=m.createEl("button",{cls:"af-btn-sm danger"}),g=f.createSpan();(0,b.setIcon)(g,"trash-2"),f.appendText(" Remove Server"),f.onclick=async()=>{try{await this.plugin.repository.deleteMcpServer(e.name),this.plugin.mcpAuth.removeToken(e.name),this.mcpProbeCache.delete(e.name),new b.Notice(`Server "${e.name}" removed.`),s.remove(),await this.plugin.refreshFromVault(),this.render()}catch(y){let v=y instanceof Error?y.message:String(y);new b.Notice(`Failed to remove server: ${v}`)}}}renderAddMcpServerPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=s.createDiv({cls:"af-detail-header"}),a=n.createDiv({cls:"af-detail-header-left"}),r=a.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(r,"plus");let o=a.createDiv();o.createDiv({cls:"af-detail-header-name",text:"Add MCP Server"}),o.createDiv({cls:"af-detail-header-desc",text:"Register a new MCP server for agents to use"});let c=n.createDiv({cls:"af-detail-header-actions"}),l={name:"",transport:"stdio",description:"",command:"",args:"",envVars:"",url:"",headers:"",auth:"none",bearerToken:""},h=s.createDiv({cls:"af-create-form"}),d=h.createDiv({cls:"af-create-section"}),u=d.createDiv({cls:"af-create-section-header"}),p=u.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(p,"plug"),u.createSpan({text:"Server Details"}),this.createFormField(d,"Name","my-server","Unique name for this MCP server",W=>{l.name=W});let m=d.createDiv({cls:"af-form-row"}),f=m.createDiv({cls:"af-form-label"});f.setText("Transport"),this.addTooltip(f,"stdio: local process, http/sse: remote server");let g=m.createEl("select",{cls:"af-form-select"});g.createEl("option",{text:"stdio",attr:{value:"stdio"}}),g.createEl("option",{text:"http",attr:{value:"http"}}),g.createEl("option",{text:"sse",attr:{value:"sse"}}),this.createFormField(d,"Description","What this server does (optional)","Shown on the server card",W=>{l.description=W});let y=h.createDiv({cls:"af-create-section"}),v=y.createDiv({cls:"af-create-section-header"}),k=v.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(k,"terminal"),v.createSpan({text:"Process Configuration"}),this.createFormField(y,"Command","npx @anthropic-ai/mcp-server-memory","The command to run",W=>{l.command=W}),this.createFormField(y,"Arguments","--port 3000","Space-separated arguments (optional)",W=>{l.args=W});let w=y.createDiv({cls:"af-form-label"});w.setText("Environment variables"),this.addTooltip(w,"One KEY=VALUE per line");let S=y.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`API_KEY=sk-...
12125
- DEBUG=true`,rows:"3"}});S.addEventListener("input",()=>{l.envVars=S.value});let T=h.createDiv({cls:"af-create-section"}),_=T.createDiv({cls:"af-create-section-header"}),D=_.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(D,"globe"),_.createSpan({text:"Remote Server Configuration"}),this.createFormField(T,"URL","https://mcp.example.com/sse","Server endpoint URL",W=>{l.url=W});let O=T.createDiv({cls:"af-form-label"});O.setText("Custom headers"),this.addTooltip(O,"One Header: Value per line (optional, non-secret)");let C=T.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"X-Custom-Header: value",rows:"2"}});C.addEventListener("input",()=>{l.headers=C.value});let E=T.createDiv({cls:"af-form-row"}),P=E.createDiv({cls:"af-form-label"});P.setText("Authentication"),this.addTooltip(P,"none, a static bearer token, or OAuth (authenticate after saving)");let N=E.createEl("select",{cls:"af-form-select"});N.createEl("option",{text:"None",attr:{value:"none"}}),N.createEl("option",{text:"Bearer token",attr:{value:"bearer"}}),N.createEl("option",{text:"OAuth",attr:{value:"oauth"}});let M=T.createDiv({cls:"af-form-row"}),U=M.createDiv({cls:"af-form-label"});U.setText("Bearer token"),this.addTooltip(U,"Stored securely in the OS keychain, never written to the vault");let $=M.createEl("input",{cls:"af-form-input",attr:{type:"password",placeholder:"sk-\u2026"}});$.addEventListener("input",()=>{l.bearerToken=$.value});let Z=()=>{M.setCssStyles({display:l.auth==="bearer"?"":"none"})};N.addEventListener("change",()=>{l.auth=N.value,Z()}),Z();let de=()=>{y.setCssStyles({display:l.transport==="stdio"?"":"none"}),T.setCssStyles({display:l.transport!=="stdio"?"":"none"})};g.addEventListener("change",()=>{l.transport=g.value,de()}),de();let me=s.createDiv({cls:"af-create-footer"}),ge=me.createEl("button",{cls:"af-btn-sm",text:"Cancel"});ge.onclick=()=>this.navigate("mcp");let X=me.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(X,"plus","af-btn-icon"),X.appendText(" Add Server"),X.onclick=async()=>{let W=l.name.trim();if(!W){new b.Notice("Server name is required.");return}if(l.transport==="stdio"){if(!l.command.trim()){new b.Notice("Command is required for stdio servers.");return}}else if(!l.url.trim()){new b.Notice("URL is required for HTTP/SSE servers.");return}let K={};if(l.envVars.trim())for(let Q of ye(l.envVars)){let ae=Q.trim();if(!ae)continue;let ve=ae.indexOf("=");if(ve<=0){new b.Notice(`Invalid env var: ${ae}`);return}K[ae.slice(0,ve)]=ae.slice(ve+1)}let ne={};if(l.headers.trim())for(let Q of ye(l.headers)){let ae=Q.trim();if(!ae)continue;let ve=ae.indexOf(":");if(ve<=0){new b.Notice(`Invalid header: ${ae}`);return}ne[ae.slice(0,ve).trim()]=ae.slice(ve+1).trim()}let G=l.args.trim()?l.args.trim().split(/\s+/):void 0;if(this.plugin.repository.getMcpServerByName(W)){new b.Notice(`An MCP server named "${W}" already exists.`);return}if(l.transport!=="stdio"&&l.auth==="bearer"&&!l.bearerToken.trim()){new b.Notice("Enter a bearer token, or choose a different auth method.");return}X.disabled=!0,X.setText("Adding...");let ee={name:W,type:l.transport,enabled:!0,source:"manual",status:"disconnected",scope:"user",tools:[],toolDetails:[]};l.transport==="stdio"?(ee.command=l.command.trim(),G&&(ee.args=G),Object.keys(K).length>0&&(ee.env=K)):(ee.url=l.url.trim(),Object.keys(ne).length>0&&(ee.headers=ne),ee.auth=l.auth);try{await this.plugin.repository.saveMcpServer(ee,l.description.trim()),l.transport!=="stdio"&&l.auth==="bearer"&&l.bearerToken.trim()&&this.plugin.mcpAuth.storeStaticToken(W,l.bearerToken.trim()),new b.Notice(`Server "${W}" added.`),await this.plugin.refreshFromVault(),this.navigate("mcp")}catch(Q){let ae=Q instanceof Error?Q.message:String(Q);new b.Notice(`Failed to add server: ${ae}`),X.disabled=!1,X.setText(""),R(X,"plus","af-btn-icon"),X.appendText(" Add Server")}}}createFormField(e,s,n,a,r,o){let c=e.createDiv({cls:"af-form-row"}),l=c.createDiv({cls:"af-form-label"});l.setText(s),a&&this.addTooltip(l,a);let h=c.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:n}});o!==void 0&&(h.value=o),h.addEventListener("input",()=>r(h.value))}addTooltip(e,s){let n=e.createSpan({cls:"af-form-tooltip"});(0,b.setIcon)(n,"info"),n.createSpan({cls:"af-tooltip-text",text:s})}};function Xo(i){switch(i){case"connected":return"idle";case"connecting":case"reconnecting":return"pending";case"needs-auth":case"error":return"error";case"stopped":case"disabled":default:return"disabled"}}function Ka(i){return i>=1e6?`${(i/1e6).toFixed(1)}M`:i>=1e4?`${Math.round(i/1e3)}K`:i>=1e3?`${(i/1e3).toFixed(1)}K`:String(i)}function kh(i){if(!i?.trim())return"not set";let t={"*/5 * * * *":"Every 5 minutes","*/10 * * * *":"Every 10 minutes","*/15 * * * *":"Every 15 minutes","*/30 * * * *":"Every 30 minutes","0 * * * *":"Every hour","0 */2 * * *":"Every 2 hours","0 */4 * * *":"Every 4 hours","0 */6 * * *":"Every 6 hours","0 */12 * * *":"Every 12 hours"};if(t[i])return t[i];let e=i.trim().split(/\s+/);if(e.length===5){let[s,n,a,,r]=e;if(a==="*"&&n&&s){let o=Number(n),c=Number(s);if(!isNaN(o)&&!isNaN(c)){let l=o>=12?"PM":"AM",d=`${o===0?12:o>12?o-12:o}:${String(c).padStart(2,"0")} ${l}`;return r==="*"?`Daily at ${d}`:r==="1-5"?`Weekdays at ${d}`:`${d} on days ${r}`}}}return i}function xh(i){switch(i){case"connected":return"green";case"connecting":case"reconnecting":return"blue";case"needs-auth":case"error":return"red";case"stopped":case"disabled":default:return""}}var z=require("obsidian");function qn(i,t){return`${i}::${t}`}function el(){return Math.random().toString(16).slice(2,10)}function tl(){return`New chat ${new Date().toLocaleString(void 0,{month:"short",day:"numeric",hour:"2-digit",minute:"2-digit"})}`}function Sh(i){let t=new Date(i).getTime();if(!Number.isFinite(t))return"";let e=Date.now()-t,s=60*1e3,n=60*s,a=24*n;return e<s?"just now":e<n?`${Math.floor(e/s)}m`:e<a?`${Math.floor(e/n)}h`:e<2*a?"yesterday":e<7*a?`${Math.floor(e/a)}d`:new Date(t).toLocaleDateString(void 0,{month:"short",day:"numeric"})}var Ch=480,Ja="http://www.w3.org/2000/svg";function Th(i){let t=activeDocument.createElementNS(Ja,"svg");t.setAttribute("class","svg-icon af-convo-toggle-icon"),t.setAttribute("viewBox","0 0 24 24"),t.setAttribute("fill","none"),t.setAttribute("stroke","currentColor"),t.setAttribute("stroke-width","2"),t.setAttribute("stroke-linecap","round"),t.setAttribute("stroke-linejoin","round");let e=activeDocument.createElementNS(Ja,"rect");for(let[n,a]of Object.entries({x:"1",y:"2",width:"22",height:"20",rx:"4"}))e.setAttribute(n,a);let s=activeDocument.createElementNS(Ja,"rect");s.setAttribute("class","af-convo-toggle-bar");for(let[n,a]of Object.entries({x:"4",y:"5",width:"2",height:"14",rx:"2",fill:"currentColor"}))s.setAttribute(n,a);t.appendChild(e),t.appendChild(s),i.appendChild(t)}var rs=class i extends z.ItemView{constructor(e,s){super(e);this.plugin=s}selectedAgentName=null;selectedConversationId="";sessions=new Map;conversationsCache=[];convoPanelEl=null;collapseBtn=null;convoPanelCollapsed=!1;userToggledCollapse=!1;resizeObserver=null;headerEl;agentSelect;messagesEl;messagesInner;textarea;sendBtn;attachStopBtn;isInStopMode=!1;attachedFiles=[];attachedImages=[];pillsRow;activityEl=null;streamingDot=null;activityUnsub=null;statsEl;statsUnsub=null;statsSourceSession=null;threadExpanded=new Map;getViewType(){return it}getDisplayText(){if(!this.selectedAgentName)return"Agent Chat";let e=this.getCurrentConversationName();return!e||this.conversationsCache.length<=1?`Chat: ${this.selectedAgentName}`:`Chat: ${this.selectedAgentName} \xB7 ${e}`}getIcon(){return"message-circle"}getState(){return{agentName:this.selectedAgentName??null,conversationId:this.selectedConversationId,convoPanelCollapsed:this.convoPanelCollapsed}}async setState(e,s){await super.setState(e,s);let n=typeof e?.conversationId=="string"&&e.conversationId?e.conversationId:void 0;typeof e?.convoPanelCollapsed=="boolean"&&(this.convoPanelCollapsed=e.convoPanelCollapsed,this.userToggledCollapse=!0,this.applyCollapsedClass()),e?.agentName&&typeof e.agentName=="string"&&this.selectAgent(e.agentName,n)}getCurrentConversationName(){return this.conversationsCache.find(s=>s.id===this.selectedConversationId)?.name??""}async onOpen(){this.plugin.subscribeView(this),this.buildShell(),await this.render(),this.observeContainerWidth()}async onClose(){for(let{session:e}of this.sessions.values())e.abort();this.sessions.clear(),this.statsUnsub?.(),this.statsUnsub=null,this.activityUnsub?.(),this.activityUnsub=null,this.resizeObserver?.disconnect(),this.resizeObserver=null,this.plugin.unsubscribeView(this)}observeContainerWidth(){typeof ResizeObserver>"u"||(this.resizeObserver=new ResizeObserver(e=>{if(!this.userToggledCollapse)for(let s of e){let a=s.contentRect.width<Ch;a!==this.convoPanelCollapsed&&(this.convoPanelCollapsed=a,this.applyCollapsedClass())}}),this.resizeObserver.observe(this.contentEl))}selectAgent(e,s){if(s){let n=this.plugin.app.workspace.getLeavesOfType(it);for(let a of n)if(a.view!==this&&a.view instanceof i&&a.view.selectedAgentName===e&&a.view.selectedConversationId===s){this.plugin.app.workspace.revealLeaf(a);return}}this.selectedAgentName=e,s&&(this.selectedConversationId=s),this.agentSelect&&(this.agentSelect.value=e),this.leaf.updateHeader(),this.switchToAgent(e,s)}buildShell(){let e=this.contentEl;e.empty(),e.addClass("af-root");let s=e.createDiv({cls:"af-chat-view-container"});this.headerEl=s.createDiv({cls:"af-chat-view-header"}),this.collapseBtn=this.headerEl.createEl("button",{cls:"clickable-icon af-chat-convo-collapse-btn",attr:{title:"Toggle conversations panel","aria-label":"Toggle conversations panel"}}),Th(this.collapseBtn),this.collapseBtn.toggleClass("is-collapsed",this.convoPanelCollapsed),this.collapseBtn.onclick=()=>this.toggleConvoPanel(),this.agentSelect=this.headerEl.createEl("select",{cls:"af-chat-view-agent-select"});let n=this.headerEl.createEl("button",{cls:"af-btn-sm af-chat-view-new-btn"});R(n,"plus","af-btn-icon"),n.appendText(" New Chat"),n.onclick=()=>void this.handleNewChat();let a=s.createDiv({cls:"af-chat-view-body"});this.convoPanelEl=a.createDiv({cls:"af-chat-convo-panel"}),this.convoPanelCollapsed&&this.convoPanelEl.addClass("collapsed");let r=a.createDiv({cls:"af-chat-main"});this.agentSelect.onchange=()=>{let h=this.agentSelect.value;h&&(this.selectedConversationId="",this.textarea.disabled=!1,this.textarea.placeholder="Message the agent\u2026 (Shift+Enter for newline)",this.switchToAgent(h))},this.messagesEl=r.createDiv({cls:"af-chat-messages"}),this.messagesInner=this.messagesEl.createDiv({cls:"af-chat-messages-inner"});let o=r.createDiv({cls:"af-chat-input-area"});this.pillsRow=o.createDiv({cls:"af-chat-pills-row"}),this.pillsRow.setCssStyles({display:"none"});let c=o.createDiv({cls:"af-chat-input-row"});this.attachStopBtn=c.createEl("button",{cls:"af-chat-attach-btn"}),R(this.attachStopBtn,"plus","af-btn-icon"),this.attachStopBtn.title="Attach active document",this.attachStopBtn.onclick=()=>{this.isInStopMode?this.handleStop():this.attachActiveDocument()},this.textarea=c.createEl("textarea",{cls:"af-chat-input",attr:{placeholder:"Message the agent\u2026 (Shift+Enter for newline)",rows:"1"}}),this.sendBtn=c.createEl("button",{cls:"af-chat-send-btn"}),R(this.sendBtn,"arrow-up","af-btn-icon"),this.sendBtn.setCssStyles({display:"none"});let l=()=>{this.textarea.setCssStyles({height:"auto"});let h=Math.min(this.textarea.scrollHeight,160);this.textarea.setCssStyles({height:`${h}px`}),this.textarea.setCssStyles({overflowY:this.textarea.scrollHeight>160?"auto":"hidden"}),this.sendBtn.setCssStyles({display:this.textarea.value.trim()?"flex":"none"})};this.textarea.addEventListener("input",l),this.textarea.addEventListener("focus",()=>{let h=this.getCurrentSession();h&&this.setStatsSource(h.session)}),this.sendBtn.onclick=()=>void this.handleSend(),this.textarea.onkeydown=h=>{h.key==="Enter"&&!h.shiftKey&&!h.isComposing&&(h.preventDefault(),this.handleSend())},this.textarea.addEventListener("paste",h=>{let d=h.clipboardData?.items;if(d)for(let u=0;u<d.length;u++){let p=d[u];if(p.type.startsWith("image/")){h.preventDefault();let m=p.getAsFile();m&&this.attachImageBlob(m);return}}}),o.addEventListener("dragover",h=>{h.preventDefault(),h.stopPropagation(),o.addClass("af-chat-input-dragover")}),o.addEventListener("dragleave",()=>{o.removeClass("af-chat-input-dragover")}),o.addEventListener("drop",h=>{h.preventDefault(),h.stopPropagation(),o.removeClass("af-chat-input-dragover");let d=h.dataTransfer?.files;if(d)for(let u=0;u<d.length;u++){let p=d[u];p.type.startsWith("image/")&&this.attachImageBlob(p)}}),this.statsEl=o.createDiv({cls:"af-chat-stats"}),this.renderStats(null)}async render(){this.populateAgentDropdown()}setStatsSource(e){this.statsSourceSession!==e&&(this.statsUnsub?.(),this.statsSourceSession=e,this.statsUnsub=e.onStatsChange(s=>this.renderStats(s)))}renderStats(e){if(this.statsEl.empty(),!e||!e.concreteModel){this.statsEl.createSpan({cls:"af-chat-stats-muted",text:"\xA0"});return}let s=this.statsEl.createSpan({cls:"af-chat-stats-line"});if(s.createSpan({cls:"af-chat-stats-model",text:e.concreteModel}),e.contextWindow&&e.contextTokensUsed){let n=Math.min(100,Math.round(e.contextTokensUsed/e.contextWindow*100)),a=10,r=Math.min(a,Math.max(0,Math.round(n/100*a))),o="\u2593".repeat(r)+"\u2591".repeat(a-r),c=s.createSpan({cls:`af-chat-stats-ctx${n>=80?" warn":""}`});c.createSpan({cls:"af-chat-stats-bar",text:o}),c.createSpan({cls:"af-chat-stats-pct",text:`${n}%`}),c.title=`Context: ${is(e.contextTokensUsed)} / ${is(e.contextWindow)} tokens (${n}%)
12035
+ ...`,rows:"6"}});ge.value=a.examplesBody,ge.addEventListener("input",()=>{d.examplesBody=ge.value});let X=s.createDiv({cls:"af-create-footer"}),W=X.createEl("button",{cls:"af-btn-sm danger"});R(W,"trash-2","af-btn-icon"),W.appendText(" Delete"),W.onclick=async()=>{await this.plugin.repository.deleteSkill(a.name),new b.Notice(`Skill "${a.name}" deleted.`),await new Promise(G=>window.setTimeout(G,200)),await this.plugin.refreshFromVault(),this.navigate("skills")},X.createDiv({cls:"af-toolbar-spacer"});let K=X.createEl("button",{cls:"af-btn-sm",text:"Cancel"});K.onclick=()=>this.navigate("skills");let ne=X.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(ne,"check","af-btn-icon"),ne.appendText(" Save Changes"),ne.onclick=async()=>{let G=ee=>ee.split(",").map(Q=>Q.trim()).filter(Boolean);try{await this.plugin.repository.updateSkill(a.name,{description:d.description.trim(),tags:G(d.tags),body:d.body.trim(),toolsBody:d.toolsBody.trim(),referencesBody:d.referencesBody.trim(),examplesBody:d.examplesBody.trim()}),new b.Notice(`Skill "${a.name}" updated.`),await this.plugin.refreshFromVault(),this.navigate("skills")}catch(ee){let Q=ee instanceof Error?ee.message:String(ee);new b.Notice(`Failed to update skill: ${Q}`)}}}renderAgentMcpPicker(e,s){e.createDiv({cls:"af-form-hint",text:"Servers from the MCP Servers tab. Checked servers are available to this agent on any adapter (Claude or Codex). Leave all unchecked to grant every enabled server."});let n=this.plugin.repository.getMcpServers();if(n.length===0){let i=e.createDiv({cls:"af-form-hint"});i.appendText("No MCP servers registered yet. ");let o=i.createEl("a",{cls:"af-link",text:"Add one in the MCP Servers tab."});o.onclick=c=>{c.preventDefault(),this.navigate("mcp")};return}let a=e.createDiv({cls:"af-create-skills-grid"});for(let i of n){let o=a.createDiv({cls:"af-mcp-agent-item"}),c=o.createEl("input",{cls:"af-form-toggle",attr:{type:"checkbox"}});c.checked=s.has(i.name),c.addEventListener("change",()=>{c.checked?s.add(i.name):s.delete(i.name)});let h=o.createDiv({cls:"af-mcp-agent-label"}).createDiv({cls:"af-mcp-agent-name-row"}),d=h.createSpan({cls:`af-mcp-status-dot ${i.enabled?"idle":"disabled"}`});d.title=i.enabled?"enabled":"disabled",h.createSpan({cls:"af-create-skill-name",text:i.name}),h.createSpan({cls:"af-mcp-agent-tool-count",text:i.type}),i.enabled||h.createSpan({cls:"af-mcp-agent-tool-count af-muted",text:"disabled"})}}renderMcpPage(e){let s=e.createDiv({cls:"af-agents-page"}),n=s.createDiv({cls:"af-agents-toolbar"});n.createDiv({cls:"af-page-title",text:"MCP Servers"}),n.createDiv({cls:"af-toolbar-spacer"});let a=n.createEl("button",{cls:"af-btn-sm primary"});R(a,"plus","af-btn-icon"),a.appendText(" Add Server"),a.onclick=()=>this.navigate("add-mcp-server");let i=this.plugin.repository.getMcpServers();if(i.length===0){this.renderEmptyState(s,"plug","No MCP servers registered","Click 'Add Server' above to register one.");return}let o=s.createDiv({cls:"af-agents-grid"});for(let c of i)this.renderMcpCard(o,c)}mcpProbeCache=new Map;mcpHasToken(e){return this.plugin.mcpAuth.hasToken(e.name)}mcpNeedsAuth(e){return e.type!=="stdio"&&(e.auth==="oauth"||e.auth==="bearer")&&!this.mcpHasToken(e)}renderMcpCard(e,s){let n=e.createDiv({cls:`af-mcp-card${s.enabled?"":" af-mcp-card-disabled"}`}),a=this.mcpNeedsAuth(s),i=n.createDiv({cls:"af-agent-card-header"}),o=s.enabled?a?"pending":"idle":"disabled",c=i.createDiv({cls:`af-agent-card-avatar ${o}`});(0,b.setIcon)(c,"plug");let l=i.createDiv({cls:"af-agent-card-titleblock"});l.createDiv({cls:"af-agent-card-name",text:s.name});let h=l.createDiv({cls:"af-agent-card-desc af-mcp-meta"});h.createSpan({cls:"af-mcp-type-badge",text:s.type}),s.source==="imported"&&h.createSpan({cls:"af-badge",text:"imported"});let d=i.createDiv({cls:`af-agent-card-toggle${s.enabled?" on":""}`});d.onclick=g=>{g.stopPropagation(),this.plugin.repository.setMcpServerEnabled(s.name,!s.enabled).then(async()=>{await this.plugin.refreshFromVault(),this.render()})};let u=n.createDiv({cls:`af-mcp-status-badge ${s.enabled?a?"needs-auth":"connected":"disabled"}`}),p=u.createSpan();if(s.enabled?a?((0,b.setIcon)(p,"alert-circle"),u.createSpan({text:" Needs auth"})):((0,b.setIcon)(p,"check-circle"),u.createSpan({text:s.type==="stdio"?" Enabled":" Authenticated"})):((0,b.setIcon)(p,"pause"),u.createSpan({text:" Disabled"})),s.description){let g=this.truncateDescription(s.description,120);n.createDiv({cls:"af-mcp-description",text:g})}let m=s.url??s.command??"";m&&n.createDiv({cls:"af-mcp-command",text:Wt(m,60)});let f=this.mcpProbeCache.get(s.name);if(f&&f.length>0){let g=n.createDiv({cls:"af-mcp-tool-footer"}),y=g.createDiv({cls:"af-mcp-tool-count"}),v=y.createSpan();(0,b.setIcon)(v,"wrench"),y.createSpan({text:` ${f.length} tools`});let k=g.createDiv({cls:"af-mcp-tool-chips"});for(let w of f.slice(0,4))k.createSpan({cls:"af-mcp-tool-chip",text:w.name});f.length>4&&k.createSpan({cls:"af-mcp-tool-chip af-mcp-tool-chip-more",text:`+${f.length-4}`})}if(this.authenticatingServers.has(s.name)){let y=n.createDiv({cls:"af-mcp-auth-row"}).createEl("button",{cls:"af-btn-sm primary",attr:{disabled:"true"}}),v=y.createSpan({cls:"af-spin"});(0,b.setIcon)(v,"loader-2"),y.appendText(" Authenticating\u2026")}else if(s.enabled&&a&&s.auth==="oauth"){let y=n.createDiv({cls:"af-mcp-auth-row"}).createEl("button",{cls:"af-btn-sm primary"}),v=y.createSpan();(0,b.setIcon)(v,"key"),y.appendText(" Authenticate"),y.onclick=k=>{k.stopPropagation(),this.authenticateMcpServer(s)}}n.onclick=()=>this.openMcpDetailSlideover(s)}async authenticateMcpServer(e){if(!e.url){new b.Notice("No URL found for this server \u2014 can't authenticate.");return}this.authenticatingServers.add(e.name),this.render(),new b.Notice(`Authenticating ${e.name}\u2026 Complete authorization in your browser.`,1e4);try{let s=e.type==="sse"?"sse":"http";await this.plugin.mcpManager.authenticateServer(e.name,e.url,s),new b.Notice(`${e.name} authenticated successfully!`)}catch(s){let n=s instanceof Error?s.message:String(s);new b.Notice(`Authentication failed: ${n}`,8e3)}finally{this.authenticatingServers.delete(e.name),this.render()}}async probeMcpServer(e){try{let s=await this.plugin.mcpManager.probeServer(e);this.mcpProbeCache.set(e.name,s),s.length===0&&new b.Notice(`No tools discovered for ${e.name}.`)}catch(s){let n=s instanceof Error?s.message:String(s);new b.Notice(`Probe failed: ${n}`)}}truncateDescription(e,s){let n=ye(e)[0]??e,a=n.split(/(?<=[.!?])\s/)[0]??n,i=a.length<n.length?a:n;return i.length<=s?i:i.slice(0,s-1)+"\u2026"}openMcpDetailSlideover(e){this.contentEl.querySelector(".af-slideover-overlay")?.remove();let s=this.contentEl.createDiv({cls:"af-slideover-overlay"}),n=s.createDiv({cls:"af-slideover"}),a=n.createDiv({cls:"af-slideover-header"});a.createDiv({cls:"af-slideover-title",text:e.name});let i=a.createEl("button",{cls:"clickable-icon"});(0,b.setIcon)(i,"cross"),i.onclick=()=>s.remove(),s.onclick=y=>{y.target===s&&s.remove()};let o=n.createDiv({cls:"af-slideover-body"});if(e.description){let y=o.createDiv({cls:"af-slideover-section"});y.createDiv({cls:"af-slideover-section-title",text:"DESCRIPTION"}),y.createDiv({cls:"af-mcp-detail-description",text:e.description})}let c=o.createDiv({cls:"af-slideover-section"});c.createDiv({cls:"af-slideover-section-title",text:"SERVER INFO"}),this.renderDetailRow(c,"Name",e.name),this.renderDetailRow(c,"Transport",e.type),this.renderDetailRow(c,"Enabled",e.enabled?"yes":"no"),e.type!=="stdio"&&(this.renderDetailRow(c,"Auth",e.auth??"none"),this.renderDetailRow(c,"Authenticated",this.mcpHasToken(e)?"yes":"no")),e.source&&this.renderDetailRow(c,"Source",e.source),e.url&&this.renderDetailRow(c,"URL",e.url),e.command&&this.renderDetailRow(c,"Command",e.command),e.args&&e.args.length>0&&this.renderDetailRow(c,"Args",e.args.join(" "));let l=this.mcpProbeCache.get(e.name)??[],h=o.createDiv({cls:"af-slideover-section"});h.createDiv({cls:"af-slideover-section-title"}).setText(`TOOLS (${l.length})`);let u=h.createEl("button",{cls:"af-btn-sm"}),p=u.createSpan();if((0,b.setIcon)(p,"wrench"),u.appendText(" Probe tools"),u.onclick=async()=>{u.disabled=!0,u.setText(" Probing\u2026"),await this.probeMcpServer(e),s.remove(),this.openMcpDetailSlideover(e)},l.length>0)for(let y of l){let v=h.createDiv({cls:"af-mcp-tool-detail"}),k=v.createDiv({cls:"af-mcp-tool-detail-header"}),w=k.createSpan({cls:"af-mcp-tool-detail-name"}),S=w.createSpan();if((0,b.setIcon)(S,"wrench"),w.createSpan({text:` ${y.name}`}),y.inputSchema){let T=y.inputSchema.required??[];T.length>0&&k.createSpan({cls:"af-mcp-tool-param-count",text:`${T.length} param${T.length!==1?"s":""}`})}if(y.description){let T=ye(y.description).filter(O=>O.trim()),_=T.slice(0,2).join(" ").trim();if(T.length>2){let O=v.createEl("details",{cls:"af-mcp-tool-detail-desc"});O.createEl("summary",{text:this.truncateDescription(_,200)}),O.createDiv({cls:"af-mcp-tool-detail-full",text:y.description})}else v.createDiv({cls:"af-mcp-tool-detail-desc",text:_})}if(y.inputSchema){let T=y.inputSchema.properties,_=new Set(y.inputSchema.required??[]);if(T&&Object.keys(T).length>0){let D=v.createDiv({cls:"af-mcp-tool-params"});for(let[O,C]of Object.entries(T)){let E=D.createDiv({cls:"af-mcp-tool-param"});E.createSpan({cls:"af-mcp-tool-param-name",text:O}),C.type&&E.createSpan({cls:"af-mcp-tool-param-type",text:C.type}),_.has(O)&&E.createSpan({cls:"af-mcp-tool-param-required",text:"required"}),C.description&&E.createSpan({cls:"af-mcp-tool-param-desc",text:Wt(C.description,80)})}}}}else h.createDiv({cls:"af-form-hint",text:'Click "Probe tools" to discover the tools this server exposes.'});let m=o.createDiv({cls:"af-slideover-section"});if(m.createDiv({cls:"af-slideover-section-title",text:"ACTIONS"}),e.enabled&&e.url&&e.auth==="oauth"&&!this.mcpHasToken(e)){let y=m.createEl("button",{cls:"af-btn-sm primary"}),v=y.createSpan();(0,b.setIcon)(v,"key"),y.appendText(" Authenticate"),y.onclick=()=>{s.remove(),this.authenticateMcpServer(e)}}let f=m.createEl("button",{cls:"af-btn-sm danger"}),g=f.createSpan();(0,b.setIcon)(g,"trash-2"),f.appendText(" Remove Server"),f.onclick=async()=>{try{await this.plugin.repository.deleteMcpServer(e.name),this.plugin.mcpAuth.removeToken(e.name),this.mcpProbeCache.delete(e.name),new b.Notice(`Server "${e.name}" removed.`),s.remove(),await this.plugin.refreshFromVault(),this.render()}catch(y){let v=y instanceof Error?y.message:String(y);new b.Notice(`Failed to remove server: ${v}`)}}}renderAddMcpServerPage(e){let s=e.createDiv({cls:"af-create-agent-page"}),n=s.createDiv({cls:"af-detail-header"}),a=n.createDiv({cls:"af-detail-header-left"}),i=a.createDiv({cls:"af-agent-card-avatar idle"});(0,b.setIcon)(i,"plus");let o=a.createDiv();o.createDiv({cls:"af-detail-header-name",text:"Add MCP Server"}),o.createDiv({cls:"af-detail-header-desc",text:"Register a new MCP server for agents to use"});let c=n.createDiv({cls:"af-detail-header-actions"}),l={name:"",transport:"stdio",description:"",command:"",args:"",envVars:"",url:"",headers:"",auth:"none",bearerToken:""},h=s.createDiv({cls:"af-create-form"}),d=h.createDiv({cls:"af-create-section"}),u=d.createDiv({cls:"af-create-section-header"}),p=u.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(p,"plug"),u.createSpan({text:"Server Details"}),this.createFormField(d,"Name","my-server","Unique name for this MCP server",W=>{l.name=W});let m=d.createDiv({cls:"af-form-row"}),f=m.createDiv({cls:"af-form-label"});f.setText("Transport"),this.addTooltip(f,"stdio: local process, http/sse: remote server");let g=m.createEl("select",{cls:"af-form-select"});g.createEl("option",{text:"stdio",attr:{value:"stdio"}}),g.createEl("option",{text:"http",attr:{value:"http"}}),g.createEl("option",{text:"sse",attr:{value:"sse"}}),this.createFormField(d,"Description","What this server does (optional)","Shown on the server card",W=>{l.description=W});let y=h.createDiv({cls:"af-create-section"}),v=y.createDiv({cls:"af-create-section-header"}),k=v.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(k,"terminal"),v.createSpan({text:"Process Configuration"}),this.createFormField(y,"Command","npx @anthropic-ai/mcp-server-memory","The command to run",W=>{l.command=W}),this.createFormField(y,"Arguments","--port 3000","Space-separated arguments (optional)",W=>{l.args=W});let w=y.createDiv({cls:"af-form-label"});w.setText("Environment variables"),this.addTooltip(w,"One KEY=VALUE per line");let S=y.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:`API_KEY=sk-...
12036
+ DEBUG=true`,rows:"3"}});S.addEventListener("input",()=>{l.envVars=S.value});let T=h.createDiv({cls:"af-create-section"}),_=T.createDiv({cls:"af-create-section-header"}),D=_.createSpan({cls:"af-create-section-icon"});(0,b.setIcon)(D,"globe"),_.createSpan({text:"Remote Server Configuration"}),this.createFormField(T,"URL","https://mcp.example.com/sse","Server endpoint URL",W=>{l.url=W});let O=T.createDiv({cls:"af-form-label"});O.setText("Custom headers"),this.addTooltip(O,"One Header: Value per line (optional, non-secret)");let C=T.createEl("textarea",{cls:"af-create-prompt-textarea",attr:{placeholder:"X-Custom-Header: value",rows:"2"}});C.addEventListener("input",()=>{l.headers=C.value});let E=T.createDiv({cls:"af-form-row"}),P=E.createDiv({cls:"af-form-label"});P.setText("Authentication"),this.addTooltip(P,"none, a static bearer token, or OAuth (authenticate after saving)");let N=E.createEl("select",{cls:"af-form-select"});N.createEl("option",{text:"None",attr:{value:"none"}}),N.createEl("option",{text:"Bearer token",attr:{value:"bearer"}}),N.createEl("option",{text:"OAuth",attr:{value:"oauth"}});let M=T.createDiv({cls:"af-form-row"}),U=M.createDiv({cls:"af-form-label"});U.setText("Bearer token"),this.addTooltip(U,"Stored securely in the OS keychain, never written to the vault");let $=M.createEl("input",{cls:"af-form-input",attr:{type:"password",placeholder:"sk-\u2026"}});$.addEventListener("input",()=>{l.bearerToken=$.value});let Z=()=>{M.setCssStyles({display:l.auth==="bearer"?"":"none"})};N.addEventListener("change",()=>{l.auth=N.value,Z()}),Z();let de=()=>{y.setCssStyles({display:l.transport==="stdio"?"":"none"}),T.setCssStyles({display:l.transport!=="stdio"?"":"none"})};g.addEventListener("change",()=>{l.transport=g.value,de()}),de();let me=s.createDiv({cls:"af-create-footer"}),ge=me.createEl("button",{cls:"af-btn-sm",text:"Cancel"});ge.onclick=()=>this.navigate("mcp");let X=me.createEl("button",{cls:"af-btn-sm primary af-create-submit"});R(X,"plus","af-btn-icon"),X.appendText(" Add Server"),X.onclick=async()=>{let W=l.name.trim();if(!W){new b.Notice("Server name is required.");return}if(l.transport==="stdio"){if(!l.command.trim()){new b.Notice("Command is required for stdio servers.");return}}else if(!l.url.trim()){new b.Notice("URL is required for HTTP/SSE servers.");return}let K={};if(l.envVars.trim())for(let Q of ye(l.envVars)){let ae=Q.trim();if(!ae)continue;let ve=ae.indexOf("=");if(ve<=0){new b.Notice(`Invalid env var: ${ae}`);return}K[ae.slice(0,ve)]=ae.slice(ve+1)}let ne={};if(l.headers.trim())for(let Q of ye(l.headers)){let ae=Q.trim();if(!ae)continue;let ve=ae.indexOf(":");if(ve<=0){new b.Notice(`Invalid header: ${ae}`);return}ne[ae.slice(0,ve).trim()]=ae.slice(ve+1).trim()}let G=l.args.trim()?l.args.trim().split(/\s+/):void 0;if(this.plugin.repository.getMcpServerByName(W)){new b.Notice(`An MCP server named "${W}" already exists.`);return}if(l.transport!=="stdio"&&l.auth==="bearer"&&!l.bearerToken.trim()){new b.Notice("Enter a bearer token, or choose a different auth method.");return}X.disabled=!0,X.setText("Adding...");let ee={name:W,type:l.transport,enabled:!0,source:"manual",status:"disconnected",scope:"user",tools:[],toolDetails:[]};l.transport==="stdio"?(ee.command=l.command.trim(),G&&(ee.args=G),Object.keys(K).length>0&&(ee.env=K)):(ee.url=l.url.trim(),Object.keys(ne).length>0&&(ee.headers=ne),ee.auth=l.auth);try{await this.plugin.repository.saveMcpServer(ee,l.description.trim()),l.transport!=="stdio"&&l.auth==="bearer"&&l.bearerToken.trim()&&this.plugin.mcpAuth.storeStaticToken(W,l.bearerToken.trim()),new b.Notice(`Server "${W}" added.`),await this.plugin.refreshFromVault(),this.navigate("mcp")}catch(Q){let ae=Q instanceof Error?Q.message:String(Q);new b.Notice(`Failed to add server: ${ae}`),X.disabled=!1,X.setText(""),R(X,"plus","af-btn-icon"),X.appendText(" Add Server")}}}createFormField(e,s,n,a,i,o){let c=e.createDiv({cls:"af-form-row"}),l=c.createDiv({cls:"af-form-label"});l.setText(s),a&&this.addTooltip(l,a);let h=c.createEl("input",{cls:"af-form-input",attr:{type:"text",placeholder:n}});o!==void 0&&(h.value=o),h.addEventListener("input",()=>i(h.value))}addTooltip(e,s){let n=e.createSpan({cls:"af-form-tooltip"});(0,b.setIcon)(n,"info"),n.createSpan({cls:"af-tooltip-text",text:s})}};function Qo(r){switch(r){case"connected":return"idle";case"connecting":case"reconnecting":return"pending";case"needs-auth":case"error":return"error";case"stopped":case"disabled":default:return"disabled"}}function Ka(r){return r>=1e6?`${(r/1e6).toFixed(1)}M`:r>=1e4?`${Math.round(r/1e3)}K`:r>=1e3?`${(r/1e3).toFixed(1)}K`:String(r)}function xh(r){if(!r?.trim())return"not set";let t={"*/5 * * * *":"Every 5 minutes","*/10 * * * *":"Every 10 minutes","*/15 * * * *":"Every 15 minutes","*/30 * * * *":"Every 30 minutes","0 * * * *":"Every hour","0 */2 * * *":"Every 2 hours","0 */4 * * *":"Every 4 hours","0 */6 * * *":"Every 6 hours","0 */12 * * *":"Every 12 hours"};if(t[r])return t[r];let e=r.trim().split(/\s+/);if(e.length===5){let[s,n,a,,i]=e;if(a==="*"&&n&&s){let o=Number(n),c=Number(s);if(!isNaN(o)&&!isNaN(c)){let l=o>=12?"PM":"AM",d=`${o===0?12:o>12?o-12:o}:${String(c).padStart(2,"0")} ${l}`;return i==="*"?`Daily at ${d}`:i==="1-5"?`Weekdays at ${d}`:`${d} on days ${i}`}}}return r}function Sh(r){switch(r){case"connected":return"green";case"connecting":case"reconnecting":return"blue";case"needs-auth":case"error":return"red";case"stopped":case"disabled":default:return""}}var z=require("obsidian");function qn(r,t){return`${r}::${t}`}function tl(){return Math.random().toString(16).slice(2,10)}function sl(){return`New chat ${new Date().toLocaleString(void 0,{month:"short",day:"numeric",hour:"2-digit",minute:"2-digit"})}`}function Ch(r){let t=new Date(r).getTime();if(!Number.isFinite(t))return"";let e=Date.now()-t,s=60*1e3,n=60*s,a=24*n;return e<s?"just now":e<n?`${Math.floor(e/s)}m`:e<a?`${Math.floor(e/n)}h`:e<2*a?"yesterday":e<7*a?`${Math.floor(e/a)}d`:new Date(t).toLocaleDateString(void 0,{month:"short",day:"numeric"})}var Th=480,Ja="http://www.w3.org/2000/svg";function _h(r){let t=activeDocument.createElementNS(Ja,"svg");t.setAttribute("class","svg-icon af-convo-toggle-icon"),t.setAttribute("viewBox","0 0 24 24"),t.setAttribute("fill","none"),t.setAttribute("stroke","currentColor"),t.setAttribute("stroke-width","2"),t.setAttribute("stroke-linecap","round"),t.setAttribute("stroke-linejoin","round");let e=activeDocument.createElementNS(Ja,"rect");for(let[n,a]of Object.entries({x:"1",y:"2",width:"22",height:"20",rx:"4"}))e.setAttribute(n,a);let s=activeDocument.createElementNS(Ja,"rect");s.setAttribute("class","af-convo-toggle-bar");for(let[n,a]of Object.entries({x:"4",y:"5",width:"2",height:"14",rx:"2",fill:"currentColor"}))s.setAttribute(n,a);t.appendChild(e),t.appendChild(s),r.appendChild(t)}var os=class r extends z.ItemView{constructor(e,s){super(e);this.plugin=s}selectedAgentName=null;selectedConversationId="";sessions=new Map;conversationsCache=[];convoPanelEl=null;collapseBtn=null;convoPanelCollapsed=!1;userToggledCollapse=!1;resizeObserver=null;headerEl;agentSelect;messagesEl;messagesInner;textarea;sendBtn;attachStopBtn;isInStopMode=!1;attachedFiles=[];attachedImages=[];pillsRow;activityEl=null;streamingDot=null;activityUnsub=null;statsEl;statsUnsub=null;statsSourceSession=null;threadExpanded=new Map;getViewType(){return rt}getDisplayText(){if(!this.selectedAgentName)return"Agent Chat";let e=this.getCurrentConversationName();return!e||this.conversationsCache.length<=1?`Chat: ${this.selectedAgentName}`:`Chat: ${this.selectedAgentName} \xB7 ${e}`}getIcon(){return"message-circle"}getState(){return{agentName:this.selectedAgentName??null,conversationId:this.selectedConversationId,convoPanelCollapsed:this.convoPanelCollapsed}}async setState(e,s){await super.setState(e,s);let n=typeof e?.conversationId=="string"&&e.conversationId?e.conversationId:void 0;typeof e?.convoPanelCollapsed=="boolean"&&(this.convoPanelCollapsed=e.convoPanelCollapsed,this.userToggledCollapse=!0,this.applyCollapsedClass()),e?.agentName&&typeof e.agentName=="string"&&this.selectAgent(e.agentName,n)}getCurrentConversationName(){return this.conversationsCache.find(s=>s.id===this.selectedConversationId)?.name??""}async onOpen(){this.plugin.subscribeView(this),this.buildShell(),await this.render(),this.observeContainerWidth()}async onClose(){for(let{session:e}of this.sessions.values())e.abort();this.sessions.clear(),this.statsUnsub?.(),this.statsUnsub=null,this.activityUnsub?.(),this.activityUnsub=null,this.resizeObserver?.disconnect(),this.resizeObserver=null,this.plugin.unsubscribeView(this)}observeContainerWidth(){typeof ResizeObserver>"u"||(this.resizeObserver=new ResizeObserver(e=>{if(!this.userToggledCollapse)for(let s of e){let a=s.contentRect.width<Th;a!==this.convoPanelCollapsed&&(this.convoPanelCollapsed=a,this.applyCollapsedClass())}}),this.resizeObserver.observe(this.contentEl))}selectAgent(e,s){if(s){let n=this.plugin.app.workspace.getLeavesOfType(rt);for(let a of n)if(a.view!==this&&a.view instanceof r&&a.view.selectedAgentName===e&&a.view.selectedConversationId===s){this.plugin.app.workspace.revealLeaf(a);return}}this.selectedAgentName=e,s&&(this.selectedConversationId=s),this.agentSelect&&(this.agentSelect.value=e),this.leaf.updateHeader(),this.switchToAgent(e,s)}buildShell(){let e=this.contentEl;e.empty(),e.addClass("af-root");let s=e.createDiv({cls:"af-chat-view-container"});this.headerEl=s.createDiv({cls:"af-chat-view-header"}),this.collapseBtn=this.headerEl.createEl("button",{cls:"clickable-icon af-chat-convo-collapse-btn",attr:{title:"Toggle conversations panel","aria-label":"Toggle conversations panel"}}),_h(this.collapseBtn),this.collapseBtn.toggleClass("is-collapsed",this.convoPanelCollapsed),this.collapseBtn.onclick=()=>this.toggleConvoPanel(),this.agentSelect=this.headerEl.createEl("select",{cls:"af-chat-view-agent-select"});let n=this.headerEl.createEl("button",{cls:"af-btn-sm af-chat-view-new-btn"});R(n,"plus","af-btn-icon"),n.appendText(" New Chat"),n.onclick=()=>void this.handleNewChat();let a=s.createDiv({cls:"af-chat-view-body"});this.convoPanelEl=a.createDiv({cls:"af-chat-convo-panel"}),this.convoPanelCollapsed&&this.convoPanelEl.addClass("collapsed");let i=a.createDiv({cls:"af-chat-main"});this.agentSelect.onchange=()=>{let h=this.agentSelect.value;h&&(this.selectedConversationId="",this.textarea.disabled=!1,this.textarea.placeholder="Message the agent\u2026 (Shift+Enter for newline)",this.switchToAgent(h))},this.messagesEl=i.createDiv({cls:"af-chat-messages"}),this.messagesInner=this.messagesEl.createDiv({cls:"af-chat-messages-inner"});let o=i.createDiv({cls:"af-chat-input-area"});this.pillsRow=o.createDiv({cls:"af-chat-pills-row"}),this.pillsRow.setCssStyles({display:"none"});let c=o.createDiv({cls:"af-chat-input-row"});this.attachStopBtn=c.createEl("button",{cls:"af-chat-attach-btn"}),R(this.attachStopBtn,"plus","af-btn-icon"),this.attachStopBtn.title="Attach active document",this.attachStopBtn.onclick=()=>{this.isInStopMode?this.handleStop():this.attachActiveDocument()},this.textarea=c.createEl("textarea",{cls:"af-chat-input",attr:{placeholder:"Message the agent\u2026 (Shift+Enter for newline)",rows:"1"}}),this.sendBtn=c.createEl("button",{cls:"af-chat-send-btn"}),R(this.sendBtn,"arrow-up","af-btn-icon"),this.sendBtn.setCssStyles({display:"none"});let l=()=>{this.textarea.setCssStyles({height:"auto"});let h=Math.min(this.textarea.scrollHeight,160);this.textarea.setCssStyles({height:`${h}px`}),this.textarea.setCssStyles({overflowY:this.textarea.scrollHeight>160?"auto":"hidden"}),this.sendBtn.setCssStyles({display:this.textarea.value.trim()?"flex":"none"})};this.textarea.addEventListener("input",l),this.textarea.addEventListener("focus",()=>{let h=this.getCurrentSession();h&&this.setStatsSource(h.session)}),this.sendBtn.onclick=()=>void this.handleSend(),this.textarea.onkeydown=h=>{h.key==="Enter"&&!h.shiftKey&&!h.isComposing&&(h.preventDefault(),this.handleSend())},this.textarea.addEventListener("paste",h=>{let d=h.clipboardData?.items;if(d)for(let u=0;u<d.length;u++){let p=d[u];if(p.type.startsWith("image/")){h.preventDefault();let m=p.getAsFile();m&&this.attachImageBlob(m);return}}}),o.addEventListener("dragover",h=>{h.preventDefault(),h.stopPropagation(),o.addClass("af-chat-input-dragover")}),o.addEventListener("dragleave",()=>{o.removeClass("af-chat-input-dragover")}),o.addEventListener("drop",h=>{h.preventDefault(),h.stopPropagation(),o.removeClass("af-chat-input-dragover");let d=h.dataTransfer?.files;if(d)for(let u=0;u<d.length;u++){let p=d[u];p.type.startsWith("image/")&&this.attachImageBlob(p)}}),this.statsEl=o.createDiv({cls:"af-chat-stats"}),this.renderStats(null)}async render(){this.populateAgentDropdown()}setStatsSource(e){this.statsSourceSession!==e&&(this.statsUnsub?.(),this.statsSourceSession=e,this.statsUnsub=e.onStatsChange(s=>this.renderStats(s)))}renderStats(e){if(this.statsEl.empty(),!e||!e.concreteModel){this.statsEl.createSpan({cls:"af-chat-stats-muted",text:"\xA0"});return}let s=this.statsEl.createSpan({cls:"af-chat-stats-line"});if(s.createSpan({cls:"af-chat-stats-model",text:e.concreteModel}),e.contextWindow&&e.contextTokensUsed){let n=Math.min(100,Math.round(e.contextTokensUsed/e.contextWindow*100)),a=10,i=Math.min(a,Math.max(0,Math.round(n/100*a))),o="\u2593".repeat(i)+"\u2591".repeat(a-i),c=s.createSpan({cls:`af-chat-stats-ctx${n>=80?" warn":""}`});c.createSpan({cls:"af-chat-stats-bar",text:o}),c.createSpan({cls:"af-chat-stats-pct",text:`${n}%`}),c.title=`Context: ${is(e.contextTokensUsed)} / ${is(e.contextWindow)} tokens (${n}%)
12126
12037
 
12127
12038
  Includes the agent's system prompt, attached skills, tool schemas, memory, and prior turns. A fresh chat starts non-zero because all of that is loaded on turn 1.
12128
12039
 
12129
- To get 1M context on Opus, set the agent's model to claude-opus-4-7[1m] (or us.anthropic.claude-opus-4-7[1m] on Bedrock).`}if(e.lastCompact){let{preTokens:n,postTokens:a}=e.lastCompact,r=s.createSpan({cls:"af-chat-stats-compact"});r.setText(`compacted ${is(n)} \u2192 ${is(a)}`),r.title=`Conversation was summarized to free up context. ${is(n)} tokens reduced to ${is(a)}.`}}populateAgentDropdown(){let e=this.plugin.runtime.getSnapshot().agents,s=this.agentSelect.value;if(this.agentSelect.empty(),e.length===0){let a=this.agentSelect.createEl("option",{text:"No agents available",attr:{value:"",disabled:"true"}});a.selected=!0,this.textarea.disabled=!0,this.showEmptyState();return}if(!this.selectedAgentName){let a=this.agentSelect.createEl("option",{text:"Select agent\u2026",attr:{value:"",disabled:"true"}});a.selected=!0}for(let a of e){let r=a.avatar?.trim(),o=r&&!/^[a-z][a-z0-9-]*$/.test(r)?`${r} `:"";this.agentSelect.createEl("option",{text:`${o}${a.name}`,attr:{value:a.name}})}if(this.selectedAgentName&&e.some(a=>a.name===this.selectedAgentName))this.agentSelect.value=this.selectedAgentName,this.textarea.disabled=!1;else if(s&&e.some(a=>a.name===s))this.agentSelect.value=s,this.selectedAgentName=s,this.textarea.disabled=!1;else{this.selectedAgentName=null,this.leaf.updateHeader(),this.textarea.disabled=!0,this.textarea.placeholder="Select an agent to start chatting\u2026",this.showEmptyState();return}this.leaf.updateHeader(),this.textarea.placeholder="Message the agent\u2026 (Shift+Enter for newline)";let n=this.messagesInner.querySelector(".af-chat-bubble")!==null;this.selectedAgentName&&!n&&this.switchToAgent(this.selectedAgentName,this.selectedConversationId||void 0)}showEmptyState(){this.messagesInner.empty();let e=this.messagesInner.createDiv({cls:"af-chat-view-empty"}),s=e.createDiv({cls:"af-chat-view-empty-icon"});this.plugin.runtime.getSnapshot().agents.length===0?((0,z.setIcon)(s,"bot"),e.createDiv({cls:"af-chat-view-empty-text",text:"No agents available"}),e.createDiv({cls:"af-chat-view-empty-hint",text:"Create an agent to start chatting"})):((0,z.setIcon)(s,"message-circle"),e.createDiv({cls:"af-chat-view-empty-text",text:"Select an agent to start"}),e.createDiv({cls:"af-chat-view-empty-hint",text:"Choose an agent from the dropdown above"}))}async switchToAgent(e,s){let a=this.plugin.runtime.getSnapshot().agents.find(h=>h.name===e);if(!a)return;await this.loadConversations(a);let r;if(s&&this.conversationsCache.some(h=>h.id===s))r=s;else if(this.conversationsCache.length>0)r=this.conversationsCache[0].id;else{r=el();try{await this.plugin.repository.createConversation(a,r,tl())}catch(h){new z.Notice(`Couldn't create conversation: ${h instanceof Error?h.message:String(h)}`);return}await this.loadConversations(a)}if(!s||s!==r){let h=this.plugin.app.workspace.getLeavesOfType(it);for(let d of h)if(d.view!==this&&d.view instanceof i&&d.view.selectedAgentName===e&&d.view.selectedConversationId===r){this.plugin.app.workspace.revealLeaf(d);return}}this.selectedAgentName=e,this.selectedConversationId=r,this.leaf.updateHeader(),this.activityEl=null,this.streamingDot=null,this.messagesInner.empty(),this.threadExpanded.clear(),this.renderConvoPanel();let o=qn(e,r),c=this.sessions.get(o);if(!c){let h=new Vt(a,this.plugin.settings,this.plugin.repository,this.app.vault,{inAppConversationId:r,mcpAuth:this.plugin.mcpAuth});h.setUsageRecorder(d=>this.plugin.runtime.recordUsage(d)),c={session:h},this.sessions.set(o,c),await h.loadPersistedState()}this.setStatsSource(c.session);for(let h of c.session.messages)if(h.role==="user")this.addBubble("user",h.content,h.attachments);else{let d=this.addBubble("assistant");if(this.renderMarkdownBubble(d,h.content),d._setRawText?.(h.content),this.attachThreadAffordance(d,h.id,c.session),h.toolCalls&&h.toolCalls.length>0){let u=this.getOrCreateAffordancesRow(d);this.buildToolSummary(h.toolCalls,u)}}this.activityUnsub?.();let l=c.session;this.activityUnsub=l.onActivityChange(()=>{this.getCurrentSession()?.session===l&&this.renderIndicators(l)}),this.textarea.disabled=!1,this.textarea.focus()}getCurrentSession(){if(this.selectedAgentName)return this.sessions.get(qn(this.selectedAgentName,this.selectedConversationId))}async loadConversations(e){this.conversationsCache=await this.plugin.repository.listConversations(e)}async refreshConversationsList(e){await this.loadConversations(e),this.renderConvoPanel(),this.leaf.updateHeader()}renderConvoPanel(){if(!this.convoPanelEl||(this.convoPanelEl.empty(),!this.selectedAgentName))return;this.convoPanelEl.createDiv({cls:"af-chat-convo-header",text:"Conversations"});let e=this.convoPanelEl.createDiv({cls:"af-chat-convo-new"}),s=e.createSpan({cls:"af-chat-convo-new-icon"});(0,z.setIcon)(s,"plus"),e.createSpan({cls:"af-chat-convo-new-label",text:"New chat"}),e.onclick=()=>void this.handleNewChat();let n=this.convoPanelEl.createDiv({cls:"af-chat-convo-list"});for(let a of this.conversationsCache)this.renderConvoRow(n,a)}renderConvoRow(e,s){let n=s.id===this.selectedConversationId,a=e.createDiv({cls:`af-chat-convo-item${n?" active":""}`}),r=a.createDiv({cls:"af-chat-convo-name",text:s.name});r.title=s.name,r.ondblclick=h=>{h.stopPropagation(),this.beginInlineRename(r,s)};let o=a.createDiv({cls:"af-chat-convo-meta"}),c=s.messageCount===1?"1 msg":`${s.messageCount} msgs`;o.appendText(`${c} \xB7 ${Sh(s.lastActive)}`);let l=a.createEl("button",{cls:"af-chat-convo-trash",attr:{title:"Delete conversation","aria-label":"Delete conversation"}});(0,z.setIcon)(l,"trash-2"),l.onclick=h=>{h.stopPropagation(),this.handleDeleteConversation(s)},a.onclick=()=>{s.id!==this.selectedConversationId&&this.handleConvoSelected(s.id)}}beginInlineRename(e,s){let n=s.name,a=activeDocument.createElement("input");a.type="text",a.value=n,a.className="af-chat-convo-name-input",e.replaceWith(a),a.focus(),a.select();let r=!1,o=l=>{let h=activeDocument.createElement("div");return h.className="af-chat-convo-name",h.textContent=l??n,h.title=l??n,h.ondblclick=d=>{d.stopPropagation();let u=this.conversationsCache.find(p=>p.id===s.id)??s;this.beginInlineRename(h,u)},a.replaceWith(h),h},c=async()=>{if(r)return;r=!0;let l=a.value.trim();if(!l||l===n){o(null);return}try{await this.saveConversationName(s.id,l)}catch(h){new z.Notice(`Couldn't rename: ${h instanceof Error?h.message:String(h)}`),o(null);return}this.renderConvoPanel(),this.leaf.updateHeader()};a.addEventListener("keydown",l=>{l.key==="Enter"?(l.preventDefault(),c()):l.key==="Escape"&&(l.preventDefault(),r=!0,o(null))}),a.addEventListener("blur",()=>{c()})}async saveConversationName(e,s){if(!this.selectedAgentName)return;let a=this.plugin.runtime.getSnapshot().agents.find(l=>l.name===this.selectedAgentName);if(!a)return;try{await this.plugin.repository.renameConversation(a,e,s)}catch{}let r=qn(this.selectedAgentName,e),o=this.sessions.get(r);o&&await o.session.setConversationName(s);let c=this.conversationsCache.findIndex(l=>l.id===e);c>=0&&(this.conversationsCache[c]={...this.conversationsCache[c],name:s})}async handleConvoSelected(e){if(!this.selectedAgentName)return;let s=this.plugin.app.workspace.getLeavesOfType(it);for(let n of s)if(n.view!==this&&n.view instanceof i&&n.view.selectedAgentName===this.selectedAgentName&&n.view.selectedConversationId===e){this.plugin.app.workspace.revealLeaf(n);return}await this.switchToAgent(this.selectedAgentName,e)}handleDeleteConversation(e){let s=this.selectedAgentName;if(!s)return;let a=this.plugin.runtime.getSnapshot().agents.find(r=>r.name===s);a&&new Ht(this.app,{title:`Delete conversation "${e.name}"?`,body:"This removes its message history. The agent and its other conversations are untouched.",confirmText:"Delete",danger:!0,onConfirm:async()=>{let r=qn(s,e.id);this.sessions.get(r)?.session.dispose(),this.sessions.delete(r);try{await this.plugin.repository.deleteConversation(a,e.id)}catch(o){new z.Notice(`Couldn't delete: ${o instanceof Error?o.message:String(o)}`);return}if(await this.refreshConversationsList(a),e.id===this.selectedConversationId){let o=this.conversationsCache[0]?.id;await this.switchToAgent(s,o)}}}).open()}toggleConvoPanel(){this.convoPanelCollapsed=!this.convoPanelCollapsed,this.userToggledCollapse=!0,this.applyCollapsedClass(),this.leaf.updateHeader()}applyCollapsedClass(){this.convoPanelEl&&(this.convoPanelCollapsed?this.convoPanelEl.addClass("collapsed"):this.convoPanelEl.removeClass("collapsed")),this.collapseBtn&&this.collapseBtn.toggleClass("is-collapsed",this.convoPanelCollapsed)}renderMarkdownBubble(e,s){let n=e.querySelector(".af-chat-copy-btn"),a=n?.parentNode?.removeChild(n)??null;e.empty(),e.addClass("af-compact-md"),z.MarkdownRenderer.render(this.app,s,e,"",this).then(()=>{a&&e.appendChild(a),this.wireBubbleLinks(e),e.querySelectorAll("pre").forEach(r=>{r.querySelector(".copy-code-button")?.remove();let o=r.querySelector("code");if(!o)return;let c=activeDocument.createElement("button");c.className="af-code-copy-btn",c.setAttribute("aria-label","Copy code"),(0,z.setIcon)(c,"copy"),c.onclick=l=>{l.stopPropagation(),navigator.clipboard.writeText(o.textContent??"").then(()=>{c.addClass("copied"),(0,z.setIcon)(c,"check"),window.setTimeout(()=>{c.removeClass("copied"),(0,z.setIcon)(c,"copy")},1500)})},r.setCssStyles({position:"relative"}),r.appendChild(c)})})}wireBubbleLinks(e){e.addEventListener("click",s=>{let n=s.target.closest("a");if(!n)return;if(n.classList.contains("internal-link")){s.preventDefault(),s.stopPropagation();let r=n.getAttribute("data-href")||n.getAttribute("href")||n.textContent||"";if(!r)return;let o=s.shiftKey?"split":"tab";this.app.workspace.openLinkText(r,"",o);return}let a=n.getAttribute("href")||"";if(a.startsWith("obsidian://")){s.preventDefault(),s.stopPropagation(),window.location.href=a;return}if(n.classList.contains("external-link")||/^https?:\/\//.test(a)){s.preventDefault(),s.stopPropagation(),window.open(a,"_blank");return}})}addCopyBtn(e,s){let n=e.createEl("button",{cls:"af-chat-copy-btn",attr:{"aria-label":"Copy message"}});(0,z.setIcon)(n,"copy"),n.onclick=a=>{a.stopPropagation(),navigator.clipboard.writeText(s()).then(()=>{n.addClass("copied"),(0,z.setIcon)(n,"check"),window.setTimeout(()=>{n.removeClass("copied"),(0,z.setIcon)(n,"copy")},1500)})}}addBubble(e,s,n){if(e==="user"&&n&&n.length>0){let r=this.messagesInner.createDiv({cls:"af-chat-bubble-attachments"});for(let o of n){let c=r.createSpan({cls:"af-chat-pill af-chat-pill-inline"}),l=c.createSpan({cls:"af-chat-pill-icon"}),h=/\.(png|jpe?g|gif|webp|svg|bmp)$/i.test(o);(0,z.setIcon)(l,h?"image":"file-text"),c.createSpan({cls:"af-chat-pill-name",text:o})}}let a=this.messagesInner.createDiv({cls:`af-chat-bubble af-chat-bubble-${e}`});if(s&&(e==="assistant"?this.renderMarkdownBubble(a,s):a.setText(s)),e==="assistant"){let r=s??"";this.addCopyBtn(a,()=>r),a._setRawText=o=>{r=o}}return this.messagesEl.scrollTop=this.messagesEl.scrollHeight,a}getOrCreateAffordancesRow(e){let s=e.parentElement;if(!s)throw new Error("bubble has no parent");let n=e.nextElementSibling;if(n&&n.classList.contains("af-chat-affordances"))return n;let a=activeDocument.createElement("div");return a.className="af-chat-affordances",s.insertBefore(a,e.nextSibling),a}attachThreadAffordance(e,s,n){let a=e.parentElement;if(!a)return;let r=this.getOrCreateAffordancesRow(e);if(r.querySelector(".af-thread-badge"))return;let o=activeDocument.createElement("div");o.className="af-thread-badge",o.setAttribute("role","button"),o.setAttribute("tabindex","0"),r.appendChild(o),(0,z.setIcon)(o,"message-circle");let c=o.createSpan({cls:"af-thread-badge-label"}),l=activeDocument.createElement("div");l.className="af-thread-container",l.setCssStyles({display:"none"}),a.insertBefore(l,r.nextSibling);let h=()=>{let f=n.getThreadIndex()[s]?.messageCount??0;f<=0?c.setText("Thread"):c.setText(`${f} ${f===1?"reply":"replies"}`),f>0&&o.addClass("has-replies")};h();let d=!1,u=async()=>{if(this.threadExpanded.get(s)===!0){l.setCssStyles({display:"none"}),this.threadExpanded.set(s,!1),o.removeClass("expanded"),this.setStatsSource(n);return}if(this.threadExpanded.set(s,!0),l.setCssStyles({display:""}),o.addClass("expanded"),!d)try{let m=await n.openOrCreateThread(s);this.renderThreadContainer(l,m,n,h),d=!0}catch(m){l.setText(`Failed to open thread: ${m instanceof Error?m.message:String(m)}`)}};o.onclick=()=>void u(),o.onkeydown=p=>{(p.key==="Enter"||p.key===" ")&&(p.preventDefault(),u())}}renderThreadContainer(e,s,n,a){e.empty();let r=e.createDiv({cls:"af-thread-wrap"}),o=r.createDiv({cls:"af-thread-messages"}),c=null,l=null,h=C=>{C?(c||(c=o.createDiv({cls:"af-chat-activity"})),c.setText(`Working\u2026 (${C})`)):c&&(c.remove(),c=null)},d=C=>{if(C&&!l){l=o.createDiv({cls:"af-chat-streaming-dot"});for(let E=0;E<3;E++)l.createSpan()}else!C&&l&&(l.remove(),l=null)},u=(C,E,P)=>{if(C==="user"&&P&&P.length>0){let M=o.createDiv({cls:"af-chat-bubble-attachments"});for(let U of P){let $=M.createSpan({cls:"af-chat-pill af-chat-pill-inline"}),Z=$.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(Z,U.match(/\.(png|jpe?g|gif|webp|svg)$/i)?"image":"file-text"),$.createSpan({cls:"af-chat-pill-name",text:U})}}let N=o.createDiv({cls:`af-thread-bubble af-thread-bubble-${C}`});return C==="assistant"?this.renderMarkdownBubble(N,E):N.setText(E),N};for(let C of s.messages)u(C.role,C.content,C.attachments);let p=[],m=[],f=r.createDiv({cls:"af-thread-composer-wrap"}),g=f.createDiv({cls:"af-chat-pills-row af-thread-pills-row"});g.setCssStyles({display:"none"});let y=()=>{if(g.empty(),p.length===0&&m.length===0){g.setCssStyles({display:"none"});return}g.setCssStyles({display:"flex"});for(let C of p){let E=g.createDiv({cls:"af-chat-pill"}),P=E.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(P,"file-text"),E.createSpan({cls:"af-chat-pill-name",text:C.name});let N=E.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(N,"x"),N.onclick=M=>{M.stopPropagation();let U=p.findIndex($=>$.path===C.path);U>=0&&p.splice(U,1),y()}}for(let C of m){let E=g.createDiv({cls:"af-chat-pill"}),P=E.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(P,"image"),E.createSpan({cls:"af-chat-pill-name",text:C.name});let N=E.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(N,"x"),N.onclick=M=>{M.stopPropagation();let U=m.findIndex($=>$.path===C.path);U>=0&&m.splice(U,1),y()}}},v=()=>{let C=this.app.workspace.getActiveFile();if(!C){new z.Notice("No active document to attach");return}if(p.some(P=>P.path===C.path)){new z.Notice(`"${C.name}" is already attached`);return}if(!new Set(["md","txt","json","yaml","yml","toml","csv","xml","html","css","js","ts","py","sh","sql","env","cfg","ini","log"]).has(C.extension.toLowerCase())){new z.Notice(`Can't attach "${C.name}" \u2014 only text files are supported`);return}p.push(C),y()},k=async C=>{let E=C.type.split("/")[1]?.replace("jpeg","jpg")??"png",P=C.name&&C.name!=="image"?C.name:`pasted-${Date.now()}.${E}`;if(m.some(M=>M.name===P)){new z.Notice(`"${P}" is already attached`);return}let N=await this.saveImageBlobToVault(C);N&&(m.push(N),y())},w=f.createDiv({cls:"af-chat-input-row af-thread-composer"}),S=w.createEl("button",{cls:"af-chat-attach-btn"});R(S,"plus","af-btn-icon"),S.title="Attach active document",S.onclick=C=>{C.preventDefault(),v()};let T=w.createEl("textarea",{cls:"af-chat-input af-thread-input",attr:{placeholder:"Message in thread\u2026 (Shift+Enter for newline)",rows:"1"}});T.addEventListener("paste",C=>{let E=C.clipboardData?.items;if(E)for(let P=0;P<E.length;P++){let N=E[P];if(N.type.startsWith("image/")){C.preventDefault();let M=N.getAsFile();M&&k(M);return}}}),f.addEventListener("dragover",C=>{C.preventDefault(),C.stopPropagation(),f.addClass("af-chat-input-dragover")}),f.addEventListener("dragleave",()=>{f.removeClass("af-chat-input-dragover")}),f.addEventListener("drop",C=>{C.preventDefault(),C.stopPropagation(),f.removeClass("af-chat-input-dragover");let E=C.dataTransfer?.files;if(E)for(let P=0;P<E.length;P++){let N=E[P];N.type.startsWith("image/")&&k(N)}});let _=w.createEl("button",{cls:"af-chat-send-btn"});R(_,"arrow-up","af-btn-icon"),_.setCssStyles({display:"none"});let D=()=>{T.setCssStyles({height:"auto"}),T.setCssStyles({height:`${Math.min(T.scrollHeight,120)}px`}),_.setCssStyles({display:T.value.trim()?"flex":"none"})};T.addEventListener("input",D),T.addEventListener("focus",()=>this.setStatsSource(s));let O=async()=>{let C=T.value.trim();if(!C||s.isStreaming)return;let E=await this.buildAttachmentContextFor(p,m),P=[...p.map($=>$.name),...m.map($=>$.name)],N=E?`${E}${C}`:void 0;T.value="",D(),p.length=0,m.length=0,y(),u("user",C,P.length>0?P:void 0),d(!0);let M=null,U="";try{await s.sendMessage(C,$=>{if($.type==="text"){M||(d(!1),h(),M=u("assistant",""),M.empty()),U+=$.content;let Z=M.querySelector(".af-chat-stream-text");Z||(Z=M.createDiv({cls:"af-chat-stream-text"})),Z.setText(U)}else $.type==="tool_use"?h($.toolName):$.type==="result"&&(h(),d(!1),M&&this.renderMarkdownBubble(M,hs(U)))},N,P.length>0?P:void 0),a()}catch($){d(!1),h();let Z=$ instanceof Error?$.message:String($);o.createDiv({cls:"af-thread-error",text:`Error: ${Z}`})}};_.onclick=()=>void O(),T.onkeydown=C=>{C.key==="Enter"&&!C.shiftKey&&!C.isComposing&&(C.preventDefault(),O())}}buildToolSummary(e,s){let a=(s??this.messagesInner).createDiv({cls:"af-chat-tool-summary"}),r=a.createEl("details"),o=r.createEl("summary"),c=new Map;for(let d of e){let u=c.get(d.name)??[];d.command&&u.push(d.command),c.set(d.name,u)}let l=o.createSpan({cls:"af-chat-tool-icon"});(0,z.setIcon)(l,"wrench"),o.appendText(` ${e.length} tool call${e.length!==1?"s":""}`);let h=r.createDiv({cls:"af-chat-tool-list"});for(let[d,u]of c){let p=u.length||(c.get(d)?.length??1),m=h.createDiv({cls:"af-chat-tool-item"}),f=p>1?`${d} (\xD7${p})`:d;m.createSpan({cls:"af-chat-tool-name",text:f}),u.length===1&&u[0]&&m.createSpan({cls:"af-chat-tool-cmd",text:u[0]})}return a}renderIndicators(e){let s=e.isStreaming,n=e.currentToolName,a=e.hasCurrentTurnText,r=!!this.messagesInner.querySelector(".af-chat-stream-text"),o=null;if(s&&n?o=`Working\u2026 (${n})`:s&&a&&!r&&(o="Replying\u2026"),o?(this.activityEl?(this.activityEl.parentElement!==this.messagesInner||this.activityEl.nextElementSibling!==null)&&this.messagesInner.appendChild(this.activityEl):this.activityEl=this.messagesInner.createDiv({cls:"af-chat-activity"}),this.activityEl.textContent!==o&&this.activityEl.setText(o)):this.activityEl&&(this.activityEl.remove(),this.activityEl=null),s&&!n&&!a)if(this.streamingDot)(this.streamingDot.parentElement!==this.messagesInner||this.streamingDot.nextElementSibling!==null)&&this.messagesInner.appendChild(this.streamingDot);else{this.streamingDot=this.messagesInner.createDiv({cls:"af-chat-streaming-dot"});for(let l=0;l<3;l++)this.streamingDot.createSpan()}else this.streamingDot&&(this.streamingDot.remove(),this.streamingDot=null);this.setAttachStopMode(s)}setAttachStopMode(e){e!==this.isInStopMode&&(this.isInStopMode=e,this.attachStopBtn.empty(),e?(R(this.attachStopBtn,"square","af-btn-icon"),this.attachStopBtn.title="Stop generation",this.attachStopBtn.addClass("af-chat-stop-mode")):(R(this.attachStopBtn,"plus","af-btn-icon"),this.attachStopBtn.title="Attach active document",this.attachStopBtn.removeClass("af-chat-stop-mode")))}handleStop(){let e=this.getCurrentSession();e&&(e.session.abort(),this.addBubble("error","Generation stopped"))}attachActiveDocument(){let e=this.app.workspace.getActiveFile();if(!e){new z.Notice("No active document to attach");return}if(this.attachedFiles.some(a=>a.path===e.path)){new z.Notice(`"${e.name}" is already attached`);return}let s=e.extension.toLowerCase();if(!new Set(["md","txt","json","yaml","yml","toml","csv","xml","html","css","js","ts","py","sh","sql","env","cfg","ini","log"]).has(s)){new z.Notice(`Can't attach "${e.name}" \u2014 only text files are supported`);return}this.attachedFiles.push(e),this.renderPills()}async saveImageBlobToVault(e){let s=e.type.split("/")[1]?.replace("jpeg","jpg")??"png",n=Date.now(),a=e.name&&e.name!=="image"?e.name:`pasted-${n}.${s}`,r=`${this.plugin.settings.fleetFolder}/chat-images`,o=`${r}/${n}-${a}`;try{this.app.vault.getAbstractFileByPath(r)||await this.app.vault.createFolder(r);let c=await e.arrayBuffer();await this.app.vault.createBinary(o,c);let h=this.app.vault.adapter.getBasePath?.()??"";return{name:a,path:`${h}/${o}`}}catch(c){let l=c instanceof Error?c.message:String(c);return new z.Notice(`Failed to save image: ${l}`),null}}async buildAttachmentContextFor(e,s){if(e.length===0&&s.length===0)return"";let n=[];for(let a of e)try{let r=await this.app.vault.cachedRead(a);n.push(`### ${a.name}
12040
+ To get 1M context on Opus, set the agent's model to claude-opus-4-7[1m] (or us.anthropic.claude-opus-4-7[1m] on Bedrock).`}if(e.lastCompact){let{preTokens:n,postTokens:a}=e.lastCompact,i=s.createSpan({cls:"af-chat-stats-compact"});i.setText(`compacted ${is(n)} \u2192 ${is(a)}`),i.title=`Conversation was summarized to free up context. ${is(n)} tokens reduced to ${is(a)}.`}}populateAgentDropdown(){let e=this.plugin.runtime.getSnapshot().agents,s=this.agentSelect.value;if(this.agentSelect.empty(),e.length===0){let a=this.agentSelect.createEl("option",{text:"No agents available",attr:{value:"",disabled:"true"}});a.selected=!0,this.textarea.disabled=!0,this.showEmptyState();return}if(!this.selectedAgentName){let a=this.agentSelect.createEl("option",{text:"Select agent\u2026",attr:{value:"",disabled:"true"}});a.selected=!0}for(let a of e){let i=a.avatar?.trim(),o=i&&!/^[a-z][a-z0-9-]*$/.test(i)?`${i} `:"";this.agentSelect.createEl("option",{text:`${o}${a.name}`,attr:{value:a.name}})}if(this.selectedAgentName&&e.some(a=>a.name===this.selectedAgentName))this.agentSelect.value=this.selectedAgentName,this.textarea.disabled=!1;else if(s&&e.some(a=>a.name===s))this.agentSelect.value=s,this.selectedAgentName=s,this.textarea.disabled=!1;else{this.selectedAgentName=null,this.leaf.updateHeader(),this.textarea.disabled=!0,this.textarea.placeholder="Select an agent to start chatting\u2026",this.showEmptyState();return}this.leaf.updateHeader(),this.textarea.placeholder="Message the agent\u2026 (Shift+Enter for newline)";let n=this.messagesInner.querySelector(".af-chat-bubble")!==null;this.selectedAgentName&&!n&&this.switchToAgent(this.selectedAgentName,this.selectedConversationId||void 0)}showEmptyState(){this.messagesInner.empty();let e=this.messagesInner.createDiv({cls:"af-chat-view-empty"}),s=e.createDiv({cls:"af-chat-view-empty-icon"});this.plugin.runtime.getSnapshot().agents.length===0?((0,z.setIcon)(s,"bot"),e.createDiv({cls:"af-chat-view-empty-text",text:"No agents available"}),e.createDiv({cls:"af-chat-view-empty-hint",text:"Create an agent to start chatting"})):((0,z.setIcon)(s,"message-circle"),e.createDiv({cls:"af-chat-view-empty-text",text:"Select an agent to start"}),e.createDiv({cls:"af-chat-view-empty-hint",text:"Choose an agent from the dropdown above"}))}async switchToAgent(e,s){let a=this.plugin.runtime.getSnapshot().agents.find(h=>h.name===e);if(!a)return;await this.loadConversations(a);let i;if(s&&this.conversationsCache.some(h=>h.id===s))i=s;else if(this.conversationsCache.length>0)i=this.conversationsCache[0].id;else{i=tl();try{await this.plugin.repository.createConversation(a,i,sl())}catch(h){new z.Notice(`Couldn't create conversation: ${h instanceof Error?h.message:String(h)}`);return}await this.loadConversations(a)}if(!s||s!==i){let h=this.plugin.app.workspace.getLeavesOfType(rt);for(let d of h)if(d.view!==this&&d.view instanceof r&&d.view.selectedAgentName===e&&d.view.selectedConversationId===i){this.plugin.app.workspace.revealLeaf(d);return}}this.selectedAgentName=e,this.selectedConversationId=i,this.leaf.updateHeader(),this.activityEl=null,this.streamingDot=null,this.messagesInner.empty(),this.threadExpanded.clear(),this.renderConvoPanel();let o=qn(e,i),c=this.sessions.get(o);if(!c){let h=new Yt(a,this.plugin.settings,this.plugin.repository,this.app.vault,{inAppConversationId:i,mcpAuth:this.plugin.mcpAuth});h.setUsageRecorder(d=>this.plugin.runtime.recordUsage(d)),c={session:h},this.sessions.set(o,c),await h.loadPersistedState()}this.setStatsSource(c.session);for(let h of c.session.messages)if(h.role==="user")this.addBubble("user",h.content,h.attachments);else{let d=this.addBubble("assistant");if(this.renderMarkdownBubble(d,h.content),d._setRawText?.(h.content),this.attachThreadAffordance(d,h.id,c.session),h.toolCalls&&h.toolCalls.length>0){let u=this.getOrCreateAffordancesRow(d);this.buildToolSummary(h.toolCalls,u)}}this.activityUnsub?.();let l=c.session;this.activityUnsub=l.onActivityChange(()=>{this.getCurrentSession()?.session===l&&this.renderIndicators(l)}),this.textarea.disabled=!1,this.textarea.focus()}getCurrentSession(){if(this.selectedAgentName)return this.sessions.get(qn(this.selectedAgentName,this.selectedConversationId))}async loadConversations(e){this.conversationsCache=await this.plugin.repository.listConversations(e)}async refreshConversationsList(e){await this.loadConversations(e),this.renderConvoPanel(),this.leaf.updateHeader()}renderConvoPanel(){if(!this.convoPanelEl||(this.convoPanelEl.empty(),!this.selectedAgentName))return;this.convoPanelEl.createDiv({cls:"af-chat-convo-header",text:"Conversations"});let e=this.convoPanelEl.createDiv({cls:"af-chat-convo-new"}),s=e.createSpan({cls:"af-chat-convo-new-icon"});(0,z.setIcon)(s,"plus"),e.createSpan({cls:"af-chat-convo-new-label",text:"New chat"}),e.onclick=()=>void this.handleNewChat();let n=this.convoPanelEl.createDiv({cls:"af-chat-convo-list"});for(let a of this.conversationsCache)this.renderConvoRow(n,a)}renderConvoRow(e,s){let n=s.id===this.selectedConversationId,a=e.createDiv({cls:`af-chat-convo-item${n?" active":""}`}),i=a.createDiv({cls:"af-chat-convo-name",text:s.name});i.title=s.name,i.ondblclick=h=>{h.stopPropagation(),this.beginInlineRename(i,s)};let o=a.createDiv({cls:"af-chat-convo-meta"}),c=s.messageCount===1?"1 msg":`${s.messageCount} msgs`;o.appendText(`${c} \xB7 ${Ch(s.lastActive)}`);let l=a.createEl("button",{cls:"af-chat-convo-trash",attr:{title:"Delete conversation","aria-label":"Delete conversation"}});(0,z.setIcon)(l,"trash-2"),l.onclick=h=>{h.stopPropagation(),this.handleDeleteConversation(s)},a.onclick=()=>{s.id!==this.selectedConversationId&&this.handleConvoSelected(s.id)}}beginInlineRename(e,s){let n=s.name,a=activeDocument.createElement("input");a.type="text",a.value=n,a.className="af-chat-convo-name-input",e.replaceWith(a),a.focus(),a.select();let i=!1,o=l=>{let h=activeDocument.createElement("div");return h.className="af-chat-convo-name",h.textContent=l??n,h.title=l??n,h.ondblclick=d=>{d.stopPropagation();let u=this.conversationsCache.find(p=>p.id===s.id)??s;this.beginInlineRename(h,u)},a.replaceWith(h),h},c=async()=>{if(i)return;i=!0;let l=a.value.trim();if(!l||l===n){o(null);return}try{await this.saveConversationName(s.id,l)}catch(h){new z.Notice(`Couldn't rename: ${h instanceof Error?h.message:String(h)}`),o(null);return}this.renderConvoPanel(),this.leaf.updateHeader()};a.addEventListener("keydown",l=>{l.key==="Enter"?(l.preventDefault(),c()):l.key==="Escape"&&(l.preventDefault(),i=!0,o(null))}),a.addEventListener("blur",()=>{c()})}async saveConversationName(e,s){if(!this.selectedAgentName)return;let a=this.plugin.runtime.getSnapshot().agents.find(l=>l.name===this.selectedAgentName);if(!a)return;try{await this.plugin.repository.renameConversation(a,e,s)}catch{}let i=qn(this.selectedAgentName,e),o=this.sessions.get(i);o&&await o.session.setConversationName(s);let c=this.conversationsCache.findIndex(l=>l.id===e);c>=0&&(this.conversationsCache[c]={...this.conversationsCache[c],name:s})}async handleConvoSelected(e){if(!this.selectedAgentName)return;let s=this.plugin.app.workspace.getLeavesOfType(rt);for(let n of s)if(n.view!==this&&n.view instanceof r&&n.view.selectedAgentName===this.selectedAgentName&&n.view.selectedConversationId===e){this.plugin.app.workspace.revealLeaf(n);return}await this.switchToAgent(this.selectedAgentName,e)}handleDeleteConversation(e){let s=this.selectedAgentName;if(!s)return;let a=this.plugin.runtime.getSnapshot().agents.find(i=>i.name===s);a&&new qt(this.app,{title:`Delete conversation "${e.name}"?`,body:"This removes its message history. The agent and its other conversations are untouched.",confirmText:"Delete",danger:!0,onConfirm:async()=>{let i=qn(s,e.id);this.sessions.get(i)?.session.dispose(),this.sessions.delete(i);try{await this.plugin.repository.deleteConversation(a,e.id)}catch(o){new z.Notice(`Couldn't delete: ${o instanceof Error?o.message:String(o)}`);return}if(await this.refreshConversationsList(a),e.id===this.selectedConversationId){let o=this.conversationsCache[0]?.id;await this.switchToAgent(s,o)}}}).open()}toggleConvoPanel(){this.convoPanelCollapsed=!this.convoPanelCollapsed,this.userToggledCollapse=!0,this.applyCollapsedClass(),this.leaf.updateHeader()}applyCollapsedClass(){this.convoPanelEl&&(this.convoPanelCollapsed?this.convoPanelEl.addClass("collapsed"):this.convoPanelEl.removeClass("collapsed")),this.collapseBtn&&this.collapseBtn.toggleClass("is-collapsed",this.convoPanelCollapsed)}renderMarkdownBubble(e,s){let n=e.querySelector(".af-chat-copy-btn"),a=n?.parentNode?.removeChild(n)??null;e.empty(),e.addClass("af-compact-md"),z.MarkdownRenderer.render(this.app,s,e,"",this).then(()=>{a&&e.appendChild(a),this.wireBubbleLinks(e),e.querySelectorAll("pre").forEach(i=>{i.querySelector(".copy-code-button")?.remove();let o=i.querySelector("code");if(!o)return;let c=activeDocument.createElement("button");c.className="af-code-copy-btn",c.setAttribute("aria-label","Copy code"),(0,z.setIcon)(c,"copy"),c.onclick=l=>{l.stopPropagation(),navigator.clipboard.writeText(o.textContent??"").then(()=>{c.addClass("copied"),(0,z.setIcon)(c,"check"),window.setTimeout(()=>{c.removeClass("copied"),(0,z.setIcon)(c,"copy")},1500)})},i.setCssStyles({position:"relative"}),i.appendChild(c)})})}wireBubbleLinks(e){e.addEventListener("click",s=>{let n=s.target.closest("a");if(!n)return;if(n.classList.contains("internal-link")){s.preventDefault(),s.stopPropagation();let i=n.getAttribute("data-href")||n.getAttribute("href")||n.textContent||"";if(!i)return;let o=s.shiftKey?"split":"tab";this.app.workspace.openLinkText(i,"",o);return}let a=n.getAttribute("href")||"";if(a.startsWith("obsidian://")){s.preventDefault(),s.stopPropagation(),window.location.href=a;return}if(n.classList.contains("external-link")||/^https?:\/\//.test(a)){s.preventDefault(),s.stopPropagation(),window.open(a,"_blank");return}})}addCopyBtn(e,s){let n=e.createEl("button",{cls:"af-chat-copy-btn",attr:{"aria-label":"Copy message"}});(0,z.setIcon)(n,"copy"),n.onclick=a=>{a.stopPropagation(),navigator.clipboard.writeText(s()).then(()=>{n.addClass("copied"),(0,z.setIcon)(n,"check"),window.setTimeout(()=>{n.removeClass("copied"),(0,z.setIcon)(n,"copy")},1500)})}}addBubble(e,s,n){if(e==="user"&&n&&n.length>0){let i=this.messagesInner.createDiv({cls:"af-chat-bubble-attachments"});for(let o of n){let c=i.createSpan({cls:"af-chat-pill af-chat-pill-inline"}),l=c.createSpan({cls:"af-chat-pill-icon"}),h=/\.(png|jpe?g|gif|webp|svg|bmp)$/i.test(o);(0,z.setIcon)(l,h?"image":"file-text"),c.createSpan({cls:"af-chat-pill-name",text:o})}}let a=this.messagesInner.createDiv({cls:`af-chat-bubble af-chat-bubble-${e}`});if(s&&(e==="assistant"?this.renderMarkdownBubble(a,s):a.setText(s)),e==="assistant"){let i=s??"";this.addCopyBtn(a,()=>i),a._setRawText=o=>{i=o}}return this.messagesEl.scrollTop=this.messagesEl.scrollHeight,a}getOrCreateAffordancesRow(e){let s=e.parentElement;if(!s)throw new Error("bubble has no parent");let n=e.nextElementSibling;if(n&&n.classList.contains("af-chat-affordances"))return n;let a=activeDocument.createElement("div");return a.className="af-chat-affordances",s.insertBefore(a,e.nextSibling),a}attachThreadAffordance(e,s,n){let a=e.parentElement;if(!a)return;let i=this.getOrCreateAffordancesRow(e);if(i.querySelector(".af-thread-badge"))return;let o=activeDocument.createElement("div");o.className="af-thread-badge",o.setAttribute("role","button"),o.setAttribute("tabindex","0"),i.appendChild(o),(0,z.setIcon)(o,"message-circle");let c=o.createSpan({cls:"af-thread-badge-label"}),l=activeDocument.createElement("div");l.className="af-thread-container",l.setCssStyles({display:"none"}),a.insertBefore(l,i.nextSibling);let h=()=>{let f=n.getThreadIndex()[s]?.messageCount??0;f<=0?c.setText("Thread"):c.setText(`${f} ${f===1?"reply":"replies"}`),f>0&&o.addClass("has-replies")};h();let d=!1,u=async()=>{if(this.threadExpanded.get(s)===!0){l.setCssStyles({display:"none"}),this.threadExpanded.set(s,!1),o.removeClass("expanded"),this.setStatsSource(n);return}if(this.threadExpanded.set(s,!0),l.setCssStyles({display:""}),o.addClass("expanded"),!d)try{let m=await n.openOrCreateThread(s);this.renderThreadContainer(l,m,n,h),d=!0}catch(m){l.setText(`Failed to open thread: ${m instanceof Error?m.message:String(m)}`)}};o.onclick=()=>void u(),o.onkeydown=p=>{(p.key==="Enter"||p.key===" ")&&(p.preventDefault(),u())}}renderThreadContainer(e,s,n,a){e.empty();let i=e.createDiv({cls:"af-thread-wrap"}),o=i.createDiv({cls:"af-thread-messages"}),c=null,l=null,h=C=>{C?(c||(c=o.createDiv({cls:"af-chat-activity"})),c.setText(`Working\u2026 (${C})`)):c&&(c.remove(),c=null)},d=C=>{if(C&&!l){l=o.createDiv({cls:"af-chat-streaming-dot"});for(let E=0;E<3;E++)l.createSpan()}else!C&&l&&(l.remove(),l=null)},u=(C,E,P)=>{if(C==="user"&&P&&P.length>0){let M=o.createDiv({cls:"af-chat-bubble-attachments"});for(let U of P){let $=M.createSpan({cls:"af-chat-pill af-chat-pill-inline"}),Z=$.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(Z,U.match(/\.(png|jpe?g|gif|webp|svg)$/i)?"image":"file-text"),$.createSpan({cls:"af-chat-pill-name",text:U})}}let N=o.createDiv({cls:`af-thread-bubble af-thread-bubble-${C}`});return C==="assistant"?this.renderMarkdownBubble(N,E):N.setText(E),N};for(let C of s.messages)u(C.role,C.content,C.attachments);let p=[],m=[],f=i.createDiv({cls:"af-thread-composer-wrap"}),g=f.createDiv({cls:"af-chat-pills-row af-thread-pills-row"});g.setCssStyles({display:"none"});let y=()=>{if(g.empty(),p.length===0&&m.length===0){g.setCssStyles({display:"none"});return}g.setCssStyles({display:"flex"});for(let C of p){let E=g.createDiv({cls:"af-chat-pill"}),P=E.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(P,"file-text"),E.createSpan({cls:"af-chat-pill-name",text:C.name});let N=E.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(N,"x"),N.onclick=M=>{M.stopPropagation();let U=p.findIndex($=>$.path===C.path);U>=0&&p.splice(U,1),y()}}for(let C of m){let E=g.createDiv({cls:"af-chat-pill"}),P=E.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(P,"image"),E.createSpan({cls:"af-chat-pill-name",text:C.name});let N=E.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(N,"x"),N.onclick=M=>{M.stopPropagation();let U=m.findIndex($=>$.path===C.path);U>=0&&m.splice(U,1),y()}}},v=()=>{let C=this.app.workspace.getActiveFile();if(!C){new z.Notice("No active document to attach");return}if(p.some(P=>P.path===C.path)){new z.Notice(`"${C.name}" is already attached`);return}if(!new Set(["md","txt","json","yaml","yml","toml","csv","xml","html","css","js","ts","py","sh","sql","env","cfg","ini","log"]).has(C.extension.toLowerCase())){new z.Notice(`Can't attach "${C.name}" \u2014 only text files are supported`);return}p.push(C),y()},k=async C=>{let E=C.type.split("/")[1]?.replace("jpeg","jpg")??"png",P=C.name&&C.name!=="image"?C.name:`pasted-${Date.now()}.${E}`;if(m.some(M=>M.name===P)){new z.Notice(`"${P}" is already attached`);return}let N=await this.saveImageBlobToVault(C);N&&(m.push(N),y())},w=f.createDiv({cls:"af-chat-input-row af-thread-composer"}),S=w.createEl("button",{cls:"af-chat-attach-btn"});R(S,"plus","af-btn-icon"),S.title="Attach active document",S.onclick=C=>{C.preventDefault(),v()};let T=w.createEl("textarea",{cls:"af-chat-input af-thread-input",attr:{placeholder:"Message in thread\u2026 (Shift+Enter for newline)",rows:"1"}});T.addEventListener("paste",C=>{let E=C.clipboardData?.items;if(E)for(let P=0;P<E.length;P++){let N=E[P];if(N.type.startsWith("image/")){C.preventDefault();let M=N.getAsFile();M&&k(M);return}}}),f.addEventListener("dragover",C=>{C.preventDefault(),C.stopPropagation(),f.addClass("af-chat-input-dragover")}),f.addEventListener("dragleave",()=>{f.removeClass("af-chat-input-dragover")}),f.addEventListener("drop",C=>{C.preventDefault(),C.stopPropagation(),f.removeClass("af-chat-input-dragover");let E=C.dataTransfer?.files;if(E)for(let P=0;P<E.length;P++){let N=E[P];N.type.startsWith("image/")&&k(N)}});let _=w.createEl("button",{cls:"af-chat-send-btn"});R(_,"arrow-up","af-btn-icon"),_.setCssStyles({display:"none"});let D=()=>{T.setCssStyles({height:"auto"}),T.setCssStyles({height:`${Math.min(T.scrollHeight,120)}px`}),_.setCssStyles({display:T.value.trim()?"flex":"none"})};T.addEventListener("input",D),T.addEventListener("focus",()=>this.setStatsSource(s));let O=async()=>{let C=T.value.trim();if(!C||s.isStreaming)return;let E=await this.buildAttachmentContextFor(p,m),P=[...p.map($=>$.name),...m.map($=>$.name)],N=E?`${E}${C}`:void 0;T.value="",D(),p.length=0,m.length=0,y(),u("user",C,P.length>0?P:void 0),d(!0);let M=null,U="";try{await s.sendMessage(C,$=>{if($.type==="text"){M||(d(!1),h(),M=u("assistant",""),M.empty()),U+=$.content;let Z=M.querySelector(".af-chat-stream-text");Z||(Z=M.createDiv({cls:"af-chat-stream-text"})),Z.setText(U)}else $.type==="tool_use"?h($.toolName):$.type==="result"&&(h(),d(!1),M&&this.renderMarkdownBubble(M,At(U)))},N,P.length>0?P:void 0),a()}catch($){d(!1),h();let Z=$ instanceof Error?$.message:String($);o.createDiv({cls:"af-thread-error",text:`Error: ${Z}`})}};_.onclick=()=>void O(),T.onkeydown=C=>{C.key==="Enter"&&!C.shiftKey&&!C.isComposing&&(C.preventDefault(),O())}}buildToolSummary(e,s){let a=(s??this.messagesInner).createDiv({cls:"af-chat-tool-summary"}),i=a.createEl("details"),o=i.createEl("summary"),c=new Map;for(let d of e){let u=c.get(d.name)??[];d.command&&u.push(d.command),c.set(d.name,u)}let l=o.createSpan({cls:"af-chat-tool-icon"});(0,z.setIcon)(l,"wrench"),o.appendText(` ${e.length} tool call${e.length!==1?"s":""}`);let h=i.createDiv({cls:"af-chat-tool-list"});for(let[d,u]of c){let p=u.length||(c.get(d)?.length??1),m=h.createDiv({cls:"af-chat-tool-item"}),f=p>1?`${d} (\xD7${p})`:d;m.createSpan({cls:"af-chat-tool-name",text:f}),u.length===1&&u[0]&&m.createSpan({cls:"af-chat-tool-cmd",text:u[0]})}return a}renderIndicators(e){let s=e.isStreaming,n=e.currentToolName,a=e.hasCurrentTurnText,i=!!this.messagesInner.querySelector(".af-chat-stream-text"),o=null;if(s&&n?o=`Working\u2026 (${n})`:s&&a&&!i&&(o="Replying\u2026"),o?(this.activityEl?(this.activityEl.parentElement!==this.messagesInner||this.activityEl.nextElementSibling!==null)&&this.messagesInner.appendChild(this.activityEl):this.activityEl=this.messagesInner.createDiv({cls:"af-chat-activity"}),this.activityEl.textContent!==o&&this.activityEl.setText(o)):this.activityEl&&(this.activityEl.remove(),this.activityEl=null),s&&!n&&!a)if(this.streamingDot)(this.streamingDot.parentElement!==this.messagesInner||this.streamingDot.nextElementSibling!==null)&&this.messagesInner.appendChild(this.streamingDot);else{this.streamingDot=this.messagesInner.createDiv({cls:"af-chat-streaming-dot"});for(let l=0;l<3;l++)this.streamingDot.createSpan()}else this.streamingDot&&(this.streamingDot.remove(),this.streamingDot=null);this.setAttachStopMode(s)}setAttachStopMode(e){e!==this.isInStopMode&&(this.isInStopMode=e,this.attachStopBtn.empty(),e?(R(this.attachStopBtn,"square","af-btn-icon"),this.attachStopBtn.title="Stop generation",this.attachStopBtn.addClass("af-chat-stop-mode")):(R(this.attachStopBtn,"plus","af-btn-icon"),this.attachStopBtn.title="Attach active document",this.attachStopBtn.removeClass("af-chat-stop-mode")))}handleStop(){let e=this.getCurrentSession();e&&(e.session.abort(),this.addBubble("error","Generation stopped"))}attachActiveDocument(){let e=this.app.workspace.getActiveFile();if(!e){new z.Notice("No active document to attach");return}if(this.attachedFiles.some(a=>a.path===e.path)){new z.Notice(`"${e.name}" is already attached`);return}let s=e.extension.toLowerCase();if(!new Set(["md","txt","json","yaml","yml","toml","csv","xml","html","css","js","ts","py","sh","sql","env","cfg","ini","log"]).has(s)){new z.Notice(`Can't attach "${e.name}" \u2014 only text files are supported`);return}this.attachedFiles.push(e),this.renderPills()}async saveImageBlobToVault(e){let s=e.type.split("/")[1]?.replace("jpeg","jpg")??"png",n=Date.now(),a=e.name&&e.name!=="image"?e.name:`pasted-${n}.${s}`,i=`${this.plugin.settings.fleetFolder}/chat-images`,o=`${i}/${n}-${a}`;try{this.app.vault.getAbstractFileByPath(i)||await this.app.vault.createFolder(i);let c=await e.arrayBuffer();await this.app.vault.createBinary(o,c);let h=this.app.vault.adapter.getBasePath?.()??"";return{name:a,path:`${h}/${o}`}}catch(c){let l=c instanceof Error?c.message:String(c);return new z.Notice(`Failed to save image: ${l}`),null}}async buildAttachmentContextFor(e,s){if(e.length===0&&s.length===0)return"";let n=[];for(let a of e)try{let i=await this.app.vault.cachedRead(a);n.push(`### ${a.name}
12130
12041
  \`\`\`
12131
- ${r}
12042
+ ${i}
12132
12043
  \`\`\``)}catch{n.push(`### ${a.name}
12133
12044
  (Could not read file)`)}for(let a of s)n.push(`### Image: ${a.name}
12134
12045
  The image file is located at: ${a.path}
@@ -12140,6 +12051,6 @@ ${n.join(`
12140
12051
 
12141
12052
  ---
12142
12053
 
12143
- `}async attachImageBlob(e){let s=e.type.split("/")[1]?.replace("jpeg","jpg")??"png",n=e.name&&e.name!=="image"?e.name:`pasted-${Date.now()}.${s}`;if(this.attachedImages.some(r=>r.name===n)){new z.Notice(`"${n}" is already attached`);return}let a=await this.saveImageBlobToVault(e);a&&(this.attachedImages.push(a),this.renderPills())}removeAttachment(e){this.attachedFiles=this.attachedFiles.filter(s=>s.path!==e),this.attachedImages=this.attachedImages.filter(s=>s.path!==e),this.renderPills()}renderPills(){if(this.pillsRow.empty(),this.attachedFiles.length+this.attachedImages.length===0){this.pillsRow.setCssStyles({display:"none"});return}this.pillsRow.setCssStyles({display:"flex"});for(let s of this.attachedFiles){let n=this.pillsRow.createDiv({cls:"af-chat-pill"}),a=n.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(a,"file-text"),n.createSpan({cls:"af-chat-pill-name",text:s.name});let r=n.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(r,"x"),r.onclick=o=>{o.stopPropagation(),this.removeAttachment(s.path)}}for(let s of this.attachedImages){let n=this.pillsRow.createDiv({cls:"af-chat-pill"}),a=n.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(a,"image"),n.createSpan({cls:"af-chat-pill-name",text:s.name});let r=n.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(r,"x"),r.onclick=o=>{o.stopPropagation(),this.removeAttachment(s.path)}}}buildAttachmentContext(){return this.buildAttachmentContextFor(this.attachedFiles,this.attachedImages)}async handleSend(){let e=this.getCurrentSession();if(!e)return;let s=this.textarea.value.trim();if(!s)return;let n=await this.buildAttachmentContext(),a=[...this.attachedFiles.map(u=>u.name),...this.attachedImages.map(u=>u.name)];this.textarea.value="",this.textarea.setCssStyles({height:"auto"}),this.sendBtn.setCssStyles({display:"none"}),this.attachedFiles=[],this.attachedImages=[],this.renderPills();let r=n?`${n}${s}`:void 0;if(this.addBubble("user",s,a.length>0?a:void 0),e.session.isStreaming){e.session.injectMessage(s,r,a.length>0?a:void 0);return}let o=null,c="",l=!1,h=e.session,d=()=>this.getCurrentSession()?.session===h;try{await e.session.sendMessage(s,u=>{if(!d()){o=null,l=!1,c="";return}if(u.type==="text"){(!l||!o||!o.isConnected)&&(o=this.addBubble("assistant"),l=!0),c+=u.content;let p=o.querySelector(".af-chat-stream-text");p||(p=o.createDiv({cls:"af-chat-stream-text"})),p.setText(c)}else if(u.type==="error"){let p=u.errorMessage?.trim()||"The agent's run ended with an error.";this.addBubble("error",`Error: ${p}`)}else if(u.type!=="compacted"){if(u.type==="result"){if(l&&o&&o.isConnected){let p=hs(c);this.renderMarkdownBubble(o,p),o._setRawText?.(p);let m=e.session.messages[e.session.messages.length-1];if(m&&m.role==="assistant"&&(this.attachThreadAffordance(o,m.id,e.session),u.toolCalls&&u.toolCalls.length>0)){let f=this.getOrCreateAffordancesRow(o);this.buildToolSummary(u.toolCalls,f)}}else u.toolCalls&&u.toolCalls.length>0&&this.buildToolSummary(u.toolCalls);c="",l=!1,o=null}}},r,a.length>0?a:void 0)}catch(u){let p=u instanceof Error?u.message:String(u);p!=="Aborted"&&this.addBubble("error",`Error: ${p}`)}}async handleNewChat(){if(!this.selectedAgentName)return;let s=this.plugin.runtime.getSnapshot().agents.find(a=>a.name===this.selectedAgentName);if(!s)return;let n=el();try{await this.plugin.repository.createConversation(s,n,tl())}catch(a){new z.Notice(`Couldn't create conversation: ${a instanceof Error?a.message:String(a)}`);return}await this.switchToAgent(this.selectedAgentName,n)}startFreshIntro(e){let s="",n=null,a=!1;e.sendMessage("Please introduce yourself and briefly describe your capabilities and what you can help with.",r=>{r.type==="text"&&(a||(n=this.addBubble("assistant"),a=!0),s+=r.content,n.setText(s))}).then(r=>{a&&n?(this.renderMarkdownBubble(n,s),n._setRawText?.(s)):r.text.trim()&&(n=this.addBubble("assistant"),this.renderMarkdownBubble(n,r.text),n._setRawText?.(r.text)),r.toolCalls.length>0&&this.buildToolSummary(r.toolCalls),this.textarea.focus()}).catch(r=>{let o=r instanceof Error?r.message:String(r);o!=="Aborted"&&this.addBubble("error",`Error: ${o}`)})}};function is(i){return i>=1e3?`${(i/1e3).toFixed(i>=1e4?0:1)}k`:`${i}`}var zn=class extends Se.Plugin{settings={...rt};repository;runtime;get mcpManager(){return this.runtime.mcpManager}mcpAuth=new gn;channelCredentials=new bn;channelManager;secretStore;statusBarEl;subscribedViews=new Set;vaultChangeTimer;suppressVaultEvents=!1;suppressTimer;runtimeUnsubscribe;async onload(){await this.loadSettings(),this.settings.claudeCliPath=await this.resolveClaudeCliPath(this.settings.claudeCliPath),this.repository=new ps(this.app,this.settings),this.repository.setChannelCredentialGetter(()=>this.channelCredentials.toRecord()),this.runtime=new ks(this.repository,this.settings,this.mcpAuth),this.registerView(_t,n=>new Fs(n,this)),this.registerView($t,n=>new Un(n,this)),this.registerView(it,n=>new rs(n,this)),this.addSettingTab(new en(this)),await this.repository.ensureFleetStructure()&&await this.repository.ensureSamples();let e=await this.repository.updateDefaults(this.settings.defaultFileHashes??{});JSON.stringify(e)!==JSON.stringify(this.settings.defaultFileHashes??{})&&(this.settings.defaultFileHashes=e,await this.saveData(this.settings)),await this.runtime.initialize(),await this.verifyClaudeCli(!1),await this.maybeResolveCodexCliPath(),this.addRibbonIcon("bot","Agent Fleet Dashboard",()=>void this.activateDashboardView()),this.addRibbonIcon("message-circle","Agent Chat",()=>{let n=this.app.workspace.getLeavesOfType(it);n.length>0?this.app.workspace.revealLeaf(n[0]):this.openChatView()}),this.addCommands(),this.registerVaultHandlers(),this.registerRuntimeListeners();let s=this.app.secretStorage;if(this.secretStore=new fn(s),this.channelCredentials.setSecretStore(this.secretStore),this.mcpAuth.setSecretStore(this.secretStore),!this.settings.secretsMigrated&&this.secretStore.available){this.channelCredentials.loadCredentials(this.settings.channelCredentials??{});for(let[n,a]of Object.entries(this.settings.mcpApiKeys??{}))typeof a=="string"&&a.trim()&&this.mcpAuth.storeStaticToken(n,a);this.settings.mcpTokens={},this.settings.mcpApiKeys={},this.settings.channelCredentials={},this.settings.secretsMigrated=!0,await this.saveData(this.settings)}else this.channelCredentials.loadCredentials(this.secretStore.available?void 0:this.settings.channelCredentials??{});this.secretStore.available||this.channelCredentials.onChanged(n=>{this.settings.channelCredentials=n,this.saveSettings()}),this.channelManager=new wn({getRepository:()=>this.repository,vault:this.app.vault,getSettings:()=>this.settings,getChannelCredentials:()=>this.channelCredentials.toRecord(),getMcpAuth:()=>this.mcpAuth,recordUsage:n=>this.runtime.recordUsage(n),adapterFactory:(n,a)=>{if(n.type==="slack")return new On(n,a);if(n.type==="telegram")return new Nn(n,a);if(n.type==="discord")return new Bn(n,a);throw new Error(`Channel type \`${n.type}\` is not yet supported in this version.`)}});try{await this.channelManager.start(this.runtime.getSnapshot())}catch(n){console.error("Agent Fleet: channel manager failed to start",n),new Se.Notice("Agent Fleet: channel manager failed to start \u2014 check console.")}this.runtime.onChannelResult((n,a,r,o,c)=>{let h=`*${o==="heartbeat"?`Heartbeat \u2014 ${n}`:`${n} \u2014 ${o}`}*
12054
+ `}async attachImageBlob(e){let s=e.type.split("/")[1]?.replace("jpeg","jpg")??"png",n=e.name&&e.name!=="image"?e.name:`pasted-${Date.now()}.${s}`;if(this.attachedImages.some(i=>i.name===n)){new z.Notice(`"${n}" is already attached`);return}let a=await this.saveImageBlobToVault(e);a&&(this.attachedImages.push(a),this.renderPills())}removeAttachment(e){this.attachedFiles=this.attachedFiles.filter(s=>s.path!==e),this.attachedImages=this.attachedImages.filter(s=>s.path!==e),this.renderPills()}renderPills(){if(this.pillsRow.empty(),this.attachedFiles.length+this.attachedImages.length===0){this.pillsRow.setCssStyles({display:"none"});return}this.pillsRow.setCssStyles({display:"flex"});for(let s of this.attachedFiles){let n=this.pillsRow.createDiv({cls:"af-chat-pill"}),a=n.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(a,"file-text"),n.createSpan({cls:"af-chat-pill-name",text:s.name});let i=n.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(i,"x"),i.onclick=o=>{o.stopPropagation(),this.removeAttachment(s.path)}}for(let s of this.attachedImages){let n=this.pillsRow.createDiv({cls:"af-chat-pill"}),a=n.createSpan({cls:"af-chat-pill-icon"});(0,z.setIcon)(a,"image"),n.createSpan({cls:"af-chat-pill-name",text:s.name});let i=n.createSpan({cls:"af-chat-pill-remove"});(0,z.setIcon)(i,"x"),i.onclick=o=>{o.stopPropagation(),this.removeAttachment(s.path)}}}buildAttachmentContext(){return this.buildAttachmentContextFor(this.attachedFiles,this.attachedImages)}async handleSend(){let e=this.getCurrentSession();if(!e)return;let s=this.textarea.value.trim();if(!s)return;let n=await this.buildAttachmentContext(),a=[...this.attachedFiles.map(u=>u.name),...this.attachedImages.map(u=>u.name)];this.textarea.value="",this.textarea.setCssStyles({height:"auto"}),this.sendBtn.setCssStyles({display:"none"}),this.attachedFiles=[],this.attachedImages=[],this.renderPills();let i=n?`${n}${s}`:void 0;if(this.addBubble("user",s,a.length>0?a:void 0),e.session.isStreaming){e.session.injectMessage(s,i,a.length>0?a:void 0);return}let o=null,c="",l=!1,h=e.session,d=()=>this.getCurrentSession()?.session===h;try{await e.session.sendMessage(s,u=>{if(!d()){o=null,l=!1,c="";return}if(u.type==="text"){(!l||!o||!o.isConnected)&&(o=this.addBubble("assistant"),l=!0),c+=u.content;let p=o.querySelector(".af-chat-stream-text");p||(p=o.createDiv({cls:"af-chat-stream-text"})),p.setText(c)}else if(u.type==="error"){let p=u.errorMessage?.trim()||"The agent's run ended with an error.";this.addBubble("error",`Error: ${p}`)}else if(u.type!=="compacted"){if(u.type==="result"){if(l&&o&&o.isConnected){let p=At(c);this.renderMarkdownBubble(o,p),o._setRawText?.(p);let m=e.session.messages[e.session.messages.length-1];if(m&&m.role==="assistant"&&(this.attachThreadAffordance(o,m.id,e.session),u.toolCalls&&u.toolCalls.length>0)){let f=this.getOrCreateAffordancesRow(o);this.buildToolSummary(u.toolCalls,f)}}else u.toolCalls&&u.toolCalls.length>0&&this.buildToolSummary(u.toolCalls);c="",l=!1,o=null}}},i,a.length>0?a:void 0)}catch(u){let p=u instanceof Error?u.message:String(u);p!=="Aborted"&&this.addBubble("error",`Error: ${p}`)}}async handleNewChat(){if(!this.selectedAgentName)return;let s=this.plugin.runtime.getSnapshot().agents.find(a=>a.name===this.selectedAgentName);if(!s)return;let n=tl();try{await this.plugin.repository.createConversation(s,n,sl())}catch(a){new z.Notice(`Couldn't create conversation: ${a instanceof Error?a.message:String(a)}`);return}await this.switchToAgent(this.selectedAgentName,n)}startFreshIntro(e){let s="",n=null,a=!1;e.sendMessage("Please introduce yourself and briefly describe your capabilities and what you can help with.",i=>{i.type==="text"&&(a||(n=this.addBubble("assistant"),a=!0),s+=i.content,n.setText(s))}).then(i=>{a&&n?(this.renderMarkdownBubble(n,s),n._setRawText?.(s)):i.text.trim()&&(n=this.addBubble("assistant"),this.renderMarkdownBubble(n,i.text),n._setRawText?.(i.text)),i.toolCalls.length>0&&this.buildToolSummary(i.toolCalls),this.textarea.focus()}).catch(i=>{let o=i instanceof Error?i.message:String(i);o!=="Aborted"&&this.addBubble("error",`Error: ${o}`)})}};function is(r){return r>=1e3?`${(r/1e3).toFixed(r>=1e4?0:1)}k`:`${r}`}var zn=class extends Se.Plugin{settings={...it};repository;runtime;get mcpManager(){return this.runtime.mcpManager}mcpAuth=new gn;channelCredentials=new bn;channelManager;secretStore;statusBarEl;subscribedViews=new Set;vaultChangeTimer;suppressVaultEvents=!1;suppressTimer;runtimeUnsubscribe;async onload(){await this.loadSettings(),this.settings.claudeCliPath=await this.resolveClaudeCliPath(this.settings.claudeCliPath),this.repository=new ps(this.app,this.settings),this.repository.setChannelCredentialGetter(()=>this.channelCredentials.toRecord()),this.runtime=new ks(this.repository,this.settings,this.mcpAuth),this.registerView(_t,a=>new Fs(a,this)),this.registerView(jt,a=>new Un(a,this)),this.registerView(rt,a=>new os(a,this)),this.addSettingTab(new en(this)),await this.repository.ensureFleetStructure()&&await this.repository.ensureSamples();let e=await this.repository.updateDefaults(this.settings.defaultFileHashes??{});JSON.stringify(e)!==JSON.stringify(this.settings.defaultFileHashes??{})&&(this.settings.defaultFileHashes=e,await this.saveData(this.settings));let s=await this.repository.migrateUsageLedgerCosts();s&&s.changed>0&&console.log(`Agent Fleet: repaired ${s.changed} usage-ledger cost rows across ${s.files} day(s).`),await this.runtime.initialize(),await this.verifyClaudeCli(!1),await this.maybeResolveCodexCliPath(),this.addRibbonIcon("bot","Agent Fleet Dashboard",()=>void this.activateDashboardView()),this.addRibbonIcon("message-circle","Agent Chat",()=>{let a=this.app.workspace.getLeavesOfType(rt);a.length>0?this.app.workspace.revealLeaf(a[0]):this.openChatView()}),this.addCommands(),this.registerVaultHandlers(),this.registerRuntimeListeners();let n=this.app.secretStorage;if(this.secretStore=new fn(n),this.channelCredentials.setSecretStore(this.secretStore),this.mcpAuth.setSecretStore(this.secretStore),!this.settings.secretsMigrated&&this.secretStore.available){this.channelCredentials.loadCredentials(this.settings.channelCredentials??{});for(let[a,i]of Object.entries(this.settings.mcpApiKeys??{}))typeof i=="string"&&i.trim()&&this.mcpAuth.storeStaticToken(a,i);this.settings.mcpTokens={},this.settings.mcpApiKeys={},this.settings.channelCredentials={},this.settings.secretsMigrated=!0,await this.saveData(this.settings)}else this.channelCredentials.loadCredentials(this.secretStore.available?void 0:this.settings.channelCredentials??{});this.secretStore.available||this.channelCredentials.onChanged(a=>{this.settings.channelCredentials=a,this.saveSettings()}),this.channelManager=new wn({getRepository:()=>this.repository,vault:this.app.vault,getSettings:()=>this.settings,getChannelCredentials:()=>this.channelCredentials.toRecord(),getMcpAuth:()=>this.mcpAuth,recordUsage:a=>this.runtime.recordUsage(a),adapterFactory:(a,i)=>{if(a.type==="slack")return new On(a,i);if(a.type==="telegram")return new Nn(a,i);if(a.type==="discord")return new Bn(a,i);throw new Error(`Channel type \`${a.type}\` is not yet supported in this version.`)}});try{await this.channelManager.start(this.runtime.getSnapshot())}catch(a){console.error("Agent Fleet: channel manager failed to start",a),new Se.Notice("Agent Fleet: channel manager failed to start \u2014 check console.")}this.runtime.onChannelResult((a,i,o,c,l)=>{let d=`*${c==="heartbeat"?`Heartbeat \u2014 ${a}`:`${a} \u2014 ${c}`}*
12144
12055
 
12145
- ${r}`;(c?this.channelManager?.postToChannelTarget(a,c,h):this.channelManager?.broadcastToChannel(a,h))?.catch(u=>{console.warn(`Agent Fleet: channel post failed for ${n}`,u)})}),this.refreshStatusBar(),this.importNativeMcpServers(),this.registerInterval(window.setInterval(()=>void this.mcpManager.refreshProbeTokens(),30*6e4)),new Se.Notice("Agent Fleet loaded.")}async importNativeMcpServers(){if(this.settings.mcpImported)return;if(this.repository.getMcpServers().length>0){this.settings.mcpImported=!0,await this.saveData(this.settings);return}let t=n=>{try{return(0,Os.existsSync)(n)?(0,Os.readFileSync)(n,"utf-8"):null}catch{return null}},e=t((0,Qa.join)((0,Xa.homedir)(),".claude.json")),s=t((0,Qa.join)((0,Xa.homedir)(),".codex","config.toml"));try{let n=Tr(e?Sr(e):{servers:[],tokens:{}},s?Cr(s):{servers:[],tokens:{}}),a=0;for(let r of n.servers)try{await this.repository.saveMcpServer(r,r.description??"");let o=n.tokens[r.name];o&&this.mcpAuth.storeStaticToken(r.name,o),a++}catch(o){console.warn(`Agent Fleet: failed to import MCP server "${r.name}":`,o)}this.settings.mcpImported=!0,await this.saveData(this.settings),a>0&&(await this.refreshFromVault(),new Se.Notice(`Agent Fleet: imported ${a} MCP server${a===1?"":"s"}.`))}catch(n){console.error("Agent Fleet: MCP import failed",n)}}onunload(){this.runtimeUnsubscribe?.(),this.runtimeUnsubscribe=void 0,this.vaultChangeTimer&&(window.clearTimeout(this.vaultChangeTimer),this.vaultChangeTimer=void 0),this.suppressTimer&&(window.clearTimeout(this.suppressTimer),this.suppressTimer=void 0),this.runtime?.shutdown(),this.channelManager?.stop(),Qi()}async loadSettings(){this.settings={...rt,...await this.loadData()}}async saveSettings(){this.settings.claudeCliPath=await this.resolveClaudeCliPath(this.settings.claudeCliPath),await this.maybeResolveCodexCliPath(),Zi(),await this.saveData(this.settings),this.repository&&this.runtime&&(this.runtime.shutdown(),this.repository=new ps(this.app,this.settings),this.repository.setChannelCredentialGetter(()=>this.channelCredentials.toRecord()),this.runtime=new ks(this.repository,this.settings,this.mcpAuth),await this.repository.ensureFleetStructure(),await this.runtime.initialize(),this.registerRuntimeListeners(),this.notifyViews(),this.refreshStatusBar(),this.channelCredentials.loadCredentials(this.secretStore.available?void 0:this.settings.channelCredentials??{}),this.channelManager?.reconcile(this.runtime.getSnapshot()))}subscribeView(t){this.subscribedViews.add(t)}unsubscribeView(t){this.subscribedViews.delete(t)}async activateDashboardView(){let t=this.app.workspace.getLeavesOfType(_t);if(t.length>0){this.app.workspace.revealLeaf(t[0]);return}await this.app.workspace.getLeaf(!0).setViewState({type:_t,active:!0})}async navigateDashboard(t,e){await this.activateDashboardView();let n=this.app.workspace.getLeavesOfType(_t)[0];if(n){let a=n.view;a instanceof Fs&&a.navigateTo(t,e)}}async activateAgentsView(){let t=this.getLeafForView($t,"left");await t.setViewState({type:$t,active:!0}),this.app.workspace.revealLeaf(t)}async openChatView(t){if(t){let s=this.app.workspace.getLeavesOfType(it);for(let n of s)if(n.view instanceof rs&&n.view.selectedAgentName===t){this.app.workspace.revealLeaf(n);return}}let e=this.app.workspace.getRightLeaf(!1)??this.app.workspace.getLeaf(!0);await e.setViewState({type:it,active:!0,state:t?{agentName:t}:{}}),this.app.workspace.revealLeaf(e),t&&e.view instanceof rs&&e.view.selectAgent(t)}async refreshFromVault(){this.suppressVaultEvents=!0;try{await this.runtime.refreshFromVault(),this.notifyViews(),this.refreshStatusBar(),this.channelManager?.reconcile(this.runtime.getSnapshot())}finally{this.suppressTimer&&window.clearTimeout(this.suppressTimer),this.suppressTimer=window.setTimeout(()=>{this.suppressTimer=void 0,this.suppressVaultEvents=!1},500)}}refreshStatusBar(){if(!this.settings.showStatusBar){this.statusBarEl?.detach(),this.statusBarEl=void 0;return}this.statusBarEl||(this.statusBarEl=this.addStatusBarItem(),this.statusBarEl.onclick=()=>void this.activateDashboardView());let t=this.runtime.getFleetStatus();this.statusBarEl.setText(`\u{1F916} ${t.running} running \xB7 ${t.pending} pending \xB7 ${t.completedToday} completed today`)}async verifyClaudeCli(t=!0){let e=await this.resolveClaudeCliPath(this.settings.claudeCliPath);return this.settings.claudeCliPath=e,await this.verifyCliBinary(e,"Claude",t)}async verifyCodexCli(t=!0){let e=await this.resolveCliPathFrom(ta(this.settings.codexCliPath),this.settings.codexCliPath);return this.settings.codexCliPath=e,await this.verifyCliBinary(e,"Codex",t)}async verifyCliBinary(t,e,s){return await new Promise(n=>{let a=dt(t,["--version"]),r="";a.stderr.on("data",o=>{r+=o.toString()}),a.on("close",o=>{let c=o===0;c||console.error(`Agent Fleet: ${e} CLI verification failed`,r),s&&new Se.Notice(c?`${e} CLI available.`:`${e} CLI verification failed \u2014 check ${e} CLI Path in settings.`),n(c)}),a.on("error",o=>{console.error(`Agent Fleet: ${e} CLI verification error`,o),s&&new Se.Notice(`${e} CLI verification failed \u2014 check ${e} CLI Path in settings.`),n(!1)})})}async openPath(t){let e=this.app.vault.getAbstractFileByPath((0,Se.normalizePath)(t));e instanceof Se.TFile&&await this.app.workspace.getLeaf(!0).openFile(e)}async createAgentTemplate(){await this.navigateDashboard("create-agent")}async createSkillTemplate(){await this.navigateDashboard("create-skill")}async openCreateTask(){await this.navigateDashboard("create-task")}async runAgentPrompt(t){let e=this.repository.getAgentByName(t);if(!e){new Se.Notice(`Unknown agent: ${t}`);return}await this.runtime.runAgentNow(e,"Run now and summarize the current state.")}async chatWithAgent(t){if(!this.repository.getAgentByName(t)){new Se.Notice(`Unknown agent: ${t}`);return}await this.openChatView(t)}async deleteAgent(t){if(!this.repository.getAgentByName(t)){new Se.Notice(`Unknown agent: ${t}`);return}let s=this.repository.getTasksForAgent(t),n=this.runtime.getRecentRuns().filter(o=>o.agent===t),a=this.repository.getMemoryPath(t),r=!!this.app.vault.getAbstractFileByPath(a);new Ks(this.app,{agentName:t,taskCount:s.length,runCount:n.length,hasMemory:r},async o=>{let c=await this.repository.deleteAgent(t,o);await new Promise(l=>window.setTimeout(l,200)),await this.refreshFromVault(),new Se.Notice(`Deleted agent "${t}" (${c.trashedFiles.length} files moved to trash)`),await this.navigateDashboard("agents")}).open()}async toggleAgent(t,e){let s=this.repository.getAgentByName(t);if(!s)return;let n=this.app.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof Se.TFile))return;let a=await this.app.vault.cachedRead(n),{frontmatter:r,body:o}=J(a);r.enabled=e,await this.app.vault.modify(n,H(r,o)),await this.refreshFromVault()}addCommands(){this.addCommand({id:"open-dashboard",name:"Open dashboard",callback:()=>void this.activateDashboardView()}),this.addCommand({id:"open-agents-panel",name:"Open agents panel",callback:()=>void this.activateAgentsView()}),this.addCommand({id:"open-chat",name:"Open agent chat",callback:()=>{let t=this.app.workspace.getLeavesOfType(it);t.length>0?this.app.workspace.revealLeaf(t[0]):this.openChatView()}}),this.addCommand({id:"new-chat-tab",name:"New chat tab",callback:()=>void this.openChatView()}),this.addCommand({id:"new-agent",name:"New agent",callback:()=>void this.createAgentTemplate()}),this.addCommand({id:"new-skill",name:"New skill",callback:()=>void this.createSkillTemplate()}),this.addCommand({id:"new-task",name:"New task",callback:()=>void this.openCreateTask()}),this.addCommand({id:"run-agent-now",name:"Run agent now",callback:()=>{let t=this.runtime.getSnapshot().agents[0];t?this.runAgentPrompt(t.name):new Se.Notice("No agents configured.")}}),this.addCommand({id:"pause-all",name:"Pause all",callback:()=>{this.runtime.scheduler.pauseAll(),new Se.Notice("Agent Fleet paused.")}}),this.addCommand({id:"resume-all",name:"Resume all",callback:()=>{this.runtime.scheduler.resumeAll(),new Se.Notice("Agent Fleet resumed.")}}),this.addCommand({id:"view-fleet-status",name:"View status",callback:()=>{let t=this.runtime.getFleetStatus();new Se.Notice(`${t.running} running \xB7 ${t.pending} pending \xB7 ${t.completedToday} completed today`)}})}debouncedVaultRefresh(){this.suppressVaultEvents||(this.vaultChangeTimer&&window.clearTimeout(this.vaultChangeTimer),this.vaultChangeTimer=window.setTimeout(()=>{this.suppressVaultEvents||this.refreshFromVault()},500))}registerVaultHandlers(){let t=e=>e.startsWith(`${this.settings.fleetFolder}/usage/`);this.registerEvent(this.app.vault.on("create",e=>{e instanceof Se.TFile&&e.path.startsWith(`${this.settings.fleetFolder}/`)&&!t(e.path)&&this.debouncedVaultRefresh()})),this.registerEvent(this.app.vault.on("modify",e=>{e instanceof Se.TFile&&e.path.startsWith(`${this.settings.fleetFolder}/`)&&!t(e.path)&&this.debouncedVaultRefresh()})),this.registerEvent(this.app.vault.on("rename",e=>{e.path.startsWith(`${this.settings.fleetFolder}/`)&&this.debouncedVaultRefresh()})),this.registerEvent(this.app.vault.on("delete",e=>{e.path.startsWith(`${this.settings.fleetFolder}/`)&&this.debouncedVaultRefresh()}))}registerRuntimeListeners(){this.runtimeUnsubscribe?.(),this.runtimeUnsubscribe=this.runtime.subscribe(()=>{this.notifyViews(),this.refreshStatusBar()})}notifyViews(){for(let t of this.subscribedViews)t.render()}async resolveClaudeCliPath(t){return this.resolveCliPathFrom(Ti(t),t)}async maybeResolveCodexCliPath(){this.runtime?.getSnapshot().agents.some(e=>kt(e.adapter)==="codex")&&(this.settings.codexCliPath=await this.resolveCliPathFrom(ta(this.settings.codexCliPath),this.settings.codexCliPath))}async resolveCliPathFrom(t,e){for(let s of t)if(sa(s)&&(0,Os.existsSync)(s)||!sa(s)&&await new Promise(a=>{let r=dt(s,["--version"]);r.on("close",o=>a(o===0)),r.on("error",()=>a(!1))}))return s;return e}getLeafForView(t,e){let s=this.app.workspace.getLeavesOfType(t)[0];return s||(e==="right"?this.app.workspace.getRightLeaf(!1)??this.app.workspace.getLeaf(!0):this.app.workspace.getLeftLeaf(!1)??this.app.workspace.getLeaf(!1))}};
12056
+ ${o}`;(l?this.channelManager?.postToChannelTarget(i,l,d):this.channelManager?.broadcastToChannel(i,d))?.catch(p=>{console.warn(`Agent Fleet: channel post failed for ${a}`,p)})}),this.refreshStatusBar(),this.importNativeMcpServers(),this.registerInterval(window.setInterval(()=>void this.mcpManager.refreshProbeTokens(),30*6e4)),new Se.Notice("Agent Fleet loaded.")}async importNativeMcpServers(){if(this.settings.mcpImported)return;if(this.repository.getMcpServers().length>0){this.settings.mcpImported=!0,await this.saveData(this.settings);return}let t=n=>{try{return(0,Os.existsSync)(n)?(0,Os.readFileSync)(n,"utf-8"):null}catch{return null}},e=t((0,Qa.join)((0,Xa.homedir)(),".claude.json")),s=t((0,Qa.join)((0,Xa.homedir)(),".codex","config.toml"));try{let n=_i(e?Ci(e):{servers:[],tokens:{}},s?Ti(s):{servers:[],tokens:{}}),a=0;for(let i of n.servers)try{await this.repository.saveMcpServer(i,i.description??"");let o=n.tokens[i.name];o&&this.mcpAuth.storeStaticToken(i.name,o),a++}catch(o){console.warn(`Agent Fleet: failed to import MCP server "${i.name}":`,o)}this.settings.mcpImported=!0,await this.saveData(this.settings),a>0&&(await this.refreshFromVault(),new Se.Notice(`Agent Fleet: imported ${a} MCP server${a===1?"":"s"}.`))}catch(n){console.error("Agent Fleet: MCP import failed",n)}}onunload(){this.runtimeUnsubscribe?.(),this.runtimeUnsubscribe=void 0,this.vaultChangeTimer&&(window.clearTimeout(this.vaultChangeTimer),this.vaultChangeTimer=void 0),this.suppressTimer&&(window.clearTimeout(this.suppressTimer),this.suppressTimer=void 0),this.runtime?.shutdown(),this.channelManager?.stop(),Zr()}async loadSettings(){this.settings={...it,...await this.loadData()}}async saveSettings(){this.settings.claudeCliPath=await this.resolveClaudeCliPath(this.settings.claudeCliPath),await this.maybeResolveCodexCliPath(),ei(),await this.saveData(this.settings),this.repository&&this.runtime&&(this.runtime.shutdown(),this.repository=new ps(this.app,this.settings),this.repository.setChannelCredentialGetter(()=>this.channelCredentials.toRecord()),this.runtime=new ks(this.repository,this.settings,this.mcpAuth),await this.repository.ensureFleetStructure(),await this.runtime.initialize(),this.registerRuntimeListeners(),this.notifyViews(),this.refreshStatusBar(),this.channelCredentials.loadCredentials(this.secretStore.available?void 0:this.settings.channelCredentials??{}),this.channelManager?.reconcile(this.runtime.getSnapshot()))}subscribeView(t){this.subscribedViews.add(t)}unsubscribeView(t){this.subscribedViews.delete(t)}async activateDashboardView(){let t=this.app.workspace.getLeavesOfType(_t);if(t.length>0){this.app.workspace.revealLeaf(t[0]);return}await this.app.workspace.getLeaf(!0).setViewState({type:_t,active:!0})}async navigateDashboard(t,e){await this.activateDashboardView();let n=this.app.workspace.getLeavesOfType(_t)[0];if(n){let a=n.view;a instanceof Fs&&a.navigateTo(t,e)}}async activateAgentsView(){let t=this.getLeafForView(jt,"left");await t.setViewState({type:jt,active:!0}),this.app.workspace.revealLeaf(t)}async openChatView(t){if(t){let s=this.app.workspace.getLeavesOfType(rt);for(let n of s)if(n.view instanceof os&&n.view.selectedAgentName===t){this.app.workspace.revealLeaf(n);return}}let e=this.app.workspace.getRightLeaf(!1)??this.app.workspace.getLeaf(!0);await e.setViewState({type:rt,active:!0,state:t?{agentName:t}:{}}),this.app.workspace.revealLeaf(e),t&&e.view instanceof os&&e.view.selectAgent(t)}async refreshFromVault(){this.suppressVaultEvents=!0;try{await this.runtime.refreshFromVault(),this.notifyViews(),this.refreshStatusBar(),this.channelManager?.reconcile(this.runtime.getSnapshot())}finally{this.suppressTimer&&window.clearTimeout(this.suppressTimer),this.suppressTimer=window.setTimeout(()=>{this.suppressTimer=void 0,this.suppressVaultEvents=!1},500)}}refreshStatusBar(){if(!this.settings.showStatusBar){this.statusBarEl?.detach(),this.statusBarEl=void 0;return}this.statusBarEl||(this.statusBarEl=this.addStatusBarItem(),this.statusBarEl.onclick=()=>void this.activateDashboardView());let t=this.runtime.getFleetStatus();this.statusBarEl.setText(`\u{1F916} ${t.running} running \xB7 ${t.pending} pending \xB7 ${t.completedToday} completed today`)}async verifyClaudeCli(t=!0){let e=await this.resolveClaudeCliPath(this.settings.claudeCliPath);return this.settings.claudeCliPath=e,await this.verifyCliBinary(e,"Claude",t)}async verifyCodexCli(t=!0){let e=await this.resolveCliPathFrom(ta(this.settings.codexCliPath),this.settings.codexCliPath);return this.settings.codexCliPath=e,await this.verifyCliBinary(e,"Codex",t)}async verifyCliBinary(t,e,s){return await new Promise(n=>{let a=dt(t,["--version"]),i="";a.stderr.on("data",o=>{i+=o.toString()}),a.on("close",o=>{let c=o===0;c||console.error(`Agent Fleet: ${e} CLI verification failed`,i),s&&new Se.Notice(c?`${e} CLI available.`:`${e} CLI verification failed \u2014 check ${e} CLI Path in settings.`),n(c)}),a.on("error",o=>{console.error(`Agent Fleet: ${e} CLI verification error`,o),s&&new Se.Notice(`${e} CLI verification failed \u2014 check ${e} CLI Path in settings.`),n(!1)})})}async openPath(t){let e=this.app.vault.getAbstractFileByPath((0,Se.normalizePath)(t));e instanceof Se.TFile&&await this.app.workspace.getLeaf(!0).openFile(e)}async createAgentTemplate(){await this.navigateDashboard("create-agent")}async createSkillTemplate(){await this.navigateDashboard("create-skill")}async openCreateTask(){await this.navigateDashboard("create-task")}async runAgentPrompt(t){let e=this.repository.getAgentByName(t);if(!e){new Se.Notice(`Unknown agent: ${t}`);return}await this.runtime.runAgentNow(e,"Run now and summarize the current state.")}async chatWithAgent(t){if(!this.repository.getAgentByName(t)){new Se.Notice(`Unknown agent: ${t}`);return}await this.openChatView(t)}async deleteAgent(t){if(!this.repository.getAgentByName(t)){new Se.Notice(`Unknown agent: ${t}`);return}let s=this.repository.getTasksForAgent(t),n=this.runtime.getRecentRuns().filter(o=>o.agent===t),a=this.repository.getMemoryPath(t),i=!!this.app.vault.getAbstractFileByPath(a);new Ks(this.app,{agentName:t,taskCount:s.length,runCount:n.length,hasMemory:i},async o=>{let c=await this.repository.deleteAgent(t,o);await new Promise(l=>window.setTimeout(l,200)),await this.refreshFromVault(),new Se.Notice(`Deleted agent "${t}" (${c.trashedFiles.length} files moved to trash)`),await this.navigateDashboard("agents")}).open()}async toggleAgent(t,e){let s=this.repository.getAgentByName(t);if(!s)return;let n=this.app.vault.getAbstractFileByPath(s.filePath);if(!(n instanceof Se.TFile))return;let a=await this.app.vault.cachedRead(n),{frontmatter:i,body:o}=J(a);i.enabled=e,await this.app.vault.modify(n,H(i,o)),await this.refreshFromVault()}addCommands(){this.addCommand({id:"open-dashboard",name:"Open dashboard",callback:()=>void this.activateDashboardView()}),this.addCommand({id:"open-agents-panel",name:"Open agents panel",callback:()=>void this.activateAgentsView()}),this.addCommand({id:"open-chat",name:"Open agent chat",callback:()=>{let t=this.app.workspace.getLeavesOfType(rt);t.length>0?this.app.workspace.revealLeaf(t[0]):this.openChatView()}}),this.addCommand({id:"new-chat-tab",name:"New chat tab",callback:()=>void this.openChatView()}),this.addCommand({id:"new-agent",name:"New agent",callback:()=>void this.createAgentTemplate()}),this.addCommand({id:"new-skill",name:"New skill",callback:()=>void this.createSkillTemplate()}),this.addCommand({id:"new-task",name:"New task",callback:()=>void this.openCreateTask()}),this.addCommand({id:"run-agent-now",name:"Run agent now",callback:()=>{let t=this.runtime.getSnapshot().agents[0];t?this.runAgentPrompt(t.name):new Se.Notice("No agents configured.")}}),this.addCommand({id:"pause-all",name:"Pause all",callback:()=>{this.runtime.scheduler.pauseAll(),new Se.Notice("Agent Fleet paused.")}}),this.addCommand({id:"resume-all",name:"Resume all",callback:()=>{this.runtime.scheduler.resumeAll(),new Se.Notice("Agent Fleet resumed.")}}),this.addCommand({id:"view-fleet-status",name:"View status",callback:()=>{let t=this.runtime.getFleetStatus();new Se.Notice(`${t.running} running \xB7 ${t.pending} pending \xB7 ${t.completedToday} completed today`)}})}debouncedVaultRefresh(){this.suppressVaultEvents||(this.vaultChangeTimer&&window.clearTimeout(this.vaultChangeTimer),this.vaultChangeTimer=window.setTimeout(()=>{this.suppressVaultEvents||this.refreshFromVault()},500))}registerVaultHandlers(){let t=e=>e.startsWith(`${this.settings.fleetFolder}/usage/`);this.registerEvent(this.app.vault.on("create",e=>{e instanceof Se.TFile&&e.path.startsWith(`${this.settings.fleetFolder}/`)&&!t(e.path)&&this.debouncedVaultRefresh()})),this.registerEvent(this.app.vault.on("modify",e=>{e instanceof Se.TFile&&e.path.startsWith(`${this.settings.fleetFolder}/`)&&!t(e.path)&&this.debouncedVaultRefresh()})),this.registerEvent(this.app.vault.on("rename",e=>{e.path.startsWith(`${this.settings.fleetFolder}/`)&&this.debouncedVaultRefresh()})),this.registerEvent(this.app.vault.on("delete",e=>{e.path.startsWith(`${this.settings.fleetFolder}/`)&&this.debouncedVaultRefresh()}))}registerRuntimeListeners(){this.runtimeUnsubscribe?.(),this.runtimeUnsubscribe=this.runtime.subscribe(()=>{this.notifyViews(),this.refreshStatusBar()})}notifyViews(){for(let t of this.subscribedViews)t.render()}async resolveClaudeCliPath(t){return this.resolveCliPathFrom(Tr(t),t)}async maybeResolveCodexCliPath(){this.runtime?.getSnapshot().agents.some(e=>kt(e.adapter)==="codex")&&(this.settings.codexCliPath=await this.resolveCliPathFrom(ta(this.settings.codexCliPath),this.settings.codexCliPath))}async resolveCliPathFrom(t,e){for(let s of t)if(sa(s)&&(0,Os.existsSync)(s)||!sa(s)&&await new Promise(a=>{let i=dt(s,["--version"]);i.on("close",o=>a(o===0)),i.on("error",()=>a(!1))}))return s;return e}getLeafForView(t,e){let s=this.app.workspace.getLeavesOfType(t)[0];return s||(e==="right"?this.app.workspace.getRightLeaf(!1)??this.app.workspace.getLeaf(!0):this.app.workspace.getLeftLeaf(!1)??this.app.workspace.getLeaf(!1))}};