@entrydesk/cli 1.11.0 → 1.11.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import F from"fs";import p from"fs";import Te from"os";import E from"path";import{HTTPError as Re,TimeoutError as xe}from"ky";import he,{TimeoutError as ye}from"ky";import x from"crypto";import u from"fs";import le from"path";var ge="ENTRYDESK_FORCE_FILE_STORAGE",A="EntryDesk CLI",de="__entrydesk_keychain_test__",ue="profile:",fe=".credentials.key",Z={keychain:!1,permissions:!1,corruptFile:!1,windowsFile:!1};function v(r,e){Z[r]||(Z[r]=!0,console.error(e))}function Q(r){if(!r||typeof r!="object")return!1;let e=r;return!(!("accessToken"in e||"refreshToken"in e||"email"in e||"workspaceId"in e)||"accessToken"in e&&typeof e.accessToken!="string"||"refreshToken"in e&&typeof e.refreshToken!="string"||"email"in e&&typeof e.email!="string"||"workspaceId"in e&&typeof e.workspaceId!="string")}function pe(r){if(!r||typeof r!="object")return!1;let e=r;return typeof e.getPassword=="function"&&typeof e.setPassword=="function"&&typeof e.deletePassword=="function"}function O(r){return o.getProfileCredentialsFile(r)}var N=class{getKeyFilePath(e){return le.join(o.getProfileDir(e),fe)}loadEncryptionKey(e){let t=this.getKeyFilePath(e);try{if(!u.existsSync(t))return null;let s=u.readFileSync(t);return s.length!==32?null:(this.ensureSecureFilePermissions(t),s)}catch{return null}}createEncryptionKey(e){o.ensureProfileDir(e);let t=this.getKeyFilePath(e),s=x.randomBytes(32);return u.writeFileSync(t,s,{mode:384}),this.ensureSecureFilePermissions(t),s}getOrCreateEncryptionKey(e){let t=this.loadEncryptionKey(e);return t||this.createEncryptionKey(e)}encrypt(e,t){let s=x.randomBytes(16),n=x.createCipheriv("aes-256-gcm",t,s),i=n.update(e,"utf8","hex");i+=n.final("hex");let c=n.getAuthTag();return`${s.toString("hex")}:${c.toString("hex")}:${i}`}decrypt(e,t){let s=e.split(":");if(s.length!==3)throw new Error("Invalid encrypted data format");let n=Buffer.from(s[0],"hex"),i=Buffer.from(s[1],"hex"),c=s[2],S=x.createDecipheriv("aes-256-gcm",t,n,{authTagLength:16});S.setAuthTag(i);let d=S.update(c,"hex","utf8");return d+=S.final("utf8"),d}tryDecrypt(e,t){try{return this.decrypt(e,t)}catch{return null}}tryParseCredentials(e){try{let t=JSON.parse(e);return Q(t)?t:null}catch{return null}}ensureSecureFilePermissions(e){if(process.platform!=="win32"){try{u.chmodSync(e,384)}catch{v("permissions","Warning: Failed to set secure permissions on credentials file.");return}try{(u.statSync(e).mode&511)!==384&&v("permissions","Warning: Credentials file permissions are too permissive.")}catch{v("permissions","Warning: Failed to verify credentials file permissions.")}}}async getCredentials(e){let t=O(e);try{if(!u.existsSync(t))return null;let s=u.readFileSync(t,"utf-8").trim();if(!s)return null;let n=this.loadEncryptionKey(e),i=n?this.tryDecrypt(s,n):null;if(i){let c=this.tryParseCredentials(i);if(c)return c}return v("corruptFile","Warning: Credentials file is unreadable. Please run `entrydesk login` again."),null}catch{return null}}async setCredentials(e,t){o.ensureProfileDir(e);let s=O(e),n=this.getOrCreateEncryptionKey(e),i=this.encrypt(JSON.stringify(t),n);u.writeFileSync(s,i,{mode:384}),this.ensureSecureFilePermissions(s)}async deleteCredentials(e){let t=O(e);try{u.existsSync(t)&&u.unlinkSync(t)}catch{}}},M=class{keytarModule=null;keytarLoadAttempted=!1;keychainAvailable=null;async getKeytar(){if(this.keytarLoadAttempted)return this.keytarModule;this.keytarLoadAttempted=!0;try{let t=await import("keytar"),s=t&&typeof t=="object"&&"default"in t?t.default:t;this.keytarModule=pe(s)?s:null}catch{this.keytarModule=null}return this.keytarModule}getAccount(e){return`${ue}${e}`}async checkKeychainAvailability(){if(this.keychainAvailable!==null)return this.keychainAvailable;try{let e=await this.getKeytar();if(!e)return this.keychainAvailable=!1,!1;let t=`${de}${x.randomBytes(8).toString("hex")}`,s="test";await e.setPassword(A,t,s);let n=await e.getPassword(A,t),i=await e.deletePassword(A,t);return this.keychainAvailable=!!(i&&n===s),this.keychainAvailable}catch{return this.keychainAvailable=!1,!1}}async isAvailable(){return this.checkKeychainAvailability()}async getCredentials(e){if(!await this.checkKeychainAvailability())throw new Error("Keychain is not available");let t=await this.getKeytar();if(!t)throw new Error("Keytar module not available");let s=await t.getPassword(A,this.getAccount(e));if(!s)return null;let n=this.tryParseKeychainPayload(s);return n||v("corruptFile",`Warning: Keychain entry for profile "${e}" is unreadable.`),n}async setCredentials(e,t){if(!await this.checkKeychainAvailability())throw new Error("Keychain is not available");let s=await this.getKeytar();if(!s)throw new Error("Keytar module not available");let n=JSON.stringify(t);await s.setPassword(A,this.getAccount(e),n)}async deleteCredentials(e){if(!await this.checkKeychainAvailability())throw new Error("Keychain is not available");let t=await this.getKeytar();if(!t)throw new Error("Keytar module not available");await t.deletePassword(A,this.getAccount(e))}tryParseKeychainPayload(e){try{let t=JSON.parse(e);return Q(t)?t:null}catch{return null}}},j=class{keychainStorage=null;fileStorage=new N;keychainAvailable=null;initPromise=null;shouldForceFileStorage(){return process.env[ge]?.toLowerCase()==="true"}async initializeStorage(){if(this.shouldForceFileStorage()){this.keychainAvailable=!1,this.keychainStorage=null;return}try{let e=new M,t=await e.isAvailable();this.keychainAvailable=t,this.keychainStorage=t?e:null}catch{this.keychainAvailable=!1,this.keychainStorage=null}!this.keychainAvailable&&!this.shouldForceFileStorage()&&(v("keychain","Keychain unavailable. Falling back to file-based credentials."),process.platform==="win32"&&v("windowsFile","Warning: File-based credentials on Windows may be readable by other users."))}async ensureInitialized(){if(this.shouldForceFileStorage()){this.keychainAvailable=!1,this.keychainStorage=null;return}this.keychainAvailable===null&&(this.initPromise||(this.initPromise=this.initializeStorage()),await this.initPromise)}disableKeychain(){this.shouldForceFileStorage()||v("keychain","Keychain error encountered. Falling back to file-based credentials."),this.keychainAvailable=!1,this.keychainStorage=null}async getKeychainStorage(){return await this.ensureInitialized(),this.shouldForceFileStorage()?null:this.keychainAvailable?this.keychainStorage:null}async getCredentials(e){let t=await this.getKeychainStorage();if(!t)return this.fileStorage.getCredentials(e);try{let n=await t.getCredentials(e);if(n)return n}catch{return this.disableKeychain(),this.fileStorage.getCredentials(e)}let s=await this.fileStorage.getCredentials(e);if(s)try{await t.setCredentials(e,s),await this.fileStorage.deleteCredentials(e)}catch{}return s}async setCredentials(e,t){let s=await this.getKeychainStorage();if(!s){await this.fileStorage.setCredentials(e,t);return}try{await s.setCredentials(e,t),await this.fileStorage.deleteCredentials(e)}catch{this.disableKeychain(),await this.fileStorage.setCredentials(e,t)}}async deleteCredentials(e){let t=await this.getKeychainStorage();if(t)try{await t.deleteCredentials(e)}catch{this.disableKeychain()}await this.fileStorage.deleteCredentials(e)}},g=new j,y={async saveAccessToken(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.accessToken=r,await g.setCredentials(t,s)},async getAccessToken(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.accessToken??null},async saveRefreshToken(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.refreshToken=r,await g.setCredentials(t,s)},async getRefreshToken(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.refreshToken??null},async saveEmail(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.email=r,await g.setCredentials(t,s)},async getEmail(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.email??null},async saveWorkspaceId(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.workspaceId=r,await g.setCredentials(t,s)},async getWorkspaceId(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.workspaceId??null},async saveAll(r,e){let t=o.getEffectiveProfileName(e);await g.setCredentials(t,r)},async getAll(r){let e=o.getEffectiveProfileName(r);return g.getCredentials(e)},async clear(r){let e=o.getEffectiveProfileName(r);await g.deleteCredentials(e)},async isLoggedIn(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.accessToken!=null}};var K=class r{client;baseUrl;constructor(e,t){this.baseUrl=e||U(t),I().ignoreSslErrors&&(process.env.NODE_TLS_REJECT_UNAUTHORIZED="0",process.env.DEBUG&&console.log("[DEBUG] ApiClient: SSL certificate validation disabled")),this.client=he.create({prefixUrl:this.baseUrl,timeout:3e4,retry:{limit:3,methods:["get","put","head","delete","options","trace"],statusCodes:[408,413,429,500,502,503,504]},hooks:{beforeError:[n=>(n.request?.headers&&(n.request.headers.delete("authorization"),n.request.headers.delete("Authorization")),n)]}})}getBaseUrl(){return this.baseUrl}async refreshOAuthToken(e){return this.client.post("v1/oauth/token",{json:{grant_type:"refresh_token",refresh_token:e.refreshToken,client_id:e.clientId}}).json()}async revokeOAuthToken(e){await this.client.post("v1/oauth/revoke",{json:{token:e.token,token_type_hint:"refresh_token",client_id:e.clientId}})}async getWorkspaces(e){try{return process.env.DEBUG&&console.log(`[DEBUG] Calling getWorkspaces API: ${this.baseUrl}/v1/workspaces`),await this.client.get("v1/workspaces",{headers:{Authorization:`Bearer ${e}`}}).json()}catch(t){throw process.env.DEBUG&&(console.error("[DEBUG] getWorkspaces error:",t),t instanceof Error&&"cause"in t&&console.error("[DEBUG] Error cause:",t.cause)),t}}async getAgents(e,t,s){let n={limit:s?.limit??100};return s?.nextKey&&(n.nextKey=s.nextKey),s?.search&&(n.search=s.search),await this.client.get(`v1/workspaces/${t}/availableAgents`,{headers:{Authorization:`Bearer ${e}`},searchParams:n}).json()}async createAgent(e,t,s){return await this.client.post(`v1/workspaces/${t}/agents`,{headers:{Authorization:`Bearer ${e}`},json:s}).json()}async updateAgent(e,t,s,n){await this.client.put(`v1/workspaces/${t}/agents/${s}`,{headers:{Authorization:`Bearer ${e}`},json:n})}async deleteAgent(e,t,s){await this.client.delete(`v1/workspaces/${t}/agents/${s}`,{headers:{Authorization:`Bearer ${e}`}})}async getModels(e,t){return await this.client.get(`v1/workspaces/${t}/models`,{headers:{Authorization:`Bearer ${e}`}}).json()}async getConnectors(e,t){return await this.client.get(`v1/workspaces/${t}/availableConnectors`,{headers:{Authorization:`Bearer ${e}`},searchParams:{limit:100,connectionStatus:"connected"}}).json()}async getChats(e,t,s=20){return await this.client.get(`v1/workspaces/${t}/chats`,{headers:{Authorization:`Bearer ${e}`},searchParams:{limit:s}}).json()}async getMessages(e,t,s,n=50){return await this.client.get(`v1/workspaces/${t}/chats/${s}/messages`,{headers:{Authorization:`Bearer ${e}`},searchParams:{limit:n}}).json()}async getAgent(e,t,s){return await this.client.get(`v1/workspaces/${t}/agents/${s}`,{headers:{Authorization:`Bearer ${e}`}}).json()}async listAvailableSkills(e,t,s){return await this.client.get(`v1/workspaces/${t}/availableSkills`,{headers:{Authorization:`Bearer ${e}`},searchParams:{createdBy:s?.createdBy,nextKey:s?.nextKey,limit:s?.limit??20}}).json()}async uploadSkill(e,t,s){return await this.client.post(`v1/workspaces/${t}/skills`,{headers:{Authorization:`Bearer ${e}`},body:s}).json()}async deleteSkill(e,t,s){await this.client.delete(`v1/workspaces/${t}/skills/${s}`,{headers:{Authorization:`Bearer ${e}`}})}async downloadSkillCurrent(e,t,s){return await this.client.get(`v1/workspaces/${t}/skills/${s}/download`,{headers:{Authorization:`Bearer ${e}`}}).json()}async chat(e,t,s){try{let n=await this.client.post(`v1/workspaces/${t}/chats`,{headers:{Authorization:`Bearer ${e}`},json:s,timeout:6e4,throwHttpErrors:!1});if(!n.ok){let i=await n.text();throw new Error(`Chat API failed (${n.status}): ${i}`)}return n}catch(n){throw n instanceof ye?new TypeError("Request timed out. Please check your connection and try again.",{cause:n}):n}}async getBudget(e,t){return await this.client.get(`v1/workspaces/${t}/budget`,{headers:{Authorization:`Bearer ${e}`}}).json()}async updateBudget(e,t,s){return await this.client.put(`v1/workspaces/${t}/budget`,{headers:{Authorization:`Bearer ${e}`},json:{budget:s}}).json()}async getMonthlyUsages(e,t,s){return await this.client.get(`v1/workspaces/${t}/monthlyUsages`,{headers:{Authorization:`Bearer ${e}`},searchParams:{nextKey:s?.nextKey,limit:s?.limit??20}}).json()}async createChatShare(e,t,s,n="public"){return await this.client.post(`v1/workspaces/${t}/chats/${s}/shares`,{headers:{Authorization:`Bearer ${e}`},json:{visibility:n}}).json()}async getSharedChat(e,t){let s={};return t&&(s.Authorization=`Bearer ${t}`),await this.client.get(`v1/shares/${e}`,{headers:s}).json()}async getSharedMessages(e,t=50,s){let n={};return s&&(n.Authorization=`Bearer ${s}`),await this.client.get(`v1/shares/${e}/messages`,{headers:n,searchParams:{limit:t}}).json()}async deleteChat(e,t,s){await this.client.delete(`v1/workspaces/${t}/chats/${s}`,{headers:{Authorization:`Bearer ${e}`}})}async getModelProviders(e,t){return await this.client.get(`v1/workspaces/${t}/modelProviders`,{headers:{Authorization:`Bearer ${e}`}}).json()}async uploadFile(e,t,s,n){let i=new FormData,c=new Blob([s.buffer],{type:s.mimeType});return i.append("file",c,s.filename),i.append("modelProviderId",n),await this.client.post(`v1/workspaces/${t}/files`,{headers:{Authorization:`Bearer ${e}`},body:i,timeout:12e4}).json()}async getSchedules(e,t,s){let n={limit:s?.limit??20};return s?.nextKey&&(n.nextKey=s.nextKey),s?.search&&(n.search=s.search),await this.client.get(`v1/workspaces/${t}/schedules`,{headers:{Authorization:`Bearer ${e}`},searchParams:n}).json()}async getSchedule(e,t,s){return await this.client.get(`v1/workspaces/${t}/schedules/${s}`,{headers:{Authorization:`Bearer ${e}`}}).json()}async createSchedule(e,t,s){return await this.client.post(`v1/workspaces/${t}/schedules`,{headers:{Authorization:`Bearer ${e}`},json:s}).json()}async updateSchedule(e,t,s,n){await this.client.put(`v1/workspaces/${t}/schedules/${s}`,{headers:{Authorization:`Bearer ${e}`},json:n})}async deleteSchedule(e,t,s){await this.client.delete(`v1/workspaces/${t}/schedules/${s}`,{headers:{Authorization:`Bearer ${e}`}})}static async getAuthenticatedClient(e){let t=await y.getAll(e);return t?{client:new r(void 0,e),accessToken:t.accessToken,workspaceId:t.workspaceId??null}:null}};function me(){return process.env.ENTRYDESK_SANDBOX==="true"}function z(){if(!me())return null;let r=process.env.ENTRYDESK_SANDBOX_ACCESS_TOKEN?.trim(),e=process.env.ENTRYDESK_SANDBOX_WORKSPACE_ID?.trim();if(!r||!e){let t=[];throw r||t.push("ENTRYDESK_SANDBOX_ACCESS_TOKEN"),e||t.push("ENTRYDESK_SANDBOX_WORKSPACE_ID"),new Error(`Sandbox mode enabled but missing credentials: ${t.join(", ")}`)}return{accessToken:r,workspaceId:e}}function ee(){let r=process.env.ENTRYDESK_TOKEN?.trim();if(!r)return null;let e=process.env.ENTRYDESK_WORKSPACE_ID?.trim();if(!e)throw new Error("ENTRYDESK_TOKEN is set but ENTRYDESK_WORKSPACE_ID is missing. Both are required for PAT authentication.");return{accessToken:r,workspaceId:e}}import{execFile as we}from"child_process";import{setTimeout as ve}from"timers/promises";import{decodeJwt as ke}from"jose";import Pe,{HTTPError as Se}from"ky";var D="entrydesk-cli",Ce="offline_access workspaces:read chats:read chats:write chats:delete agents:read agents:write connectors:read connectors:write tools:read tools:execute skills:read skills:write meetings:read meetings:write publishes:read publishes:write calendar:read calendar:write schedules:read schedules:write user:read user:write";function Ae(r){return new Promise((e,t)=>{let s=process.platform,n,i;switch(s){case"darwin":n="open",i=[r];break;case"win32":n="cmd",i=["/c","start","",r];break;default:n="xdg-open",i=[r];break}we(n,i,c=>{c?t(new Error(`Failed to open browser. Please manually visit: ${r}`,{cause:c})):e()})})}async function tt(r){let e=U(),t=te(),s=I(),n=process.env.NODE_TLS_REJECT_UNAUTHORIZED;s.ignoreSslErrors&&(process.env.NODE_TLS_REJECT_UNAUTHORIZED="0",console.error("WARNING: SSL certificate validation is disabled globally for this process during the login flow. All connections in this process will skip TLS verification. Only use ENTRYDESK_IGNORE_SSL_ERRORS for local development."));try{let i=Pe.create({prefixUrl:e,timeout:3e4,hooks:{beforeError:[a=>(a.request?.headers&&(a.request.headers.delete("authorization"),a.request.headers.delete("Authorization")),a)]}}),c;try{c=await i.post("v1/oauth/device/code",{json:{client_id:D,scope:Ce}}).json()}catch(a){let w=a instanceof Error?a:new Error("Failed to request device code");throw r?.onError?.(w),w}let{device_code:S,user_code:d,expires_in:oe,interval:ae}=c,J=new URL("/oauth/device",t);J.searchParams.set("user_code",d);let q=J.toString();r?.onDeviceCodeReceived?.(d,q),r?.onBrowserOpening?.();try{await Ae(q)}catch(a){process.env.DEBUG&&console.error("[DEBUG] Failed to open browser:",a)}r?.onWaitingForAuthorization?.();let V=Math.max(ae,5)*1e3,ce=Date.now()+oe*1e3;for(;Date.now()<ce;){await ve(V);try{let a=await i.post("v1/oauth/token",{json:{grant_type:"urn:ietf:params:oauth:grant-type:device_code",device_code:S,client_id:D}}).json();r?.onAuthorized?.();let w={};try{w=ke(a.access_token)}catch{}return{accessToken:a.access_token,refreshToken:a.refresh_token??void 0,email:w.email||"",scopes:a.scope.split(" ").filter(R=>R.length>0)}}catch(a){if(a instanceof Se){let R;try{let X=await a.response.json();R=X.code||X.error}catch{continue}switch(R){case"authorization_pending":continue;case"slow_down":V+=5e3;continue;case"access_denied":throw new Error("Authorization denied by user",{cause:a});case"expired_token":throw new Error("Device code expired. Please try again.",{cause:a});default:throw new Error(`Authorization failed: ${R||"Unknown error"}`,{cause:a})}}let w=a instanceof Error?a:new Error("Unknown error during authorization");throw r?.onError?.(w),w}}throw new Error("Device code expired. Please try again.")}finally{n===void 0?delete process.env.NODE_TLS_REJECT_UNAUTHORIZED:process.env.NODE_TLS_REJECT_UNAUTHORIZED=n}}var m=class extends Error{code;constructor(e,t){super(t),this.code=e}};function be(r){try{let e=r.split(".");if(e.length<2)return null;let t=e[1].replace(/-/g,"+").replace(/_/g,"/"),s=(4-t.length%4)%4,n=t.padEnd(t.length+s,"="),i=Buffer.from(n,"base64").toString("utf8");return JSON.parse(i)}catch{return null}}function Ee(r,e=300){let s=be(r)?.exp;if(typeof s!="number")return!1;let n=Math.floor(Date.now()/1e3);return s-n<=e}async function re(r){let e=r.accessToken;if(!Ee(e))return e;if(!r.refreshToken)throw await y.clear(),new m("session-expired","Session expired. Please run `entrydesk login` again.");try{let s=await new K().refreshOAuthToken({refreshToken:r.refreshToken,clientId:D});return await y.saveAccessToken(s.access_token),s.refresh_token&&await y.saveRefreshToken(s.refresh_token),s.access_token}catch{throw await y.clear(),new m("session-expired","Session expired. Please run `entrydesk login` again.")}}async function at(){let r=z();if(r)return r.accessToken;let e=await y.getAll();if(!e?.accessToken)throw new m("not-logged-in","Not logged in. Run `entrydesk login` first.");return re(e)}async function ct(){let r=z();if(r)return{accessToken:r.accessToken,workspaceId:r.workspaceId};let e=ee();if(e)return e;let t=await y.getAll();if(!t?.accessToken)throw new m("not-logged-in","Not logged in. Run `entrydesk login` first.");if(!t.workspaceId)throw new m("no-workspace","No workspace selected. Run `entrydesk workspaces` first.");return{accessToken:await re(t),workspaceId:t.workspaceId}}var f=class extends Error{constructor(t,s=1,n){super(t);this.exitCode=s;this.cause=n;this.name="CLIError"}};function ut(r){let e=!!process.env.DEBUG;r instanceof f&&(console.error(`Error: ${r.message}`),r.cause&&(console.error(`Cause: ${r.cause.message}`),e&&r.cause.stack&&console.error(r.cause.stack)),process.exit(r.exitCode)),r instanceof m&&(console.error(r.message),e&&r.stack&&console.error(r.stack),process.exit(1)),r instanceof xe&&(console.error("Request timed out. Please check your connection and try again."),e&&r.stack&&console.error(r.stack),process.exit(1)),r instanceof Re&&(console.error(`Request failed (${r.response.status}): ${r.response.statusText}`),e&&r.stack&&console.error(r.stack),process.exit(1)),r instanceof Error&&(console.error(`Unexpected error: ${r.message}`),e&&r.stack&&console.error(r.stack),process.exit(1)),console.error("An unknown error occurred"),process.exit(1)}var P="default",_e=/^[A-Za-z0-9_-]+$/,L=null;function H(){return E.join(Te.homedir(),".entrydesk")}function ne(){return E.join(H(),"profiles")}function ie(){return E.join(H(),"profiles.json")}function $(r){return E.join(ne(),r)}function G(r){if(!p.existsSync(r))p.mkdirSync(r,{mode:448,recursive:!0});else try{p.chmodSync(r,448)}catch{}}function Ke(){G(H())}function W(){G(ne())}function Ue(){let r=ie();try{if(!p.existsSync(r))return null;let e=p.readFileSync(r,"utf-8");return JSON.parse(e)}catch{return null}}function b(r){Ke();let e=ie();p.writeFileSync(e,JSON.stringify(r,null,2),{mode:384});try{p.chmodSync(e,384)}catch{}}function Ie(r){let e=r.profiles?.length?Array.from(new Set(r.profiles)):[],t=r.currentProfile&&e.includes(r.currentProfile)?r.currentProfile:e[0]||P;return e.length===0&&e.push(P),e.includes(t)||e.unshift(t),{currentProfile:t,profiles:e}}function k(){let r=Ue();if(!r){let t={currentProfile:P,profiles:[P]};return b(t),W(),_(P),t}let e=Ie(r);return(e.currentProfile!==r.currentProfile||e.profiles.length!==r.profiles.length)&&b(e),W(),e}function C(r){if(!r)throw new f("Profile name is required.");if(!_e.test(r))throw new f("Invalid profile name. Use letters, numbers, dash, or underscore.")}function T(r,e){if(!r.profiles.includes(e))throw new f(`Profile not found: ${e}`)}function se(r,e){if(r.profiles.includes(e))throw new f(`Profile already exists: ${e}`)}function _(r){W(),G($(r))}var o={DEFAULT_PROFILE:P,setOverrideProfile(r){L=r},getOverrideProfile(){return L},getProfileDir(r){return $(r)},getProfileConfigFile(r){return E.join(o.getProfileDir(r),"config.json")},getProfileCredentialsFile(r){return E.join(o.getProfileDir(r),"credentials.json")},getProfileNames(){return[...k().profiles]},getProfileCount(){return o.getProfileNames().length},getCurrentProfileName(){return k().currentProfile},getEffectiveProfileName(r){let e=k(),t=r||L||e.currentProfile;return C(t),T(e,t),t},ensureProfileExists(r){let e=k();C(r),T(e,r)},createProfile(r){C(r);let e=k();se(e,r);let t={currentProfile:e.currentProfile,profiles:[...e.profiles,r]};b(t),_(r)},deleteProfile(r){C(r);let e=k();if(T(e,r),e.profiles.length===1)throw new f("Cannot delete the last profile.");let t=e.profiles.filter(i=>i!==r),s=e.currentProfile;s===r&&(s=t.includes(P)?P:t[0]),b({currentProfile:s,profiles:t});try{p.rmSync(o.getProfileDir(r),{recursive:!0,force:!0})}catch{}return{currentProfile:s}},renameProfile(r,e){C(r),C(e);let t=k();T(t,r),se(t,e);let s=$(r),n=$(e);if(p.existsSync(s))try{p.renameSync(s,n)}catch(d){throw new f("Failed to rename profile directory.",1,d)}else _(e);let i=t.profiles.map(d=>d===r?e:d),c=t.currentProfile===r?e:t.currentProfile;return b({currentProfile:c,profiles:i}),{currentProfile:c}},setCurrentProfile(r){C(r);let e=k();T(e,r);let t={currentProfile:r,profiles:e.profiles};b(t),_(r)},ensureProfileDir:_};function De(r){let e=o.getEffectiveProfileName(r);return o.getProfileConfigFile(e)}function l(r){let e=De(r);try{if(!F.existsSync(e))return{};let t=F.readFileSync(e,"utf-8");return JSON.parse(t)}catch{return{}}}function h(r,e){let t=o.getEffectiveProfileName(e);o.ensureProfileDir(t);let s=o.getProfileConfigFile(t);F.writeFileSync(s,JSON.stringify(r,null,2),{mode:384});try{F.chmodSync(s,384)}catch{}}var Y={getApiUrl(r){return l(r).apiUrl??null},setApiUrl(r,e){let t=l(e);t.apiUrl=r,h(t,e)},clearApiUrl(r){let e=l(r);delete e.apiUrl,h(e,r)},getHubUrl(r){return l(r).hubUrl??null},setHubUrl(r,e){let t=l(e);t.hubUrl=r,h(t,e)},getAll(r){return l(r)},setAll(r,e){h(r,e)},set(r,e,t){let s=l(t);s[r]=e,h(s,t)},get(r,e){return l(e)[r]},unset(r,e){let t=l(e);delete t[r],h(t,e)},getDefaultModel(r){return l(r).model?.default??null},setDefaultModel(r,e){let t=l(e);t.model={...t.model,default:r},h(t,e)},getDefaultConnectorIds(r){return l(r).connectors?.defaults??null},setDefaultConnectorIds(r,e){let t=l(e);t.connectors={...t.connectors,defaults:r},h(t,e)},getCapabilities(r){return l(r).capabilities??null},setCapabilities(r,e){let t=l(e);t.capabilities={...t.capabilities,...r},h(t,e)},getLocalTools(r){return l(r).localTools??null},setLocalTools(r,e){let t=l(e);t.localTools={...t.localTools,...r},h(t,e)}};var $e="https://entrydesk-api.dcard.io",Fe="https://entrydesk.dcard.io",Be=3100,Oe=300*1e3;function B(r){return{apiUrl:Y.getApiUrl(r)||process.env.ENTRYDESK_API_URL||$e,hubUrl:Y.getHubUrl(r)||process.env.ENTRYDESK_HUB_URL||Fe,mcp:{httpPort:Be},auth:{loginTimeout:Oe,ignoreSslErrors:process.env.ENTRYDESK_IGNORE_SSL_ERRORS==="true"||!1}}}function U(r){return B(r).apiUrl}function te(r){return B(r).hubUrl}function Ct(){return B().mcp}function I(){return B().auth}export{D as a,tt as b,y as c,m as d,Ee as e,re as f,at as g,ct as h,f as i,ut as j,o as k,Y as l,B as m,U as n,te as o,Ct as p,I as q,K as r};
|
|
2
|
+
import F from"fs";import p from"fs";import Te from"os";import E from"path";import{HTTPError as Re,TimeoutError as xe}from"ky";import he,{TimeoutError as ye}from"ky";import x from"crypto";import u from"fs";import le from"path";var ge="ENTRYDESK_FORCE_FILE_STORAGE",A="EntryDesk CLI",de="__entrydesk_keychain_test__",ue="profile:",fe=".credentials.key",Z={keychain:!1,permissions:!1,corruptFile:!1,windowsFile:!1};function v(r,e){Z[r]||(Z[r]=!0,console.error(e))}function Q(r){if(!r||typeof r!="object")return!1;let e=r;return!(!("accessToken"in e||"refreshToken"in e||"email"in e||"workspaceId"in e)||"accessToken"in e&&typeof e.accessToken!="string"||"refreshToken"in e&&typeof e.refreshToken!="string"||"email"in e&&typeof e.email!="string"||"workspaceId"in e&&typeof e.workspaceId!="string")}function pe(r){if(!r||typeof r!="object")return!1;let e=r;return typeof e.getPassword=="function"&&typeof e.setPassword=="function"&&typeof e.deletePassword=="function"}function O(r){return o.getProfileCredentialsFile(r)}var N=class{getKeyFilePath(e){return le.join(o.getProfileDir(e),fe)}loadEncryptionKey(e){let t=this.getKeyFilePath(e);try{if(!u.existsSync(t))return null;let s=u.readFileSync(t);return s.length!==32?null:(this.ensureSecureFilePermissions(t),s)}catch{return null}}createEncryptionKey(e){o.ensureProfileDir(e);let t=this.getKeyFilePath(e),s=x.randomBytes(32);return u.writeFileSync(t,s,{mode:384}),this.ensureSecureFilePermissions(t),s}getOrCreateEncryptionKey(e){let t=this.loadEncryptionKey(e);return t||this.createEncryptionKey(e)}encrypt(e,t){let s=x.randomBytes(16),n=x.createCipheriv("aes-256-gcm",t,s),i=n.update(e,"utf8","hex");i+=n.final("hex");let c=n.getAuthTag();return`${s.toString("hex")}:${c.toString("hex")}:${i}`}decrypt(e,t){let s=e.split(":");if(s.length!==3)throw new Error("Invalid encrypted data format");let n=Buffer.from(s[0],"hex"),i=Buffer.from(s[1],"hex"),c=s[2],S=x.createDecipheriv("aes-256-gcm",t,n,{authTagLength:16});S.setAuthTag(i);let d=S.update(c,"hex","utf8");return d+=S.final("utf8"),d}tryDecrypt(e,t){try{return this.decrypt(e,t)}catch{return null}}tryParseCredentials(e){try{let t=JSON.parse(e);return Q(t)?t:null}catch{return null}}ensureSecureFilePermissions(e){if(process.platform!=="win32"){try{u.chmodSync(e,384)}catch{v("permissions","Warning: Failed to set secure permissions on credentials file.");return}try{(u.statSync(e).mode&511)!==384&&v("permissions","Warning: Credentials file permissions are too permissive.")}catch{v("permissions","Warning: Failed to verify credentials file permissions.")}}}async getCredentials(e){let t=O(e);try{if(!u.existsSync(t))return null;let s=u.readFileSync(t,"utf-8").trim();if(!s)return null;let n=this.loadEncryptionKey(e),i=n?this.tryDecrypt(s,n):null;if(i){let c=this.tryParseCredentials(i);if(c)return c}return v("corruptFile","Warning: Credentials file is unreadable. Please run `entrydesk login` again."),null}catch{return null}}async setCredentials(e,t){o.ensureProfileDir(e);let s=O(e),n=this.getOrCreateEncryptionKey(e),i=this.encrypt(JSON.stringify(t),n);u.writeFileSync(s,i,{mode:384}),this.ensureSecureFilePermissions(s)}async deleteCredentials(e){let t=O(e);try{u.existsSync(t)&&u.unlinkSync(t)}catch{}}},M=class{keytarModule=null;keytarLoadAttempted=!1;keychainAvailable=null;async getKeytar(){if(this.keytarLoadAttempted)return this.keytarModule;this.keytarLoadAttempted=!0;try{let t=await import("keytar"),s=t&&typeof t=="object"&&"default"in t?t.default:t;this.keytarModule=pe(s)?s:null}catch{this.keytarModule=null}return this.keytarModule}getAccount(e){return`${ue}${e}`}async checkKeychainAvailability(){if(this.keychainAvailable!==null)return this.keychainAvailable;try{let e=await this.getKeytar();if(!e)return this.keychainAvailable=!1,!1;let t=`${de}${x.randomBytes(8).toString("hex")}`,s="test";await e.setPassword(A,t,s);let n=await e.getPassword(A,t),i=await e.deletePassword(A,t);return this.keychainAvailable=!!(i&&n===s),this.keychainAvailable}catch{return this.keychainAvailable=!1,!1}}async isAvailable(){return this.checkKeychainAvailability()}async getCredentials(e){if(!await this.checkKeychainAvailability())throw new Error("Keychain is not available");let t=await this.getKeytar();if(!t)throw new Error("Keytar module not available");let s=await t.getPassword(A,this.getAccount(e));if(!s)return null;let n=this.tryParseKeychainPayload(s);return n||v("corruptFile",`Warning: Keychain entry for profile "${e}" is unreadable.`),n}async setCredentials(e,t){if(!await this.checkKeychainAvailability())throw new Error("Keychain is not available");let s=await this.getKeytar();if(!s)throw new Error("Keytar module not available");let n=JSON.stringify(t);await s.setPassword(A,this.getAccount(e),n)}async deleteCredentials(e){if(!await this.checkKeychainAvailability())throw new Error("Keychain is not available");let t=await this.getKeytar();if(!t)throw new Error("Keytar module not available");await t.deletePassword(A,this.getAccount(e))}tryParseKeychainPayload(e){try{let t=JSON.parse(e);return Q(t)?t:null}catch{return null}}},j=class{keychainStorage=null;fileStorage=new N;keychainAvailable=null;initPromise=null;shouldForceFileStorage(){return process.env[ge]?.toLowerCase()==="true"}async initializeStorage(){if(this.shouldForceFileStorage()){this.keychainAvailable=!1,this.keychainStorage=null;return}try{let e=new M,t=await e.isAvailable();this.keychainAvailable=t,this.keychainStorage=t?e:null}catch{this.keychainAvailable=!1,this.keychainStorage=null}!this.keychainAvailable&&!this.shouldForceFileStorage()&&(v("keychain","Keychain unavailable. Falling back to file-based credentials."),process.platform==="win32"&&v("windowsFile","Warning: File-based credentials on Windows may be readable by other users."))}async ensureInitialized(){if(this.shouldForceFileStorage()){this.keychainAvailable=!1,this.keychainStorage=null;return}this.keychainAvailable===null&&(this.initPromise||(this.initPromise=this.initializeStorage()),await this.initPromise)}disableKeychain(){this.shouldForceFileStorage()||v("keychain","Keychain error encountered. Falling back to file-based credentials."),this.keychainAvailable=!1,this.keychainStorage=null}async getKeychainStorage(){return await this.ensureInitialized(),this.shouldForceFileStorage()?null:this.keychainAvailable?this.keychainStorage:null}async getCredentials(e){let t=await this.getKeychainStorage();if(!t)return this.fileStorage.getCredentials(e);try{let n=await t.getCredentials(e);if(n)return n}catch{return this.disableKeychain(),this.fileStorage.getCredentials(e)}let s=await this.fileStorage.getCredentials(e);if(s)try{await t.setCredentials(e,s),await this.fileStorage.deleteCredentials(e)}catch{}return s}async setCredentials(e,t){let s=await this.getKeychainStorage();if(!s){await this.fileStorage.setCredentials(e,t);return}try{await s.setCredentials(e,t),await this.fileStorage.deleteCredentials(e)}catch{this.disableKeychain(),await this.fileStorage.setCredentials(e,t)}}async deleteCredentials(e){let t=await this.getKeychainStorage();if(t)try{await t.deleteCredentials(e)}catch{this.disableKeychain()}await this.fileStorage.deleteCredentials(e)}},g=new j,y={async saveAccessToken(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.accessToken=r,await g.setCredentials(t,s)},async getAccessToken(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.accessToken??null},async saveRefreshToken(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.refreshToken=r,await g.setCredentials(t,s)},async getRefreshToken(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.refreshToken??null},async saveEmail(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.email=r,await g.setCredentials(t,s)},async getEmail(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.email??null},async saveWorkspaceId(r,e){let t=o.getEffectiveProfileName(e),s=await g.getCredentials(t)||{};s.workspaceId=r,await g.setCredentials(t,s)},async getWorkspaceId(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.workspaceId??null},async saveAll(r,e){let t=o.getEffectiveProfileName(e);await g.setCredentials(t,r)},async getAll(r){let e=o.getEffectiveProfileName(r);return g.getCredentials(e)},async clear(r){let e=o.getEffectiveProfileName(r);await g.deleteCredentials(e)},async isLoggedIn(r){let e=o.getEffectiveProfileName(r);return(await g.getCredentials(e))?.accessToken!=null}};var K=class r{client;baseUrl;constructor(e,t){this.baseUrl=e||U(t),I().ignoreSslErrors&&(process.env.NODE_TLS_REJECT_UNAUTHORIZED="0",process.env.DEBUG&&console.log("[DEBUG] ApiClient: SSL certificate validation disabled")),this.client=he.create({prefixUrl:this.baseUrl,timeout:3e4,retry:{limit:3,methods:["get","put","head","delete","options","trace"],statusCodes:[408,413,429,500,502,503,504]},hooks:{beforeError:[n=>(n.request?.headers&&(n.request.headers.delete("authorization"),n.request.headers.delete("Authorization")),n)]}})}getBaseUrl(){return this.baseUrl}async refreshOAuthToken(e){return this.client.post("v1/oauth/token",{json:{grant_type:"refresh_token",refresh_token:e.refreshToken,client_id:e.clientId}}).json()}async revokeOAuthToken(e){await this.client.post("v1/oauth/revoke",{json:{token:e.token,token_type_hint:"refresh_token",client_id:e.clientId}})}async getWorkspaces(e){try{return process.env.DEBUG&&console.log(`[DEBUG] Calling getWorkspaces API: ${this.baseUrl}/v1/workspaces`),await this.client.get("v1/workspaces",{headers:{Authorization:`Bearer ${e}`}}).json()}catch(t){throw process.env.DEBUG&&(console.error("[DEBUG] getWorkspaces error:",t),t instanceof Error&&"cause"in t&&console.error("[DEBUG] Error cause:",t.cause)),t}}async getAgents(e,t,s){let n={limit:s?.limit??100};return s?.nextKey&&(n.nextKey=s.nextKey),s?.search&&(n.search=s.search),await this.client.get(`v1/workspaces/${t}/availableAgents`,{headers:{Authorization:`Bearer ${e}`},searchParams:n}).json()}async createAgent(e,t,s){return await this.client.post(`v1/workspaces/${t}/agents`,{headers:{Authorization:`Bearer ${e}`},json:s}).json()}async updateAgent(e,t,s,n){await this.client.put(`v1/workspaces/${t}/agents/${s}`,{headers:{Authorization:`Bearer ${e}`},json:n})}async deleteAgent(e,t,s){await this.client.delete(`v1/workspaces/${t}/agents/${s}`,{headers:{Authorization:`Bearer ${e}`}})}async getModels(e,t){return await this.client.get(`v1/workspaces/${t}/models`,{headers:{Authorization:`Bearer ${e}`}}).json()}async getConnectors(e,t){return await this.client.get(`v1/workspaces/${t}/availableConnectors`,{headers:{Authorization:`Bearer ${e}`},searchParams:{limit:100,connectionStatus:"connected"}}).json()}async getChats(e,t,s=20){return await this.client.get(`v1/workspaces/${t}/chats`,{headers:{Authorization:`Bearer ${e}`},searchParams:{limit:s}}).json()}async getMessages(e,t,s,n=50){return await this.client.get(`v1/workspaces/${t}/chats/${s}/messages`,{headers:{Authorization:`Bearer ${e}`},searchParams:{limit:n}}).json()}async getAgent(e,t,s){return await this.client.get(`v1/workspaces/${t}/agents/${s}`,{headers:{Authorization:`Bearer ${e}`}}).json()}async listAvailableSkills(e,t,s){return await this.client.get(`v1/workspaces/${t}/availableSkills`,{headers:{Authorization:`Bearer ${e}`},searchParams:{createdBy:s?.createdBy,nextKey:s?.nextKey,limit:s?.limit??20}}).json()}async uploadSkill(e,t,s){return await this.client.post(`v1/workspaces/${t}/skills`,{headers:{Authorization:`Bearer ${e}`},body:s}).json()}async deleteSkill(e,t,s){await this.client.delete(`v1/workspaces/${t}/skills/${s}`,{headers:{Authorization:`Bearer ${e}`}})}async downloadSkillCurrent(e,t,s){return await this.client.get(`v1/workspaces/${t}/skills/${s}/download`,{headers:{Authorization:`Bearer ${e}`}}).json()}async chat(e,t,s){try{let n=await this.client.post(`v1/workspaces/${t}/chats`,{headers:{Authorization:`Bearer ${e}`},json:s,timeout:6e4,throwHttpErrors:!1});if(!n.ok){let i=await n.text();throw new Error(`Chat API failed (${n.status}): ${i}`)}return n}catch(n){throw n instanceof ye?new TypeError("Request timed out. Please check your connection and try again.",{cause:n}):n}}async getBudget(e,t){return await this.client.get(`v1/workspaces/${t}/budget`,{headers:{Authorization:`Bearer ${e}`}}).json()}async updateBudget(e,t,s){return await this.client.put(`v1/workspaces/${t}/budget`,{headers:{Authorization:`Bearer ${e}`},json:{budget:s}}).json()}async getMonthlyUsages(e,t,s){return await this.client.get(`v1/workspaces/${t}/monthlyUsages`,{headers:{Authorization:`Bearer ${e}`},searchParams:{nextKey:s?.nextKey,limit:s?.limit??20}}).json()}async createChatShare(e,t,s,n="public"){return await this.client.post(`v1/workspaces/${t}/chats/${s}/shares`,{headers:{Authorization:`Bearer ${e}`},json:{visibility:n}}).json()}async getSharedChat(e,t){let s={};return t&&(s.Authorization=`Bearer ${t}`),await this.client.get(`v1/shares/${e}`,{headers:s}).json()}async getSharedMessages(e,t=50,s){let n={};return s&&(n.Authorization=`Bearer ${s}`),await this.client.get(`v1/shares/${e}/messages`,{headers:n,searchParams:{limit:t}}).json()}async deleteChat(e,t,s){await this.client.delete(`v1/workspaces/${t}/chats/${s}`,{headers:{Authorization:`Bearer ${e}`}})}async getModelProviders(e,t){return await this.client.get(`v1/workspaces/${t}/modelProviders`,{headers:{Authorization:`Bearer ${e}`}}).json()}async uploadFile(e,t,s,n){let i=new FormData,c=new Blob([s.buffer],{type:s.mimeType});return i.append("file",c,s.filename),i.append("modelProviderId",n),await this.client.post(`v1/workspaces/${t}/files`,{headers:{Authorization:`Bearer ${e}`},body:i,timeout:12e4}).json()}async getSchedules(e,t,s){let n={limit:s?.limit??20};return s?.nextKey&&(n.nextKey=s.nextKey),s?.search&&(n.search=s.search),await this.client.get(`v1/workspaces/${t}/schedules`,{headers:{Authorization:`Bearer ${e}`},searchParams:n}).json()}async getSchedule(e,t,s){return await this.client.get(`v1/workspaces/${t}/schedules/${s}`,{headers:{Authorization:`Bearer ${e}`}}).json()}async createSchedule(e,t,s){return await this.client.post(`v1/workspaces/${t}/schedules`,{headers:{Authorization:`Bearer ${e}`},json:s}).json()}async updateSchedule(e,t,s,n){await this.client.put(`v1/workspaces/${t}/schedules/${s}`,{headers:{Authorization:`Bearer ${e}`},json:n})}async deleteSchedule(e,t,s){await this.client.delete(`v1/workspaces/${t}/schedules/${s}`,{headers:{Authorization:`Bearer ${e}`}})}static async getAuthenticatedClient(e){let t=await y.getAll(e);return t?{client:new r(void 0,e),accessToken:t.accessToken,workspaceId:t.workspaceId??null}:null}};function me(){return process.env.ENTRYDESK_SANDBOX==="true"}function z(){if(!me())return null;let r=process.env.ENTRYDESK_SANDBOX_ACCESS_TOKEN?.trim(),e=process.env.ENTRYDESK_SANDBOX_WORKSPACE_ID?.trim();if(!r||!e){let t=[];throw r||t.push("ENTRYDESK_SANDBOX_ACCESS_TOKEN"),e||t.push("ENTRYDESK_SANDBOX_WORKSPACE_ID"),new Error(`Sandbox mode enabled but missing credentials: ${t.join(", ")}`)}return{accessToken:r,workspaceId:e}}function ee(){let r=process.env.ENTRYDESK_TOKEN?.trim();if(!r)return null;let e=process.env.ENTRYDESK_WORKSPACE_ID?.trim();if(!e)throw new Error("ENTRYDESK_TOKEN is set but ENTRYDESK_WORKSPACE_ID is missing. Both are required for PAT authentication.");return{accessToken:r,workspaceId:e}}import{execFile as we}from"child_process";import{setTimeout as ve}from"timers/promises";import{decodeJwt as ke}from"jose";import Pe,{HTTPError as Se}from"ky";var D="entrydesk-cli",Ce="offline_access workspaces:read chats:read chats:write chats:delete agents:read agents:write connectors:read connectors:write tools:read tools:execute skills:read skills:write meetings:read meetings:write publishes:read publishes:write calendar:read calendar:write schedules:read schedules:write user:read user:write";function Ae(r){return new Promise((e,t)=>{let s=process.platform,n,i;switch(s){case"darwin":n="open",i=[r];break;case"win32":n="cmd",i=["/c","start","",r];break;default:n="xdg-open",i=[r];break}we(n,i,c=>{c?t(new Error(`Failed to open browser. Please manually visit: ${r}`,{cause:c})):e()})})}async function tt(r){let e=U(),t=te(),s=I(),n=process.env.NODE_TLS_REJECT_UNAUTHORIZED;s.ignoreSslErrors&&(process.env.NODE_TLS_REJECT_UNAUTHORIZED="0",console.error("WARNING: SSL certificate validation is disabled globally for this process during the login flow. All connections in this process will skip TLS verification. Only use ENTRYDESK_IGNORE_SSL_ERRORS for local development."));try{let i=Pe.create({prefixUrl:e,timeout:3e4,hooks:{beforeError:[a=>(a.request?.headers&&(a.request.headers.delete("authorization"),a.request.headers.delete("Authorization")),a)]}}),c;try{c=await i.post("v1/oauth/device/code",{json:{client_id:D,scope:Ce}}).json()}catch(a){let w=a instanceof Error?a:new Error("Failed to request device code");throw r?.onError?.(w),w}let{device_code:S,user_code:d,expires_in:oe,interval:ae}=c,J=new URL("/oauth/device",t);J.searchParams.set("user_code",d);let q=J.toString();r?.onDeviceCodeReceived?.(d,q),r?.onBrowserOpening?.();try{await Ae(q)}catch(a){process.env.DEBUG&&console.error("[DEBUG] Failed to open browser:",a)}r?.onWaitingForAuthorization?.();let V=Math.max(ae,5)*1e3,ce=Date.now()+oe*1e3;for(;Date.now()<ce;){await ve(V);try{let a=await i.post("v1/oauth/token",{json:{grant_type:"urn:ietf:params:oauth:grant-type:device_code",device_code:S,client_id:D}}).json();r?.onAuthorized?.();let w={};try{w=ke(a.access_token)}catch{}return{accessToken:a.access_token,refreshToken:a.refresh_token??void 0,email:w.email||"",scopes:a.scope.split(" ").filter(R=>R.length>0)}}catch(a){if(a instanceof Se){let R;try{let X=await a.response.json();R=X.code||X.error}catch{continue}switch(R){case"authorization_pending":continue;case"slow_down":V+=5e3;continue;case"access_denied":throw new Error("Authorization denied by user",{cause:a});case"expired_token":throw new Error("Device code expired. Please try again.",{cause:a});default:throw new Error(`Authorization failed: ${R||"Unknown error"}`,{cause:a})}}let w=a instanceof Error?a:new Error("Unknown error during authorization");throw r?.onError?.(w),w}}throw new Error("Device code expired. Please try again.")}finally{n===void 0?delete process.env.NODE_TLS_REJECT_UNAUTHORIZED:process.env.NODE_TLS_REJECT_UNAUTHORIZED=n}}var m=class extends Error{code;constructor(e,t){super(t),this.code=e}};function be(r){try{let e=r.split(".");if(e.length<2)return null;let t=e[1].replace(/-/g,"+").replace(/_/g,"/"),s=(4-t.length%4)%4,n=t.padEnd(t.length+s,"="),i=Buffer.from(n,"base64").toString("utf8");return JSON.parse(i)}catch{return null}}function Ee(r,e=300){let s=be(r)?.exp;if(typeof s!="number")return!1;let n=Math.floor(Date.now()/1e3);return s-n<=e}async function re(r){let e=r.accessToken;if(!Ee(e))return e;if(!r.refreshToken)throw await y.clear(),new m("session-expired","Session expired. Please run `entrydesk login` again.");try{let s=await new K().refreshOAuthToken({refreshToken:r.refreshToken,clientId:D});return await y.saveAccessToken(s.access_token),s.refresh_token&&await y.saveRefreshToken(s.refresh_token),s.access_token}catch{throw await y.clear(),new m("session-expired","Session expired. Please run `entrydesk login` again.")}}async function at(){let r=z();if(r)return r.accessToken;let e=await y.getAll();if(!e?.accessToken)throw new m("not-logged-in","Not logged in. Run `entrydesk login` first.");return re(e)}async function ct(){let r=z();if(r)return{accessToken:r.accessToken,workspaceId:r.workspaceId};let e=ee();if(e)return e;let t=await y.getAll();if(!t?.accessToken)throw new m("not-logged-in","Not logged in. Run `entrydesk login` first.");if(!t.workspaceId)throw new m("no-workspace","No workspace selected. Run `entrydesk workspaces` first.");return{accessToken:await re(t),workspaceId:t.workspaceId}}var f=class extends Error{constructor(t,s=1,n){super(t);this.exitCode=s;this.cause=n;this.name="CLIError"}};function ut(r){let e=!!process.env.DEBUG;r instanceof f&&(console.error(`Error: ${r.message}`),r.cause&&(console.error(`Cause: ${r.cause.message}`),e&&r.cause.stack&&console.error(r.cause.stack)),process.exit(r.exitCode)),r instanceof m&&(console.error(r.message),e&&r.stack&&console.error(r.stack),process.exit(1)),r instanceof xe&&(console.error("Request timed out. Please check your connection and try again."),e&&r.stack&&console.error(r.stack),process.exit(1)),r instanceof Re&&(console.error(`Request failed (${r.response.status}): ${r.response.statusText}`),e&&r.stack&&console.error(r.stack),process.exit(1)),r instanceof Error&&(console.error(`Unexpected error: ${r.message}`),e&&r.stack&&console.error(r.stack),process.exit(1)),console.error("An unknown error occurred"),process.exit(1)}var P="default",_e=/^[A-Za-z0-9_-]+$/,L=null;function H(){return E.join(Te.homedir(),".entrydesk")}function ne(){return E.join(H(),"profiles")}function ie(){return E.join(H(),"profiles.json")}function $(r){return E.join(ne(),r)}function G(r){if(!p.existsSync(r))p.mkdirSync(r,{mode:448,recursive:!0});else try{p.chmodSync(r,448)}catch{}}function Ke(){G(H())}function W(){G(ne())}function Ue(){let r=ie();try{if(!p.existsSync(r))return null;let e=p.readFileSync(r,"utf-8");return JSON.parse(e)}catch{return null}}function b(r){Ke();let e=ie();p.writeFileSync(e,JSON.stringify(r,null,2),{mode:384});try{p.chmodSync(e,384)}catch{}}function Ie(r){let e=r.profiles?.length?Array.from(new Set(r.profiles)):[],t=r.currentProfile&&e.includes(r.currentProfile)?r.currentProfile:e[0]||P;return e.length===0&&e.push(P),e.includes(t)||e.unshift(t),{currentProfile:t,profiles:e}}function k(){let r=Ue();if(!r){let t={currentProfile:P,profiles:[P]};return b(t),W(),_(P),t}let e=Ie(r);return(e.currentProfile!==r.currentProfile||e.profiles.length!==r.profiles.length)&&b(e),W(),e}function C(r){if(!r)throw new f("Profile name is required.");if(!_e.test(r))throw new f("Invalid profile name. Use letters, numbers, dash, or underscore.")}function T(r,e){if(!r.profiles.includes(e))throw new f(`Profile not found: ${e}`)}function se(r,e){if(r.profiles.includes(e))throw new f(`Profile already exists: ${e}`)}function _(r){W(),G($(r))}var o={DEFAULT_PROFILE:P,setOverrideProfile(r){L=r},getOverrideProfile(){return L},getProfileDir(r){return $(r)},getProfileConfigFile(r){return E.join(o.getProfileDir(r),"config.json")},getProfileCredentialsFile(r){return E.join(o.getProfileDir(r),"credentials.json")},getProfileNames(){return[...k().profiles]},getProfileCount(){return o.getProfileNames().length},getCurrentProfileName(){return k().currentProfile},getEffectiveProfileName(r){let e=k(),t=r||L||e.currentProfile;return C(t),T(e,t),t},ensureProfileExists(r){let e=k();C(r),T(e,r)},createProfile(r){C(r);let e=k();se(e,r);let t={currentProfile:e.currentProfile,profiles:[...e.profiles,r]};b(t),_(r)},deleteProfile(r){C(r);let e=k();if(T(e,r),e.profiles.length===1)throw new f("Cannot delete the last profile.");let t=e.profiles.filter(i=>i!==r),s=e.currentProfile;s===r&&(s=t.includes(P)?P:t[0]),b({currentProfile:s,profiles:t});try{p.rmSync(o.getProfileDir(r),{recursive:!0,force:!0})}catch{}return{currentProfile:s}},renameProfile(r,e){C(r),C(e);let t=k();T(t,r),se(t,e);let s=$(r),n=$(e);if(p.existsSync(s))try{p.renameSync(s,n)}catch(d){throw new f("Failed to rename profile directory.",1,d)}else _(e);let i=t.profiles.map(d=>d===r?e:d),c=t.currentProfile===r?e:t.currentProfile;return b({currentProfile:c,profiles:i}),{currentProfile:c}},setCurrentProfile(r){C(r);let e=k();T(e,r);let t={currentProfile:r,profiles:e.profiles};b(t),_(r)},ensureProfileDir:_};function De(r){let e=o.getEffectiveProfileName(r);return o.getProfileConfigFile(e)}function l(r){let e=De(r);try{if(!F.existsSync(e))return{};let t=F.readFileSync(e,"utf-8");return JSON.parse(t)}catch{return{}}}function h(r,e){let t=o.getEffectiveProfileName(e);o.ensureProfileDir(t);let s=o.getProfileConfigFile(t);F.writeFileSync(s,JSON.stringify(r,null,2),{mode:384});try{F.chmodSync(s,384)}catch{}}var Y={getApiUrl(r){return l(r).apiUrl??null},setApiUrl(r,e){let t=l(e);t.apiUrl=r,h(t,e)},clearApiUrl(r){let e=l(r);delete e.apiUrl,h(e,r)},getHubUrl(r){return l(r).hubUrl??null},setHubUrl(r,e){let t=l(e);t.hubUrl=r,h(t,e)},getAll(r){return l(r)},setAll(r,e){h(r,e)},set(r,e,t){let s=l(t);s[r]=e,h(s,t)},get(r,e){return l(e)[r]},unset(r,e){let t=l(e);delete t[r],h(t,e)},getDefaultModel(r){return l(r).model?.default??null},setDefaultModel(r,e){let t=l(e);t.model={...t.model,default:r},h(t,e)},getDefaultConnectorIds(r){return l(r).connectors?.defaults??null},setDefaultConnectorIds(r,e){let t=l(e);t.connectors={...t.connectors,defaults:r},h(t,e)},getCapabilities(r){return l(r).capabilities??null},setCapabilities(r,e){let t=l(e);t.capabilities={...t.capabilities,...r},h(t,e)},getLocalTools(r){return l(r).localTools??null},setLocalTools(r,e){let t=l(e);t.localTools={...t.localTools,...r},h(t,e)}};var $e="https://api.entrydesk.com",Fe="https://app.entrydesk.com",Be=3100,Oe=300*1e3;function B(r){return{apiUrl:Y.getApiUrl(r)||process.env.ENTRYDESK_API_URL||$e,hubUrl:Y.getHubUrl(r)||process.env.ENTRYDESK_HUB_URL||Fe,mcp:{httpPort:Be},auth:{loginTimeout:Oe,ignoreSslErrors:process.env.ENTRYDESK_IGNORE_SSL_ERRORS==="true"||!1}}}function U(r){return B(r).apiUrl}function te(r){return B(r).hubUrl}function Ct(){return B().mcp}function I(){return B().auth}export{D as a,tt as b,y as c,m as d,Ee as e,re as f,at as g,ct as h,f as i,ut as j,o as k,Y as l,B as m,U as n,te as o,Ct as p,I as q,K as r};
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{m as a,n as b,o as c,p as d,q as e}from"./chunk-
|
|
2
|
+
import{m as a,n as b,o as c,p as d,q as e}from"./chunk-ZGVZBGOQ.js";export{b as getApiUrl,e as getAuthConfig,a as getConfig,c as getHubUrl,d as getMcpConfig};
|
package/dist/index.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{a as nn,b as sn,c as Se,d as Xe,e as rn,f as an,g as ct,h as fe,i as _t,j as ke,k as G,l as oe,m as Ze,r as V}from"./chunk-
|
|
2
|
+
import{a as nn,b as sn,c as Se,d as Xe,e as rn,f as an,g as ct,h as fe,i as _t,j as ke,k as G,l as oe,m as Ze,r as V}from"./chunk-ZGVZBGOQ.js";import{render as Vi}from"ink";function To(){console.log(`
|
|
3
3
|
Usage: entrydesk agents [list|create|update|delete] [options]
|
|
4
4
|
|
|
5
5
|
Commands:
|
|
@@ -200,7 +200,7 @@ ID: ${u.id}`)}}}else if(f==="connectors"){let c=Array.from(nt);S(c.length>0?c:vo
|
|
|
200
200
|
`)||it.length===0)return;if(Ge===null){yt(e);let u=it.length-1;qe(u),t(it[u]);return}let c=Math.max(0,Ge-1);qe(c),t(it[c]);return}if(g.downArrow){if(e.includes(`
|
|
201
201
|
`)||Ge===null)return;if(Ge>=it.length-1){qe(null),t(Yt);return}let c=Ge+1;qe(c),t(it[c])}}),{suggestions:q,suggestionIndex:Te,showSuggestions:ne,argMode:f,argItems:T,argIndex:U,selectedConnectorIds:nt,selectedCapabilities:Le,deleteConfirmMode:at,handleInputChange:Xt,handleSubmit:Zt}}import{useCallback as Mt,useRef as jr,useState as Fr}from"react";function is({credentials:e,currentModel:t,modelProviders:o,onUploadSuccess:s,onUploadError:n}){let[r,i]=Fr([]),a=jr(0),l=Mt(h=>{if(!(!Array.isArray(o)||o.length===0)){for(let b of o)if(b.models.some(K=>K.id===h))return b.id;return o[0]?.id}},[o]),d=Mt(async h=>{if(!e){n?.(h.filename,"Not logged in");return}if(!Array.isArray(o)||o.length===0){n?.(h.filename,"Model providers not loaded yet. Please wait and try again.");return}let b=t?l(t):o[0]?.id;if(!b){n?.(h.filename,"No model provider available");return}let L=`file-${++a.current}`;i(K=>[...K,{id:L,filename:h.filename,mimeType:h.mimeType,status:"uploading"}]);try{let F=await new V().uploadFile(e.accessToken,e.workspaceId,h,b);i(re=>re.map(Y=>Y.id===L?{...Y,status:"uploaded",fileId:F.id}:Y)),s?.(h.filename)}catch(K){let F=K instanceof Error?K.message:"Upload failed";i(re=>re.map(Y=>Y.id===L?{...Y,status:"error",error:F}:Y)),n?.(h.filename,F)}},[e,t,l,o,s,n]),k=Mt(h=>{i(b=>b.filter(L=>L.id!==h))},[]),C=Mt(()=>{i([])},[]),$=Mt(()=>r.filter(h=>h.status==="uploaded"&&h.fileId).map(h=>h.fileId),[r]),S=r.some(h=>h.status==="uploading"),y=r.length>0;return{pendingFiles:r,addFile:d,removeFile:k,clearFiles:C,getUploadedFileIds:$,hasUploadingFiles:S,hasFiles:y}}import{jsx as X,jsxs as ue}from"react/jsx-runtime";var Mr=[" ______ _ ____ _","| ____| | | | _ \\ | |","| |__ _ __ | |_ _ __ _ _ | | | | ___ ___| | __","| __| | '_ \\| __| '__| | | || | | |/ _ \\/ __| |/ /","| |____| | | | |_| | | |_| || |_| | __/\\__ \\ <","|______|_| |_|\\__|_| \\__, ||____/ \\___||___/_|\\_\\"," __/ |"," |___/"],Br=[Wo,Wo,Fo,Fo,pt,pt,pt,pt],ls=Mr.map((e,t)=>`${Br[t]??pt}${e}${se}`).join(`
|
|
202
202
|
`),Jr=`${pt}EntryDesk${se}`,Gr=500;function Vr(e){let t="",o=0;for(;o<e.length;){if(e.charCodeAt(o)===27&&e[o+1]==="["){for(o+=2;o<e.length;){let s=e.charCodeAt(o);if(s>=64&&s<=126){o+=1;break}o+=1}continue}t+=e[o],o+=1}return t}function Hr(e){let t=Vr(e);return Math.max(...t.split(`
|
|
203
|
-
`).map(o=>o.length))}function Kr(e){let t=Hr(ls);return e>=t?ls:Jr}function cs({agentId:e,model:t,message:o,chatId:s,chatTitle:n,connectorIds:r,enableSandbox:i=!0,enableWebSearch:a=!1,enableImageGeneration:l=!1,enableChart:d=!1,enableFileSystem:k=!0}){let C=process.stdout.columns??80,$=Kr(C),S=as(null),y=Mn(s,n),{state:h,setState:b,messages:L,setMessages:K,input:F,setInput:re,chatId:Y,setChatId:$e,currentChatTitle:j,setCurrentChatTitle:A,hasPersistedChat:N,setHasPersistedChat:R,credentials:M,setCredentials:E,currentResponse:ge,setCurrentResponse:_,fullOutput:p,setFullOutput:ie,addSystemMessage:we,startNewChat:z}=y,{credentials:P}=Jn(b);po(()=>{P&&!M&&E(P)},[P,M,E]);let{availableConnectors:ee,availableModels:Q,availableAgents:pe,modelProviders:le,loading:B}=Gn(M),{currentModel:Z,setCurrentModel:q,currentAgentId:de,setCurrentAgentId:Te,activeConnectorIds:W,setActiveConnectorIds:ne}=Qn(t,e,r),O=as(!1);po(()=>{if(!O.current){if(O.current=!0,!t&&!e){let x=oe.getDefaultModel();x&&q(x)}if(!r&&!e){let x=oe.getDefaultConnectorIds();x&&ne(x.length>0?x:void 0)}}},[t,e,r,q,ne]);let{currentAgentDetails:f}=Zn(de,M),{sandbox:w,setSandbox:T,webSearch:te,setWebSearch:U,imageGeneration:be,setImageGeneration:nt,chart:st,setChart:Le,fileSystem:rt,setFileSystem:Co,capabilityLabelMap:zt}=Xn(i,a,l,d,k),{pendingFiles:We,addFile:ht,removeFile:ze,clearFiles:qt,getUploadedFileIds:it,hasUploadingFiles:$o,hasFiles:Ge}=is({credentials:M,currentModel:Z,modelProviders:le,onUploadSuccess:x=>{we(`\u2713 Attached: ${x}`)},onUploadError:(x,D)=>{we(`\u2717 Failed to upload ${x}: ${D}`)}}),{sendMessage:qe}=Kn({chatId:Y,credentials:M,currentAgentId:de,currentModel:Z,activeConnectorIds:W,sandbox:w,webSearch:te,imageGeneration:be,chart:st,localTools:rt,fullOutput:p,setMessages:K,setState:b,setCurrentResponse:_,setCredentials:E,setHasPersistedChat:R,getFileIds:it,clearFiles:qt}),Yt=Vo(x=>{z(x),_("")},[z]),[yt,ft]=Wr(!1),at=Vo(async x=>{let D=x.text.trim(),me=co(D);if(/^https?:\/\//.test(me))return;if(await Un(D)&&Rn(me)){let ce=await jn(D);ce&&(x.preventDefault(),await ht(ce))}},[ht]),wt=Vo(async()=>{if(!(yt||h.status!=="ready")){if(B||le.length===0){we("Please wait for resources to load before pasting images.");return}we("Reading clipboard..."),ft(!0);try{let x=await Ln();if(x){let D=`clipboard-${Date.now()}.png`;we(`Uploading: ${D} (${Math.round(x.buffer.length/1024)}KB)`),await ht({buffer:x.buffer,filename:D,mimeType:x.mime})}else we("No image found in clipboard. Copy an image first.")}catch(x){let D=x instanceof Error?x.message:"Failed to read clipboard";we(`Clipboard error: ${D}`)}finally{ft(!1)}}},[ht,we,yt,h.status,B,le.length]),{suggestions:kt,suggestionIndex:xt,showSuggestions:Ee,argMode:he,argItems:Ye,argIndex:bt,selectedConnectorIds:Xt,selectedCapabilities:Et,deleteConfirmMode:Zt,handleInputChange:v,handleSubmit:g}=rs({input:F,setInput:re,chatId:Y,setChatId:$e,currentChatTitle:j,setCurrentChatTitle:A,currentModel:Z,setCurrentModel:q,currentAgentId:de,setCurrentAgentId:Te,currentAgentDetails:f,setCurrentAgentDetails:()=>{},activeConnectorIds:W,setActiveConnectorIds:ne,sandbox:w,setSandbox:T,webSearch:te,setWebSearch:U,imageGeneration:be,setImageGeneration:nt,chart:st,setChart:Le,fileSystem:rt,setFileSystem:Co,availableModels:Q,availableAgents:pe,availableConnectors:ee,messages:L,hasPersistedChat:N,setHasPersistedChat:R,setMessages:K,addSystemMessage:we,startNewChat:Yt,sendMessage:qe,setFullOutput:ie,credentials:M,setCredentials:E,setState:b,defaultCapabilities:{sandbox:i,webSearch:a,imageGeneration:l,chart:d,fileSystem:k},capabilityLabelMap:zt,hasUploadingFiles:$o});if(Jo((x,D)=>{(D.ctrl||D.meta)&&x==="v"&&wt()},{isActive:h.status==="ready"&&!he}),Jo((x,D)=>{if(!D.escape)return;let me=Date.now(),J=S.current;S.current=me,!(!J||me-J>Gr)&&(he||Ee||(S.current=null,Ge&&qt(),F.length>0&&v("")))},{isActive:h.status==="ready"}),Jo((x,D)=>{D.return&&(Yt("Started a new chat."),b({status:"ready"}))},{isActive:h.status==="error"}),po(()=>{if(M&&!B&&h.status==="loading")if(!Z&&!de){let x=oe.getDefaultModel(),D=x&&Q.some(J=>J.id===x),me=D?x:Q[0]?.id;me?(D||oe.setDefaultModel(me),q(me),b({status:"ready"})):pe.length===0?b({status:"no-model"}):b({status:"ready"})}else b({status:"ready"})},[M,B,Z,de,Q,pe,h.status,q,b]),po(()=>{o&&M&&h.status==="ready"&&qe(o)},[M,o]),h.status==="loading")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),ue(Ae,{children:[X(H,{color:"cyan",children:X(Go,{type:"dots"})}),X(H,{children:" Initializing chat..."})]})]});if(h.status==="not-logged-in")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),X(H,{color:"yellow",children:"Not logged in"}),X(H,{dimColor:!0,children:"Run `entrydesk login` to authenticate first."})]});if(h.status==="no-workspace")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),X(H,{color:"yellow",children:"No workspace selected"}),X(H,{dimColor:!0,children:"Run `entrydesk workspaces` to select a workspace first."})]});if(h.status==="no-model")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),X(H,{color:"yellow",children:"No model or agent specified"}),s&&X(H,{dimColor:!0,children:"Could not determine the model used in this chat."}),X(H,{dimColor:!0,children:"Please specify a model with --model or use an agent with -a."}),X(H,{dimColor:!0,children:"Example: entrydesk chat -c 1 -i --model gemini-2.5-flash-lite"})]});if(h.status==="error")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),ue(H,{color:"red",children:["Error: ",h.message]}),X(H,{dimColor:!0,children:"Press Enter to start a new chat, or Ctrl+C to exit"})]});let c=de?f?f.connectors&&f.connectors.length>0?f.connectors.map(x=>x.name).join(", "):"none":"loading...":W&&W.length>0?ee.filter(x=>W.includes(x.id)).map(x=>x.name).join(", ")||`${W.length} connector(s)`:"none",u=[];if(de)if(f?.capabilities)for(let x of f.capabilities)u.push(zt.get(x)??x);else u.push("loading...");else w&&u.push("Sandbox"),te&&u.push("Web"),be&&u.push("Image"),st&&u.push("Chart");rt&&u.push("FileSystem");let m=G.getCurrentProfileName(),I=m!=="default";return ue(Ae,{flexDirection:"column",children:[ue(Ae,{marginBottom:1,flexDirection:"column",children:[X(H,{children:$}),ue(Ae,{children:[X(H,{bold:!0,color:"cyan",children:j?`Continuing: ${j}`:"EntryDesk CLI"}),ue(H,{dimColor:!0,children:[" v","1.11.
|
|
203
|
+
`).map(o=>o.length))}function Kr(e){let t=Hr(ls);return e>=t?ls:Jr}function cs({agentId:e,model:t,message:o,chatId:s,chatTitle:n,connectorIds:r,enableSandbox:i=!0,enableWebSearch:a=!1,enableImageGeneration:l=!1,enableChart:d=!1,enableFileSystem:k=!0}){let C=process.stdout.columns??80,$=Kr(C),S=as(null),y=Mn(s,n),{state:h,setState:b,messages:L,setMessages:K,input:F,setInput:re,chatId:Y,setChatId:$e,currentChatTitle:j,setCurrentChatTitle:A,hasPersistedChat:N,setHasPersistedChat:R,credentials:M,setCredentials:E,currentResponse:ge,setCurrentResponse:_,fullOutput:p,setFullOutput:ie,addSystemMessage:we,startNewChat:z}=y,{credentials:P}=Jn(b);po(()=>{P&&!M&&E(P)},[P,M,E]);let{availableConnectors:ee,availableModels:Q,availableAgents:pe,modelProviders:le,loading:B}=Gn(M),{currentModel:Z,setCurrentModel:q,currentAgentId:de,setCurrentAgentId:Te,activeConnectorIds:W,setActiveConnectorIds:ne}=Qn(t,e,r),O=as(!1);po(()=>{if(!O.current){if(O.current=!0,!t&&!e){let x=oe.getDefaultModel();x&&q(x)}if(!r&&!e){let x=oe.getDefaultConnectorIds();x&&ne(x.length>0?x:void 0)}}},[t,e,r,q,ne]);let{currentAgentDetails:f}=Zn(de,M),{sandbox:w,setSandbox:T,webSearch:te,setWebSearch:U,imageGeneration:be,setImageGeneration:nt,chart:st,setChart:Le,fileSystem:rt,setFileSystem:Co,capabilityLabelMap:zt}=Xn(i,a,l,d,k),{pendingFiles:We,addFile:ht,removeFile:ze,clearFiles:qt,getUploadedFileIds:it,hasUploadingFiles:$o,hasFiles:Ge}=is({credentials:M,currentModel:Z,modelProviders:le,onUploadSuccess:x=>{we(`\u2713 Attached: ${x}`)},onUploadError:(x,D)=>{we(`\u2717 Failed to upload ${x}: ${D}`)}}),{sendMessage:qe}=Kn({chatId:Y,credentials:M,currentAgentId:de,currentModel:Z,activeConnectorIds:W,sandbox:w,webSearch:te,imageGeneration:be,chart:st,localTools:rt,fullOutput:p,setMessages:K,setState:b,setCurrentResponse:_,setCredentials:E,setHasPersistedChat:R,getFileIds:it,clearFiles:qt}),Yt=Vo(x=>{z(x),_("")},[z]),[yt,ft]=Wr(!1),at=Vo(async x=>{let D=x.text.trim(),me=co(D);if(/^https?:\/\//.test(me))return;if(await Un(D)&&Rn(me)){let ce=await jn(D);ce&&(x.preventDefault(),await ht(ce))}},[ht]),wt=Vo(async()=>{if(!(yt||h.status!=="ready")){if(B||le.length===0){we("Please wait for resources to load before pasting images.");return}we("Reading clipboard..."),ft(!0);try{let x=await Ln();if(x){let D=`clipboard-${Date.now()}.png`;we(`Uploading: ${D} (${Math.round(x.buffer.length/1024)}KB)`),await ht({buffer:x.buffer,filename:D,mimeType:x.mime})}else we("No image found in clipboard. Copy an image first.")}catch(x){let D=x instanceof Error?x.message:"Failed to read clipboard";we(`Clipboard error: ${D}`)}finally{ft(!1)}}},[ht,we,yt,h.status,B,le.length]),{suggestions:kt,suggestionIndex:xt,showSuggestions:Ee,argMode:he,argItems:Ye,argIndex:bt,selectedConnectorIds:Xt,selectedCapabilities:Et,deleteConfirmMode:Zt,handleInputChange:v,handleSubmit:g}=rs({input:F,setInput:re,chatId:Y,setChatId:$e,currentChatTitle:j,setCurrentChatTitle:A,currentModel:Z,setCurrentModel:q,currentAgentId:de,setCurrentAgentId:Te,currentAgentDetails:f,setCurrentAgentDetails:()=>{},activeConnectorIds:W,setActiveConnectorIds:ne,sandbox:w,setSandbox:T,webSearch:te,setWebSearch:U,imageGeneration:be,setImageGeneration:nt,chart:st,setChart:Le,fileSystem:rt,setFileSystem:Co,availableModels:Q,availableAgents:pe,availableConnectors:ee,messages:L,hasPersistedChat:N,setHasPersistedChat:R,setMessages:K,addSystemMessage:we,startNewChat:Yt,sendMessage:qe,setFullOutput:ie,credentials:M,setCredentials:E,setState:b,defaultCapabilities:{sandbox:i,webSearch:a,imageGeneration:l,chart:d,fileSystem:k},capabilityLabelMap:zt,hasUploadingFiles:$o});if(Jo((x,D)=>{(D.ctrl||D.meta)&&x==="v"&&wt()},{isActive:h.status==="ready"&&!he}),Jo((x,D)=>{if(!D.escape)return;let me=Date.now(),J=S.current;S.current=me,!(!J||me-J>Gr)&&(he||Ee||(S.current=null,Ge&&qt(),F.length>0&&v("")))},{isActive:h.status==="ready"}),Jo((x,D)=>{D.return&&(Yt("Started a new chat."),b({status:"ready"}))},{isActive:h.status==="error"}),po(()=>{if(M&&!B&&h.status==="loading")if(!Z&&!de){let x=oe.getDefaultModel(),D=x&&Q.some(J=>J.id===x),me=D?x:Q[0]?.id;me?(D||oe.setDefaultModel(me),q(me),b({status:"ready"})):pe.length===0?b({status:"no-model"}):b({status:"ready"})}else b({status:"ready"})},[M,B,Z,de,Q,pe,h.status,q,b]),po(()=>{o&&M&&h.status==="ready"&&qe(o)},[M,o]),h.status==="loading")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),ue(Ae,{children:[X(H,{color:"cyan",children:X(Go,{type:"dots"})}),X(H,{children:" Initializing chat..."})]})]});if(h.status==="not-logged-in")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),X(H,{color:"yellow",children:"Not logged in"}),X(H,{dimColor:!0,children:"Run `entrydesk login` to authenticate first."})]});if(h.status==="no-workspace")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),X(H,{color:"yellow",children:"No workspace selected"}),X(H,{dimColor:!0,children:"Run `entrydesk workspaces` to select a workspace first."})]});if(h.status==="no-model")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),X(H,{color:"yellow",children:"No model or agent specified"}),s&&X(H,{dimColor:!0,children:"Could not determine the model used in this chat."}),X(H,{dimColor:!0,children:"Please specify a model with --model or use an agent with -a."}),X(H,{dimColor:!0,children:"Example: entrydesk chat -c 1 -i --model gemini-2.5-flash-lite"})]});if(h.status==="error")return ue(Ae,{flexDirection:"column",children:[X(H,{children:$}),ue(H,{color:"red",children:["Error: ",h.message]}),X(H,{dimColor:!0,children:"Press Enter to start a new chat, or Ctrl+C to exit"})]});let c=de?f?f.connectors&&f.connectors.length>0?f.connectors.map(x=>x.name).join(", "):"none":"loading...":W&&W.length>0?ee.filter(x=>W.includes(x.id)).map(x=>x.name).join(", ")||`${W.length} connector(s)`:"none",u=[];if(de)if(f?.capabilities)for(let x of f.capabilities)u.push(zt.get(x)??x);else u.push("loading...");else w&&u.push("Sandbox"),te&&u.push("Web"),be&&u.push("Image"),st&&u.push("Chart");rt&&u.push("FileSystem");let m=G.getCurrentProfileName(),I=m!=="default";return ue(Ae,{flexDirection:"column",children:[ue(Ae,{marginBottom:1,flexDirection:"column",children:[X(H,{children:$}),ue(Ae,{children:[X(H,{bold:!0,color:"cyan",children:j?`Continuing: ${j}`:"EntryDesk CLI"}),ue(H,{dimColor:!0,children:[" v","1.11.1"]}),I&&ue(H,{color:"magenta",children:[" [",m,"]"]}),X(H,{dimColor:!0,children:" (type /help for commands)"})]})]}),L.map((x,D)=>{let me=x.parts&&p?mt(x.parts,!0):x.content,J=x.role==="user"?"green":x.role==="system"?"yellow":"blue",ce=x.role==="user"?"You":x.role==="system"?"System":"Assistant";return ue(Ae,{flexDirection:"column",marginBottom:1,children:[ue(H,{bold:!0,color:J,children:[ce,":"]}),X(jo,{text:me})]},D)}),ge&&ue(Ae,{flexDirection:"column",marginBottom:1,children:[X(H,{bold:!0,color:"blue",children:"Assistant:"}),X(jo,{text:ge})]}),h.status==="ready"&&X(wn,{suggestions:kt,activeIndex:xt,visible:Ee}),h.status==="ready"&&he&&X(yn,{items:Ye,activeIndex:bt,pageSize:he==="agent"||he==="skills"||he==="model"?5:10,selectedIds:he==="connectors"?Xt:he==="capabilities"?Et:void 0,multiSelect:he==="connectors"||he==="capabilities",visible:!0,title:he==="model"?"Select model:":he==="agent"?"Select agent:":he==="skills"?"Select skill:":he==="connectors"?"Select connectors:":he==="capabilities"?"Select capabilities:":"Select chat:"}),h.status==="sending"&&!ge?ue(Ae,{children:[X(H,{color:"cyan",children:X(Go,{type:"dots"})}),X(H,{children:" Thinking..."})]}):h.status==="ready"&&!he?ue(Ae,{flexDirection:"column",children:[Zt&&X(Ae,{marginBottom:1,children:X(H,{color:"yellow",children:"\u26A0\uFE0F Delete this chat? (y/N) "})}),Ge&&!Zt&&ue(Ae,{marginBottom:1,flexDirection:"row",gap:1,children:[X(H,{dimColor:!0,children:"Attachments:"}),We.map((x,D)=>ue(H,{children:[x.status==="uploading"?ue(H,{color:"yellow",children:[X(Go,{type:"dots"})," ",x.filename]}):x.status==="error"?ue(H,{color:"red",children:["\u2717 ",x.filename]}):ue(H,{color:"green",children:["\u2713 ",x.filename]}),D<We.length-1&&X(H,{children:", "})]},x.id))]}),ue(Ae,{children:[X(H,{color:"green",children:"> "}),X(In,{value:F,onChange:v,onSubmit:g,onPaste:at,leadingOffset:2,multiline:!0})]})]}):null,ue(Ae,{marginTop:1,flexDirection:"column",children:[ue(Ae,{gap:2,children:[ue(H,{children:[X(H,{dimColor:!0,children:"Model: "}),X(H,{color:"magenta",children:de?f?.model?.name||"loading...":Q.find(x=>x.id===Z)?.name||Z||"none"})]}),ue(H,{children:[X(H,{dimColor:!0,children:"Connectors: "}),X(H,{color:"cyan",children:c})]}),ue(H,{children:[X(H,{dimColor:!0,children:"Capabilities: "}),X(H,{color:"green",children:u.length>0?u.join(", "):"none"})]})]}),ue(H,{dimColor:!0,children:["Esc Esc: Clear input & attachments | Ctrl+V: Paste image | Ctrl+O: Toggle full output ",p?"(ON)":"(OFF)"," | Ctrl+C: Exit"]})]})]})}import{jsx as Zr}from"react/jsx-runtime";async function qr(){let e=[];for await(let t of process.stdin)e.push(t);return Buffer.concat(e).toString("utf8").trim()}async function Yr(e,t,o,s){if(s<1)return console.error("Chat number must be 1 or greater"),null;let{chats:n}=await e.getChats(t,o,20);if(n.length===0)return console.error("No chats found. Start a new chat first."),null;if(s>n.length)return console.error(`Chat number ${s} not found. You have ${n.length} recent chats.`),console.error("Use `entrydesk chats` to see available chats."),null;let r=n[s-1],i;if(!r.agentId){let{messages:a}=await e.getMessages(t,o,r.id,10);i=a.find(d=>d.role==="assistant"&&d.modelId)?.modelId}return{chat:r,modelId:i}}var Tt="\x1B[2m",Oe="\x1B[0m",Ho="\x1B[36m",ps="\x1B[33m",go="\x1B[32m",us="stream-json";function ds(e,t,o,s){e&&t&&(console.error(`Error: Cannot use both ${o} and ${s}`),process.exit(1))}function ho(e,t){let o=t===void 0?{event:e}:{event:e,data:t};process.stdout.write(`${JSON.stringify(o)}
|
|
204
204
|
`)}function yo(e,t){if(t)return e;let o=e.split(`
|
|
205
205
|
`);if(o.length<=20)return e;let s=o.slice(0,5),n=o.slice(-5),r=o.length-10;return[...s,`${Tt}... (${r} lines omitted, use --full-output to see all) ...${Oe}`,...n].join(`
|
|
206
206
|
`)}function Xr(e){try{return JSON.stringify(e,null,2)}catch{return String(e)}}function fs(e,t){if(!e)return"(no result)";if(typeof e!="object")return yo(String(e),t);let o=e;if("structuredContent"in o&&o.structuredContent){let s=o.structuredContent,n=[];if(s.stdout&&typeof s.stdout=="string"&&s.stdout.trim()&&n.push(s.stdout.trim()),s.stderr&&typeof s.stderr=="string"&&s.stderr.trim()&&n.push(`${ps}[stderr]${Oe} ${s.stderr.trim()}`),s.urls&&Array.isArray(s.urls)){let r=s.urls;for(let i of r)i.path&&i.url&&n.push(`${i.path}: ${i.url}`)}return n.length===0&&n.push(JSON.stringify(s,null,2)),yo(n.join(`
|
|
@@ -402,7 +402,7 @@ Local Development:
|
|
|
402
402
|
$ entrydesk config set hubUrl http://localhost:3000
|
|
403
403
|
|
|
404
404
|
For more information, visit https://entrydesk.com/docs/cli
|
|
405
|
-
`)}async function Ss(e){if(process.env.DEBUG){let{getApiUrl:r,getHubUrl:i}=await import("./config-
|
|
405
|
+
`)}async function Ss(e){if(process.env.DEBUG){let{getApiUrl:r,getHubUrl:i}=await import("./config-OY6KLQUX.js");console.log(`[DEBUG] Hub URL: ${i()}`),console.log(`[DEBUG] API URL: ${r()}`)}let t=await sn({onDeviceCodeReceived:(r,i)=>{e.json||(console.log(""),console.log("To authenticate, please:"),console.log(""),console.log(` 1. Visit: ${i}`),console.log(` 2. Enter code: ${r}`),console.log(""))},onBrowserOpening:()=>{e.json||console.log("Opening browser...")},onWaitingForAuthorization:()=>{e.json||console.log("Waiting for authorization...")},onAuthorized:()=>{e.json||console.log("Authorization successful!")}}),o=new V,{workspaces:s}=await o.getWorkspaces(t.accessToken),n=s[0];await Se.saveAll({accessToken:t.accessToken,refreshToken:t.refreshToken,email:t.email,workspaceId:n?.id}),e.json?console.log(JSON.stringify({email:t.email,workspaceId:n?.id,workspaceName:n?.name})):(console.log(`Logged in as ${t.email}`),n&&console.log(`Workspace: ${n.name}`))}async function Ts(e){try{let t=await Se.getAll();if(!t){console.log("Not logged in");return}try{t.refreshToken&&await new V().revokeOAuthToken({token:t.refreshToken,clientId:nn})}catch{}await Se.clear(),console.log("Logged out successfully")}catch(t){ke(t)}}import ti from"http";import{stderr as oi}from"process";import{Client as ni}from"@modelcontextprotocol/sdk/client/index.js";import{StreamableHTTPClientTransport as si}from"@modelcontextprotocol/sdk/client/streamableHttp.js";import{Server as ri}from"@modelcontextprotocol/sdk/server/index.js";import{StdioServerTransport as ii}from"@modelcontextprotocol/sdk/server/stdio.js";import{StreamableHTTPServerTransport as ai}from"@modelcontextprotocol/sdk/server/streamableHttp.js";import{CallToolRequestSchema as li,ListToolsRequestSchema as ci}from"@modelcontextprotocol/sdk/types.js";function Be(e){oi.write(`${e}
|
|
406
406
|
`)}async function Is(e,t,o){let s=new ni({name:"entrydesk-cli-proxy",version:"1.0.0"}),n=new URL(`${e}/v1/workspaces/${o}/mcp`),r=new si(n,{requestInit:{headers:{Authorization:`Bearer ${t}`}}});return await s.connect(r),s}function vs(e){let t=new ri({name:"entrydesk-cli",version:"1.0.0"},{capabilities:{tools:{}}});return t.setRequestHandler(ci,async()=>({tools:(await(await e()).listTools()).tools})),t.setRequestHandler(li,async o=>{let{name:s,arguments:n}=o.params,i=await(await e()).callTool({name:s,arguments:n}),a=Array.isArray(i.content)?[...i.content]:[];return i.structuredContent&&a.push({type:"text",text:JSON.stringify(i.structuredContent,null,2)}),{content:a,structuredContent:i.structuredContent,isError:i.isError}}),t}async function ui(e,t,o){let s=ti.createServer(async(n,r)=>{if(n.method==="OPTIONS"){r.setHeader("Access-Control-Allow-Origin","*"),r.setHeader("Access-Control-Allow-Methods","GET, POST, DELETE, OPTIONS"),r.setHeader("Access-Control-Allow-Headers","Content-Type, Accept"),r.writeHead(204),r.end();return}if(n.url!=="/mcp"){r.writeHead(404,{"Content-Type":"application/json"}),r.end(JSON.stringify({error:"Not found"}));return}let i=vs(e),a=new ai({sessionIdGenerator:void 0});if(r.setHeader("Access-Control-Allow-Origin","*"),r.on("close",()=>{a.close(),i.close()}),await i.connect(a),n.method==="POST"){let l="";n.on("data",d=>{l+=d.toString()}),n.on("end",async()=>{try{let d=JSON.parse(l);await a.handleRequest(n,r,d)}catch{r.writeHead(400,{"Content-Type":"application/json"}),r.end(JSON.stringify({error:"Invalid JSON"}))}})}else await a.handleRequest(n,r)});s.listen(o,()=>{Be(`EntryDesk MCP server running at http://localhost:${o}/mcp`),Be("Press Ctrl+C to stop")}),process.on("SIGINT",async()=>{Be(`
|
|
407
407
|
Shutting down...`),s.close(),await t(),process.exit(0)}),process.on("SIGTERM",async()=>{s.close(),await t(),process.exit(0)})}async function di(e,t){let o=vs(e),s=new ii;s.onclose=async()=>{await t()},await o.connect(s)}async function As(e={mode:"stdio"}){process.on("unhandledRejection",t=>{console.error("[entrydesk-mcp] Unhandled rejection:",t)}),process.on("uncaughtException",t=>{console.error("[entrydesk-mcp] Uncaught exception:",t)});try{Be("[entrydesk-mcp] Starting MCP server...");let t,o;try{let d=await fe();t=d.accessToken,o=d.workspaceId}catch(d){let k=d instanceof Error?d.message:"Authentication failed";console.error(k),process.exit(1)}let{apiUrl:s,mcp:n}=Ze();Be(`[entrydesk-mcp] Connecting to ${s}...`),Be(`[entrydesk-mcp] Workspace: ${o}`);let r=await Is(s,t,o);Be("[entrydesk-mcp] Connected to EntryDesk API");let i=null,a=async()=>rn(t)?i||(i=(async()=>{Be("[entrydesk-mcp] Token expiring soon, refreshing...");let d=await Se.getAll();if(!d?.accessToken)throw new Error("Not logged in. Please run `entrydesk login` again.");let k=await an(d);if(k!==t){t=k;try{await r.close()}catch{}r=await Is(s,t,o),Be("[entrydesk-mcp] Token refreshed, reconnected")}return r})().finally(()=>{i=null}),i):r,l=async()=>{try{await r.close()}catch{}};if(e.mode==="http"){let d=e.port??n.httpPort;await ui(a,l,d)}else Be("[entrydesk-mcp] Starting stdio server..."),await di(a,l),Be("[entrydesk-mcp] Stdio server started")}catch(t){console.error("[entrydesk-mcp] Fatal error:",t),process.exit(1)}}async function Ps(e){try{let t,o;try{let r=await fe();t=r.accessToken,o=r.workspaceId}catch(r){let i=r instanceof Error?r.message:"Authentication failed";console.error(i),process.exit(1)}let s=new V,{models:n}=await s.getModels(t,o);if(e.json){console.log(JSON.stringify(n));return}if(n.length===0){console.log("No models available in this workspace.");return}console.log(`Available Models (${n.length}):
|
|
408
408
|
`);for(let r of n)console.log(` ${r.name}`),r.description&&console.log(` ${r.description.slice(0,80)}`),console.log(` ID: ${r.id}`),console.log("")}catch(t){ke(t)}}function fi(){console.log(`
|
|
@@ -528,7 +528,7 @@ Examples:
|
|
|
528
528
|
`)}async function Js(e){if(e.help){Gi();return}let t,o;try{let y=await fe();t=y.accessToken,o=y.workspaceId}catch(y){let h=y instanceof Error?y.message:"Authentication failed";console.error(h),process.exit(1)}let s=new V,n=[],r,i=0,a=100;do{let y=await s.getMonthlyUsages(t,o,{nextKey:r,limit:100});if(n.push(...y.monthlyUsages),r=y.nextKey,i++,i>=a){console.error(`Warning: Showing first ${n.length} entries (pagination limit reached)`);break}}while(r);if(e.json){console.log(JSON.stringify(n));return}if(n.length===0){console.log("No usage data found for this workspace.");return}console.log(`Current Month Usage:
|
|
529
529
|
`);let l=0,d=0;for(let y of n)l+=parseFloat(y.totalCost),d+=y.operationCount;console.log(`Total Cost: $${l.toFixed(2)}`),console.log(`Total Operations: ${d.toLocaleString()}`),console.log(`Users: ${n.length}
|
|
530
530
|
`);let k=e.sortBy||"cost",C=[...n].toSorted((y,h)=>k==="cost"?parseFloat(h.totalCost)-parseFloat(y.totalCost):h.operationCount-y.operationCount),$=e.limit&&e.limit>0?e.limit:C.length,S=C.slice(0,$);console.log(`Usage by User (sorted by ${k}, showing ${S.length} of ${n.length}):
|
|
531
|
-
`);for(let y of S){let h=parseFloat(y.totalCost);console.log(` ${y.user.name} (${y.user.email})`),console.log(` Cost: $${h.toFixed(2)}`),console.log(` Operations: ${y.operationCount.toLocaleString()}`),console.log("")}S.length<n.length&&console.log(`... and ${n.length-S.length} more users`)}function en(){console.log("entrydesk 1.11.
|
|
531
|
+
`);for(let y of S){let h=parseFloat(y.totalCost);console.log(` ${y.user.name} (${y.user.email})`),console.log(` Cost: $${h.toFixed(2)}`),console.log(` Operations: ${y.operationCount.toLocaleString()}`),console.log("")}S.length<n.length&&console.log(`... and ${n.length-S.length} more users`)}function en(){console.log("entrydesk 1.11.1")}function tn(){console.log(`
|
|
532
532
|
Usage: entrydesk workspaces [list|switch|use] [options]
|
|
533
533
|
|
|
534
534
|
Commands:
|