@nalvietnam/avatar-cli 2.0.0 → 3.0.0-beta.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/dist/index.js +76 -103
- package/dist/index.js.map +1 -1
- package/dist/templates/CLAUDE.md.tpl +14 -38
- package/package.json +1 -1
- package/src/templates/CLAUDE.md.tpl +14 -38
package/dist/index.js
CHANGED
|
@@ -1,95 +1,83 @@
|
|
|
1
1
|
// @nalvietnam/avatar-cli — built with tsup
|
|
2
|
-
var
|
|
3
|
-
`,n)}var k=Ze(()=>{"use strict"});import{spawnSync as Qs}from"child_process";import{chmodSync as ta,existsSync as ki,readFileSync as wi,renameSync as ea,writeFileSync as na}from"fs";import{join as ia}from"path";function xt(t){return ia(t,oa)}function vi(t){let e=t.indexOf(yi);if(e===-1)return null;let n=t.indexOf(cn,e);return n===-1?null:t.slice(e,n+cn.length)}function bi(t){let e=new Map,n=/^export\s+([A-Z][A-Z0-9_]*)="((?:[^"\\]|\\.)*)"$/gm,i;for(;(i=n.exec(t))!==null;){let o=i[1],s=i[2];if(!o||s===void 0)continue;let a=s.replace(/\\"/g,'"').replace(/\\\\/g,"\\");e.set(o,a)}return e}function sa(t){return t.replace(/\\/g,"\\\\").replace(/"/g,'\\"')}function aa(t){let e=[ra],n=[...t.keys()].sort();for(let i of n){let o=t.get(i);o!==void 0&&e.push(`export ${i}="${sa(o)}"`)}return e.push(cn),e.join(`
|
|
4
|
-
`)}async function st(t,e){let n=xt(t),i="";ki(n)&&(i=wi(n,"utf8"));let o=vi(i),s=o?bi(o):new Map,a=s.size;for(let[f,S]of Object.entries(e))S===null?s.delete(f):s.set(f,S);let c=aa(s),l,u=!1;o?(l=i.replace(o,c),u=i.replace(o,"").trim().length>0):i.trim()?(u=!0,l=`${i.replace(/\s+$/,"")}
|
|
5
|
-
|
|
6
|
-
${c}
|
|
7
|
-
`):l=`${c}
|
|
8
|
-
`;let m=`${n}.avatar-tmp-${process.pid}`;na(m,l,{encoding:"utf8",mode:384}),ea(m,n),ta(n,384);let g=null;try{g=Qs("direnv",["allow",n],{encoding:"utf8",timeout:5e3}).status===0}catch{g=null}return{envrcPath:n,varsBefore:a,varsAfter:s.size,direnvAllowOk:g,hadExistingNonAvatarContent:u}}function V(t){let e=xt(t);if(!ki(e))return new Map;let n=wi(e,"utf8"),i=vi(n);return i?bi(i):new Map}var oa,yi,cn,ra,Ut=Ze(()=>{"use strict";oa=".envrc",yi="# >>> avatar-cli secrets >>>",cn="# <<< avatar-cli secrets <<<",ra=`${yi}
|
|
9
|
-
# Per-project secrets managed by avatar-cli.
|
|
10
|
-
# DO NOT COMMIT \u2014 must be gitignored (avatar enforces this on setup).
|
|
11
|
-
# Edit via: avatar secrets set <NAME>
|
|
12
|
-
# Remove via: avatar secrets rm <NAME>
|
|
13
|
-
# View: avatar secrets list`});var ur={};is(ur,{MIGRATABLE_SECRET_KEYS:()=>lr,detectPlaintextSecretsInSettings:()=>It,migrateSecretsFromSettingsToEnvrc:()=>ze});import{promises as _u}from"fs";import{join as Iu}from"path";function Nu(t){return Iu(t,...Ou)}function ju(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)}${String(e.getMonth()+1).padStart(2,"0")}${String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-secrets-migration-${n}`}async function It(t){let e=Nu(t);if(!await d(e))return{keys:[],settingsPath:e};let n;try{n=await y(e)}catch{return{keys:[],settingsPath:e}}let i=n.env??{},o=[];for(let s of lr)typeof i[s]=="string"&&i[s].length>0&&o.push(s);return{keys:o,settingsPath:e}}async function ze(t){let{keys:e,settingsPath:n}=await It(t);if(e.length===0)return{migratedKeys:[],settingsJsonBackupPath:null,noOp:!0};let i=await y(n),o=i.env??{},s={};for(let c of e){let l=o[c];typeof l=="string"&&l.length>0&&(s[c]=l)}await st(t,s);let a=ju(n);await _u.copyFile(n,a);for(let c of e)delete o[c];return Object.keys(o).length===0?i.env=void 0:i.env=o,await $(n,i),{migratedKeys:e,settingsJsonBackupPath:a,noOp:!1}}var lr,Ou,Kn=Ze(()=>{"use strict";k();Ut();lr=["ANTHROPIC_API_KEY","ANTHROPIC_AUTH_TOKEN","OPENAI_API_KEY","GEMINI_API_KEY","DEEPSEEK_API_KEY","GROQ_API_KEY","PERPLEXITY_API_KEY","ANTHROPIC_VERTEX_PROJECT_ID","GOOGLE_APPLICATION_CREDENTIALS"],Ou=[".claude","settings.json"]});import{Command as zp}from"commander";k();import{promises as xa}from"fs";import{join as Ti}from"path";import{confirm as Sa}from"@inquirer/prompts";import{existsSync as re,readFileSync as ss}from"fs";import{dirname as as,join as se}from"path";var cs=5;function ls(t){let e=re(se(t,".claude")),n=re(se(t,"CLAUDE.md"));if(!e||!n)return!1;let i=re(se(t,"src")),o=se(t,".gitmodules");if(i)return!0;if(re(o))try{let s=ss(o,"utf8");return s.includes("path = src")||s.includes("path = ./src")}catch{return!1}return!1}function vt(t){let e=t;for(let n=0;n<cs;n++){if(ls(e))return e;let i=as(e);if(i===e)return null;e=i}return null}import{promises as ni}from"fs";import{homedir as ps}from"os";import{join as Mt}from"path";import{z as h}from"zod";var Qn=h.object({email:h.string().email(),name:h.string(),access_token:h.string().min(1),refresh_token:h.string().min(1),expires_at:h.string().datetime(),id_token:h.string().min(1)}),us=h.object({installed_tools:h.record(h.string(),h.object({version:h.string().optional(),installed_at:h.string().datetime(),install_method:h.string()})).default({}),tool_inputs:h.record(h.string(),h.unknown()).default({})}),rm=h.object({$schema:h.string().optional(),includeCoAuthoredBy:h.boolean().optional(),env:h.record(h.string(),h.string()).default({}),permissions:h.object({allow:h.array(h.string()).default([]),deny:h.array(h.string()).default([])}).partial().optional(),hooks:h.record(h.string(),h.array(h.unknown())).optional(),statusLine:h.object({type:h.string(),command:h.string(),padding:h.number().optional()}).optional()}),sm=h.enum(["internal","client","library"]);k();var Lt=Mt(ps(),".avatar"),Q=Mt(Lt,"config.json"),mm=Mt(Lt,"state.json"),Qe=Mt(Lt,"audit.log"),dm=Mt(Lt,"backups"),ms=384;async function tn(){await T(Lt)}async function tt(){if(!await d(Q))return null;let t=await y(Q),e=Qn.safeParse(t);return e.success?e.data:null}async function ti(t){await tn(),await $(Q,t,ms)}async function ei(){if(await d(Q)){let{promises:t}=await import("fs");await t.unlink(Q)}}function et(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}async function ds(){try{await ni.chmod(Qe,384)}catch{}}async function v(t,e){await tn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},i=`${JSON.stringify(n)}
|
|
14
|
-
`;await ni.appendFile(Qe,i,{encoding:"utf8",mode:384}),await ds()}import{spawnSync as en}from"child_process";import p from"chalk";import gs from"ora";var r={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
|
|
2
|
+
var Ui=Object.defineProperty;var Ut=(t,e)=>()=>(t&&(e=t(t=0)),e);var Mn=(t,e)=>{for(var n in e)Ui(t,n,{get:e[n],enumerable:!0})};import{spawnSync as fs}from"child_process";function Wt(){let t=fs("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(t.status!==0)throw new Error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c GitHub username: ${t.stderr?.trim()}`);return t.stdout.trim()}var Ge=Ut(()=>{"use strict"});import p from"chalk";import hs from"ora";function mt(t){return hs({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Pt(t){let e=Date.now(),n=mt(`${t} (0:00)`),r=()=>{let s=Math.floor((Date.now()-e)/1e3),a=Math.floor(s/60),c=s%60;return`${a}:${String(c).padStart(2,"0")}`},o=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:s=>{clearInterval(o),n.succeed(`${s} (${r()})`)},fail:s=>{clearInterval(o),n.fail(`${s} (${r()})`)},stop:()=>{clearInterval(o),n.stop()}}}var i,f=Ut(()=>{"use strict";i={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
|
|
15
3
|
`),success:t=>process.stdout.write(`${p.green("\u2713")} ${t}
|
|
16
4
|
`),warn:t=>process.stdout.write(`${p.yellow("\u26A0")} ${t}
|
|
17
5
|
`),error:t=>process.stderr.write(`${p.red("\u2717")} ${t}
|
|
18
6
|
`),dim:t=>process.stdout.write(`${p.dim(t)}
|
|
19
7
|
`),plain:t=>process.stdout.write(`${t}
|
|
20
|
-
`)};
|
|
21
|
-
|
|
22
|
-
|
|
8
|
+
`)}});var Ar={};Mn(Ar,{createEmptyGithubRepoAndGetUrl:()=>Bs});import{spawnSync as Fs}from"child_process";async function Bs(t,e,n){let o=`${n??Wt()}/${t}`;i.info(`T\u1EA1o GitHub repo r\u1ED7ng ${o} (${e})...`);let s=Fs("gh",["repo","create",o,`--${e}`,"--add-readme"],{stdio:"inherit"});if(s.status!==0)throw new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${s.status}). Repo c\xF3 th\u1EC3 \u0111\xE3 t\u1ED3n t\u1EA1i.`);return i.success(`\u0110\xE3 t\u1EA1o: git@github.com:${o}.git`),`git@github.com:${o}.git`}var Sr=Ut(()=>{"use strict";Ge();f()});var Ze={};Mn(Ze,{HOSTED_DOMAIN:()=>Yt,SCOPES:()=>$r,buildUserConfig:()=>qe,buildVerificationUrl:()=>Xe,decodeIdToken:()=>ze,pollForToken:()=>We,refreshAccessToken:()=>ra,requestDeviceCode:()=>Be,revokeToken:()=>Ye,verifyHostedDomain:()=>Je,verifyIdTokenClaims:()=>_r});async function Be(){let t=new URLSearchParams({client_id:Xt,scope:$r.join(" ")}),e=await fetch(Qs,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:t});if(!e.ok){let n=await e.text();throw new Error(`Device code request failed (${e.status}): ${n}`)}return await e.json()}async function We(t){let e=new URLSearchParams({client_id:Xt,client_secret:Tr,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Rr,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(n.ok)return await n.json();let r="";try{r=(await n.json()).error??""}catch{r=""}if(r==="authorization_pending"||r==="slow_down")return null;throw r==="access_denied"?new Error("User t\u1EEB ch\u1ED1i quy\u1EC1n truy c\u1EADp"):r==="expired_token"?new Error("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."):new Error(`OAuth token endpoint tr\u1EA3 l\u1ED7i: ${r||n.status}`)}function ze(t){let e=t.split(".");if(e.length!==3)throw new Error("id_token format kh\xF4ng h\u1EE3p l\u1EC7");let n=e[1];if(!n)throw new Error("id_token thi\u1EBFu payload");let r=n.replace(/-/g,"+").replace(/_/g,"/"),o=Buffer.from(r,"base64").toString("utf8");return JSON.parse(o)}function _r(t){if(!ea.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==Xt)throw new Error("id_token audience kh\xF4ng kh\u1EDBp client ID Avatar. Token c\xF3 th\u1EC3 \u0111\u01B0\u1EE3c sign cho app kh\xE1c.");let e=Math.floor(Date.now()/1e3);if(t.exp+na<e){let n=e-t.exp;throw new Error(`id_token \u0111\xE3 h\u1EBFt h\u1EA1n ${n}s tr\u01B0\u1EDBc. Login l\u1EA1i \u0111\u1EC3 l\u1EA5y token m\u1EDBi.`)}if(t.hd!==Yt)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${Yt}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}function qe(t,e){let n=new Date(Date.now()+t.expires_in*1e3).toISOString();return{email:e.email,name:e.name??e.email,access_token:t.access_token,refresh_token:t.refresh_token,expires_at:n,id_token:t.id_token}}async function ra(t){let e=new URLSearchParams({client_id:Xt,client_secret:Tr,refresh_token:t,grant_type:"refresh_token"}),n=await fetch(Rr,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(!n.ok){let r=await n.text();throw new Error(`Refresh token failed (${n.status}): ${r}`)}return await n.json()}async function Ye(t){let e=new URLSearchParams({token:t});await fetch(ta,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function Xe(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",Yt),e.toString()}var Xt,Tr,Yt,$r,Qs,Rr,ta,ea,na,Je,Zt=Ut(()=>{"use strict";Xt="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Tr="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",Yt="nal.vn",$r=["openid","email","profile"],Qs="https://oauth2.googleapis.com/device/code",Rr="https://oauth2.googleapis.com/token",ta="https://oauth2.googleapis.com/revoke";ea=new Set(["https://accounts.google.com","accounts.google.com"]),na=60;Je=_r});import{Command as Qu}from"commander";import{resolve as Ws}from"path";import{confirm as Fe,input as qt,select as Pr}from"@inquirer/prompts";import{join as je}from"path";import{spawn as Di,spawnSync as Hi}from"child_process";import{createHash as op}from"crypto";import{constants as Ki,promises as G,createReadStream as sp}from"fs";import{dirname as jn,join as cp,relative as lp}from"path";import{Readable as Vi}from"stream";async function d(t){try{return await G.access(t,Ki.F_OK),!0}catch{return!1}}async function A(t){await G.mkdir(t,{recursive:!0})}async function N(t){return await G.readFile(t,"utf8")}async function k(t){return JSON.parse(await N(t))}async function dt(t,e,n){await A(jn(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`;await G.writeFile(r,e,"utf8"),n!==void 0&&await G.chmod(r,n),await G.rename(r,t)}async function b(t,e,n){await dt(t,`${JSON.stringify(e,null,2)}
|
|
9
|
+
`,n)}async function Ht(t){await G.rm(t,{recursive:!0,force:!0})}var Fi=3e5;async function Ln(t,e,n=Fi){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await A(jn(e));let{createWriteStream:s}=await import("fs");try{let a=await fetch(t,{signal:r.signal});if(!a.ok)throw new Error(`Download fail (${a.status}) t\u1EEB ${t.slice(0,80)}...`);if(!a.body)throw new Error("Response kh\xF4ng c\xF3 body \u0111\u1EC3 t\u1EA3i.");let c=s(e);await new Promise((l,u)=>{let m=Vi.fromWeb(a.body);m.on("error",u),c.on("error",u),c.on("finish",()=>l()),m.pipe(c)})}catch(a){throw await G.rm(e,{force:!0}).catch(()=>{}),a instanceof Error&&a.name==="AbortError"?new Error(`T\u1EA3i file qu\xE1 ${n/1e3}s (signed URL c\xF3 th\u1EC3 \u0111\xE3 h\u1EBFt h\u1EA1n).`):a}finally{clearTimeout(o)}}var Dt=null;function Bi(){return Dt!==null||(Dt=Hi("tar",["--version"],{stdio:"ignore"}).status===0),Dt}async function Gn(t,e){if(!Bi())throw new Error("Kh\xF4ng t\xECm th\u1EA5y l\u1EC7nh `tar` tr\xEAn h\u1EC7 th\u1ED1ng. C\u1EA7n \u0111\u1EC3 gi\u1EA3i n\xE9n pack.\n Windows: c\xE0i qua WSL2, Git Bash, ho\u1EB7c MSYS2.\n macOS/Linux: tar th\u01B0\u1EDDng c\xF3 s\u1EB5n \u2014 ki\u1EC3m tra PATH.");await A(e),await new Promise((n,r)=>{let o=Di("tar",["-xzf",t,"-C",e],{stdio:["ignore","ignore","pipe"]}),s="";o.stderr.on("data",a=>{s+=a.toString()}),o.on("error",r),o.on("close",a=>{a===0?n():r(new Error(`tar extract exit ${a}: ${s.trim()}`))})})}import{join as mp}from"path";import{simpleGit as Wi}from"simple-git";function Kt(t=process.cwd()){return Wi({baseDir:t,binary:"git"})}import{promises as Vn}from"fs";import{join as U}from"path";import{existsSync as Ji}from"fs";import{dirname as Dn,join as St}from"path";import{fileURLToPath as qi}from"url";var zi=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g;function Un(t,e){return t.replace(zi,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var Yi=Dn(qi(import.meta.url)),Hn=Qi(Yi),Xi=St(Hn,"src","templates"),Zi=St(Hn,"src","hooks");function Qi(t){let e=t;for(;;){if(Ji(St(e,"package.json")))return e;let n=Dn(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function ts(t){return await N(St(Xi,`${t}.tpl`))}async function Vt(t,e){let n=await ts(t);return Un(n,e)}async function Kn(t){return await N(St(Zi,`${t}.sh.tpl`))}async function es(t){if(!await d(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,r=n,o=1;for(;await d(r);)if(r=`${n}-${o}`,o++,o>5)throw new Error(`Could not find free backup name for ${t}`);return await Vn.rename(t,r),r}async function Fn(t,e,n){let r=await es(t);return await dt(t,e,n),r}var ns=["state","_pending","_backup"];async function Bn(t){let e=U(t,".claude");await A(e);for(let n of ns){let r=U(e,n);await A(r),await dt(U(r,".gitkeep"),"")}}async function Wn(t,e){return[]}async function zn(t,e){let n=await Vt("CLAUDE.md",e);return await Fn(U(t,"CLAUDE.md"),n)}async function Jn(t,e){let n=await Vt("settings.json",e);return await Fn(U(t,".claude","settings.json"),n)}async function qn(t){let e=U(t,".gitignore"),n=await Vt("gitignore",{}),r="# Avatar \u2014 git-ignored entries injected on `avatar init`",o="";if(await d(e)&&(o=await Vn.readFile(e,"utf8"),o.includes(r)))return;let s=o.endsWith(`
|
|
10
|
+
`)||o.length===0?"":`
|
|
11
|
+
`;await dt(e,`${o}${s}
|
|
12
|
+
${n}`)}async function Yn(t,e){let n=await Kn(e),r=U(t,"hooks");await A(r);let o=U(r,e);await dt(o,n,493)}import{join as rs}from"path";var os=".claude/repos.json";function Xn(t){return rs(t,os)}async function Zn(t){let e=Xn(t);if(!await d(e))return[];try{let n=await k(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function Qn(t,e){let r=(await Zn(t)).filter(o=>o.name!==e.name);r.push(e),await b(Xn(t),{repos:r})}async function tr(t,e){return(await Zn(t)).some(r=>r.name===e)}function is(t){return!t||t.trim().length===0?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng.":/[/\\]/.test(t)||t.includes("..")?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..'.":t==="pack"?"T\xEAn 'pack' b\u1ECB gi\u1EEF cho team-ai-pack.":null}function er(t){return(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"")||"repo"}async function nr(t){let e=is(t.name);if(e)throw new Error(e);if(await tr(t.workspaceRoot,t.name))throw new Error(`Repo "${t.name}" \u0111\xE3 c\xF3 trong workspace. Ch\u1ECDn t\xEAn kh\xE1c ho\u1EB7c g\u1EE1 tr\u01B0\u1EDBc.`);let n=je(t.workspaceRoot,"src");await A(n);let r=je(n,t.name);if(await d(r))throw new Error(`Th\u01B0 m\u1EE5c src/${t.name} \u0111\xE3 t\u1ED3n t\u1EA1i. Ch\u1ECDn t\xEAn kh\xE1c.`);await Kt(n).clone(t.url,t.name);let o=je(r,".git");return await Yn(o,"pre-push").catch(()=>{}),await Qn(t.workspaceRoot,{name:t.name,path:`src/${t.name}`,url:t.url,addedAt:new Date().toISOString()}),{name:t.name,path:r,url:t.url}}async function rr(t){try{let e=await Kt(t).status();return e.isClean()?"":e.files.map(n=>`${n.path}`).join(", ")}catch{return""}}async function or(t){try{let n=(await Kt(t).getRemotes(!0)).find(r=>r.name==="origin");return n?.refs.fetch??n?.refs.push??null}catch{return null}}import{spawnSync as ir}from"child_process";import{platform as ss}from"os";function X(){let t=ss();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var as=5e3,cs=/(\d+\.\d+\.\d+)/;function ls(){let e=X()==="win32"?"where":"which",n=ir(e,["gitnexus"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let r=(n.stdout||"").trim();return r?r.split(/\r?\n/)[0].trim():null}function us(){let t=ir("gitnexus",["--version"],{encoding:"utf8",timeout:as});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return cs.exec(e)?.[1]??null}var Z=null;function Q(){if(Z!==null)return Z;let t=ls();return t?(Z={installed:!0,version:us(),path:t},Z):(Z={installed:!1,version:null,path:null},Z)}function Ft(){Z=null}import{spawnSync as ps}from"child_process";function Bt(){let t=ps("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as ds}from"child_process";function ms(t){let e=X();return ds(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function sr(){let t=X(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(ms(n))return n;return null}import{spawnSync as mr}from"child_process";import{input as Es,select as Ts}from"@inquirer/prompts";import{spawnSync as De}from"child_process";import{promises as ks}from"fs";import{basename as ys,join as pr}from"path";import{confirm as vs,select as bs}from"@inquirer/prompts";import{spawnSync as gs}from"child_process";var Le=class extends Error{constructor(e){super(`Repo "${e}" \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. \u0110\u1ED5i t\xEAn ho\u1EB7c x\xF3a repo c\u0169.`),this.name="RepoAlreadyExistsError"}};function ar(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=gs("gh",n,{stdio:"inherit"});if(r.status!==0)throw r.status===1?new Le(e):new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${r.status})`);return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}Ge();f();var ws=/^[a-zA-Z0-9._-]{1,100}$/,Ue=class extends Error{constructor(e){super(`T\xEAn repo "${e}" kh\xF4ng h\u1EE3p l\u1EC7. Ch\u1EC9 d\xF9ng ch\u1EEF/s\u1ED1/d\u1EA5u ch\u1EA5m/g\u1EA1ch/underscore, d\xE0i 1-100 k\xFD t\u1EF1.`),this.name="InvalidRepoNameError"}};function cr(t){if(!ws.test(t))throw new Ue(t)}function lr(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function ur(t){cr(t.name),lr(t.visibility);let e=t.org??Wt();i.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=ar({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return i.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}f();function xs(){let t=new Date;return`${t.getFullYear().toString().slice(-2)}${String(t.getMonth()+1).padStart(2,"0")}${String(t.getDate()).padStart(2,"0")}-${String(t.getHours()).padStart(2,"0")}${String(t.getMinutes()).padStart(2,"0")}`}async function As(t){let e=pr(t,".git");if(!await d(e))throw new Error(`.git kh\xF4ng t\u1ED3n t\u1EA1i \u1EDF ${t} \u2014 kh\xF4ng c\u1EA7n reset.`);let n=`.git.backup-${xs()}`,r=pr(t,n);return await ks.rename(e,r),i.success(`Backup .git \u2192 ${n}`),r}function Ss(t){let e=De("git",["-C",t,"init","-b","main"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(e.status!==0)throw new Error(`git init th\u1EA5t b\u1EA1i: ${e.stderr||e.stdout}`);i.success("Git init m\u1EDBi (branch main)"),De("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=De("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0?i.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(n.stderr||"").slice(0,200)}`):i.success("Initial commit")}async function dr(t){let e=ys(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await vs({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
|
|
13
|
+
1. Backup .git \u2192 .git.backup-{timestamp}
|
|
14
|
+
2. Git init m\u1EDBi (branch main, initial commit)
|
|
15
|
+
3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
|
|
16
|
+
Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await bs({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await As(t.folderPath);return Ss(t.folderPath),{newRemoteUrl:ur({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:o}}f();import{spawnSync as Ps}from"child_process";var He=5e3;function Cs(t){let e=t.toLowerCase();return e.includes("authentication")||e.includes("could not read username")||e.includes("permission denied")||e.includes("403")||e.includes("access denied")||e.includes("repository not found")?"no-access":e.includes("404")||e.includes("does not exist")?"not-found":e.includes("could not resolve host")||e.includes("network")||e.includes("connection refused")||e.includes("connection timed out")?"network":"unknown"}function zt(t){let e=Ps("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:He,stdio:["ignore","pipe","pipe"]});if(e.error){let o=e.error;return o.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:o.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${He/1e3}s`}:{ok:!1,reason:"unknown",detail:o.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${He/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:Cs(n),detail:n.slice(0,300)}}var Ct=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function $s(){let t=mr("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Rs(){i.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=mr("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&i.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function _s(t,e,n){switch(t){case"no-access":return n?`Repo c\xF3 th\u1EC3 private v\xE0 account '${n}' kh\xF4ng c\xF3 quy\u1EC1n access. Switch sang account c\xF3 quy\u1EC1n, ho\u1EB7c xin invite t\u1EEB owner repo.`:"Repo c\xF3 th\u1EC3 private v\xE0 gh CLI ch\u01B0a login. Login tr\u01B0\u1EDBc r\u1ED3i retry.";case"not-found":return`URL c\xF3 th\u1EC3 sai ch\xEDnh t\u1EA3, ho\u1EB7c repo \u0111\xE3 b\u1ECB x\xF3a/rename. Nh\u1EADp l\u1EA1i URL \u0111\xFAng. Check: ${e}`;case"network":return"Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c GitHub. Check m\u1EA1ng / VPN / firewall.";case"timeout":return"Network ch\u1EADm > 5s. Check m\u1EA1ng r\u1ED3i retry.";default:return"L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh. URL c\xF3 th\u1EC3 sai \u2014 nh\u1EADp l\u1EA1i, ho\u1EB7c check gh auth status."}}function Is(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function gr(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let o=$s();i.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),i.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),i.info(_s(n,e,o)),o&&i.dim(` gh CLI hi\u1EC7n \u0111ang login: ${o}`);let s=[{name:"Nh\u1EADp l\u1EA1i URL \u0111\xFAng (recommended khi URL sai ch\xEDnh t\u1EA3)",value:"re-input-url"},{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"T\xF4i v\u1EEBa fix (accept invite / s\u1EEDa permission) \u2014 retry verify",value:"retry"}];t.folderPath&&s.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),s.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let a=await Ts({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new Ct(`User ch\u1ECDn t\u1EA1m ng\u01B0ng. Fix access ${e} r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.`);if(a==="reset-folder"){if(!t.folderPath){i.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await dr({folderPath:t.folderPath,visibility:t.defaultVisibility});return i.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),i.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){i.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await Es({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>Is(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&Rs(),i.info(`Verify remote l\u1EA1i: ${e}...`);let c=zt(e);if(c.ok)return i.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}f();import{spawnSync as Os}from"child_process";var Ns={brew:{cmd:"brew",args:["install","gh"]},apt:{cmd:"sudo",args:["apt-get","install","-y","gh"]},dnf:{cmd:"sudo",args:["dnf","install","-y","gh"]},pacman:{cmd:"sudo",args:["pacman","-S","--noconfirm","github-cli"]},winget:{cmd:"winget",args:["install","--id","GitHub.cli","-e","--silent"]}};function fr(t){let e=Ns[t];i.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=Os(e.cmd,e.args,{stdio:"inherit"});if(n.status!==0)throw new Error(`C\xE0i gh CLI th\u1EA5t b\u1EA1i qua ${t} (exit ${n.status}). C\xE0i tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);i.success("\u0110\xE3 c\xE0i gh CLI")}f();import{input as Ed,select as Ms}from"@inquirer/prompts";var S=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function _(t){i.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&i.info(t.hint);let e=[{name:"Th\u1EED l\u1EA1i (Retry)",value:"retry"}];return t.allowSkip&&e.push({name:"B\u1ECF qua b\u01B0\u1EDBc n\xE0y v\xE0 ti\u1EBFp t\u1EE5c (Skip)",value:"skip"}),e.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i sau (Abort)",value:"abort"}),await Ms({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}f();import{spawnSync as js}from"child_process";function hr(){if(js("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){i.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}i.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}f();f();import{spawnSync as Ls}from"child_process";function wr(){i.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=Ls("gh",["auth","login","--hostname","github.com","--web","--git-protocol","ssh"],{stdio:"inherit"});if(t.status!==0)throw new Error(`gh auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'gh auth login' tay.`);i.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}async function kr(t){for(;Bt()==="not-installed";){i.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=sr();if(!e){if(await _({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{fr(e)}catch(n){if(await _({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;Bt()==="not-authenticated";){i.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{wr()}catch(e){if(await _({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(Bt()!=="authenticated"&&await _({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(i.success("gh CLI s\u1EB5n s\xE0ng"),hr(),t){let e=zt(t);return e.ok?(i.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await gr({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{existsSync as Ke}from"fs";import{dirname as Gs,join as Ve}from"path";var Us=5;function Ds(t){let e=Ke(Ve(t,".claude")),n=Ke(Ve(t,"CLAUDE.md")),r=Ke(Ve(t,"src"));return e&&n&&r}function D(t){let e=t;for(let n=0;n<Us;n++){if(Ds(e))return e;let r=Gs(e);if(r===e)return null;e=r}return null}f();import{spawnSync as vr}from"child_process";import{existsSync as Hs}from"fs";import{join as yr}from"path";var Ks=120*1e3,Vs=300*1e3,H=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,o=null,s){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=o,this.stderr=s}};function br(t,e,n,r){if(n==="SIGTERM")return new H(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,r);let o=r.toLowerCase();return o.includes("eacces")||o.includes("permission denied")?new H(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new H(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function Jt(t,e){return t.split(`
|
|
17
|
+
`).slice(-e).join(`
|
|
18
|
+
`)}function xr(){let t=Pt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=vr("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:Ks,encoding:"utf8"});if(e.status!==0||e.signal==="SIGTERM"){t.fail("GitNexus setup failed");let n=(e.stderr||"").trim(),r=(e.stdout||"").trim();throw n?process.stderr.write(`${Jt(n,30)}
|
|
19
|
+
`):r&&process.stderr.write(`${Jt(r,30)}
|
|
20
|
+
`),br("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function gt(t){let e=Pt(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=vr("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:Vs,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let o=(n.stderr||"").trim(),s=(n.stdout||"").trim();throw o?process.stderr.write(`${Jt(o,30)}
|
|
21
|
+
`):s&&process.stderr.write(`${Jt(s,30)}
|
|
22
|
+
`),br("analyze",n.status,n.signal,o)}let r=yr(t,".gitnexus","meta.json");if(!Hs(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new H("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${r}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);e.succeed(`Analyze OK (index t\u1EA1i ${yr(t,".gitnexus")})`)}f();function Cr(t){t.command("add").description("Th\xEAm t\xE0i nguy\xEAn v\xE0o workspace").command("repo").description("Clone 1 repo code v\xE0o src/ (repo c\xF3 s\u1EB5n / th\u01B0 m\u1EE5c / d\u1EF1 \xE1n m\u1EDBi)").option("--url <url>","URL git repo c\xF3 s\u1EB5n (b\u1ECF qua prompt ngu\u1ED3n)").option("--name <name>","T\xEAn th\u01B0 m\u1EE5c trong src/ (m\u1EB7c \u0111\u1ECBnh suy t\u1EEB URL)").option("--yes","Auto-confirm prompt").action(async n=>{try{await Js(n)}catch(r){i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}function zs(){let t=D(process.cwd());return t||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
23
|
+
C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function Js(t){let e=zs(),n=!0,r=t.url,o=t.name;for(;n;){let s=await qs(r),a=o??await Ys(s);i.info(`Clone ${s} \u2192 src/${a} ...`);let c=await nr({workspaceRoot:e,url:s,name:a});if(i.success(`\u2713 \u0110\xE3 clone src/${c.name}`),await Xs(c.path,t.yes),i.success(`\u2713 Done repo: ${c.name}`),r=void 0,o=void 0,t.yes)break;n=await Fe({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function qs(t){if(t)return t;let e=await Pr({message:"Ngu\u1ED3n repo?",choices:[{name:"1. Repo c\xF3 s\u1EB5n (\u0111i\u1EC1n URL git)",value:"url"},{name:"2. Th\u01B0 m\u1EE5c \u0111\xE3 c\xF3 tr\xEAn m\xE1y (clone t\u1EEB remote c\u1EE7a n\xF3)",value:"folder"},{name:"3. D\u1EF1 \xE1n m\u1EDBi ho\xE0n to\xE0n (t\u1EA1o repo GitHub)",value:"new"}]});if(e==="url")return await qt({message:"URL git repo:",validate:s=>s.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"});if(e==="folder"){let s=Ws(await qt({message:"\u0110\u01B0\u1EDDng d\u1EABn folder:",validate:l=>l.trim().length>0?!0:"Path b\u1EAFt bu\u1ED9c"})),a=await rr(s);if(a&&(i.warn(`\u26A0 Folder c\xF3 thay \u0111\u1ED5i ch\u01B0a commit: ${a.slice(0,120)}`),i.warn(" Clone l\u1EA5y b\u1EA3n REMOTE \u2192 c\xE1c thay \u0111\u1ED5i local n\xE0y s\u1EBD KH\xD4NG c\xF3 trong src/<name>."),!await Fe({message:"Ti\u1EBFp t\u1EE5c clone t\u1EEB remote (b\u1ECF thay \u0111\u1ED5i local ch\u01B0a push)?",default:!1})))throw new Error("H\u1EE7y. Commit + push thay \u0111\u1ED5i trong folder g\u1ED1c r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar add repo'.");let c=await or(s);if(!c)throw new Error(`Folder ${s} kh\xF4ng c\xF3 remote origin \u2192 kh\xF4ng clone \u0111\u01B0\u1EE3c. T\u1EA1o remote (gh repo create) r\u1ED3i th\u1EED l\u1EA1i.`);return c}await kr();let n=await qt({message:"T\xEAn repo m\u1EDBi:",validate:s=>s.trim().length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),r=await Pr({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createEmptyGithubRepoAndGetUrl:o}=await Promise.resolve().then(()=>(Sr(),Ar));return await o(n.trim(),r)}async function Ys(t){let e=er(t);return await qt({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function Xs(t,e){if(!Q().installed){i.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}if(e||await Fe({message:"Index repo n\xE0y b\u1EB1ng GitNexus?",default:!0}))try{gt(t),i.success(` \u2713 GitNexus indexed src/${t.split("/").pop()}`)}catch(o){i.warn(` ! GitNexus index fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${o instanceof Error?o.message:o}`)}}import{promises as gc}from"fs";import{join as io}from"path";import{confirm as fc}from"@inquirer/prompts";import{promises as Or}from"fs";import{homedir as oa}from"os";import{join as Tt}from"path";import{z as w}from"zod";var Er=w.object({email:w.string().email(),name:w.string(),access_token:w.string().min(1),refresh_token:w.string().min(1),expires_at:w.string().datetime(),id_token:w.string().min(1)}),Zs=w.object({installed_tools:w.record(w.string(),w.object({version:w.string().optional(),installed_at:w.string().datetime(),install_method:w.string()})).default({}),tool_inputs:w.record(w.string(),w.unknown()).default({})}),gm=w.object({$schema:w.string().optional(),includeCoAuthoredBy:w.boolean().optional(),env:w.record(w.string(),w.string()).default({}),permissions:w.object({allow:w.array(w.string()).default([]),deny:w.array(w.string()).default([])}).partial().optional(),hooks:w.record(w.string(),w.array(w.unknown())).optional(),statusLine:w.object({type:w.string(),command:w.string(),padding:w.number().optional()}).optional()}),fm=w.enum(["internal","client","library"]);var $t=Tt(oa(),".avatar"),tt=Tt($t,"config.json"),bm=Tt($t,"state.json"),Qe=Tt($t,"audit.log"),xm=Tt($t,"backups"),ia=384;async function tn(){await A($t)}async function K(){if(!await d(tt))return null;let t=await k(tt),e=Er.safeParse(t);return e.success?e.data:null}async function en(t){await tn(),await b(tt,t,ia)}async function Ir(){if(await d(tt)){let{promises:t}=await import("fs");await t.unlink(tt)}}function et(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var Et=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function sa(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Zt(),Ze));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function Qt(){let t=await K();if(!t)throw new Et("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Zt(),Ze));if(!await sa(t.id_token))return r(n(t.id_token)),t.id_token;let o;try{o=await e(t.refresh_token)}catch(a){throw new Et(`Token h\u1EBFt h\u1EA1n v\xE0 refresh th\u1EA5t b\u1EA1i (${a instanceof Error?a.message:a}). Ch\u1EA1y 'avatar login' l\u1EA1i.`)}if(!o.id_token)throw new Et("Refresh kh\xF4ng tr\u1EA3 id_token m\u1EDBi (id_token c\u0169 \u0111\xE3 h\u1EBFt h\u1EA1n). Ch\u1EA1y 'avatar login' l\u1EA1i.");r(n(o.id_token));let s={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await en(s),o.id_token}async function aa(){try{await Or.chmod(Qe,384)}catch{}}async function v(t,e){await tn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
|
|
24
|
+
`;await Or.appendFile(Qe,r,{encoding:"utf8",mode:384}),await aa()}f();import{spawnSync as nn}from"child_process";var Nr=6e4,ca="ok";function rn(){let t=nn("claude",["auth","status"],{encoding:"utf8"});if(t.error||t.status!==0)return{state:"not-authenticated"};let e=(t.stdout||"").trim();if(!e.startsWith("{"))return{state:"authenticated"};try{let n=JSON.parse(e);return n.loggedIn!==!0?{state:"not-authenticated"}:{state:"authenticated",email:n.email,subscriptionType:n.subscriptionType,apiProvider:n.apiProvider}}catch{return{state:"authenticated"}}}function on(){i.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=nn("claude",["auth","login"],{stdio:"inherit"});if(t.status!==0)throw new Error(`claude auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'claude auth login' tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);i.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function la(t){let e=t.toLowerCase();return e.includes("credit_balance_too_low")||e.includes("credit balance too low")?"credit_balance_too_low":e.includes("insufficient_quota")||e.includes("insufficient quota")||e.includes("quota_exceeded")||e.includes("quota exceeded")||e.includes("usage limit")||e.includes("you've used all")?"insufficient_quota":e.includes("401")||e.includes("invalid authentication")||e.includes("authentication credentials")||e.includes("failed to authenticate")||e.includes("authentication failed")||e.includes("unauthorized")?"auth-expired":e.includes("invalid_api_key")||e.includes("invalid api key")?"invalid_api_key":e.includes("rate_limit")||e.includes("rate limit")||e.includes("429")?"rate_limit":"unknown"}function Mr(t){switch(t){case"auth-expired":return"Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n/b\u1ECB revoke. Ch\u1EA1y: `claude auth logout && claude auth login`.";case"credit_balance_too_low":case"insufficient_quota":return"H\u1EBFt quota subscription. Upgrade plan ho\u1EB7c d\xF9ng LLMLite (avatar ai setup \u2192 ch\u1ECDn LLMLite).";case"invalid_api_key":return"API key invalid. Re-login: `claude auth login`.";case"rate_limit":return"B\u1ECB rate limit t\u1EA1m th\u1EDDi. Ch\u1EDD v\xE0i ph\xFAt r\u1ED3i ch\u1EA1y `avatar ai setup`.";case"timeout":return"Timeout 60s: (1) m\u1EA1ng VN ch\u1EADm \u2014 th\u1EED VPN, (2) Anthropic API spike \u2014 retry v\xE0i ph\xFAt, (3) token v\u1EABn auth nh\u01B0ng quota h\u1EBFt \xE2m th\u1EA7m \u2014 check t\u1EA1i claude.ai/settings/usage.";default:return"L\u1ED7i ch\u01B0a bi\u1EBFt. Xem stderr \u1EDF tr\xEAn + ch\u1EA1y `claude --print ok` tay \u0111\u1EC3 debug."}}function sn(){let t=nn("claude",["--print",ca],{encoding:"utf8",timeout:Nr,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${Nr/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",r=t.stdout||"";if(t.status===0)return{ok:!0};let o=r.trim(),s=n.toLowerCase();if(o.length>20&&!s.includes("error")&&!s.includes("limit")&&!s.includes("quota")&&!s.includes("401"))return i.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${o.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=la(`${n}
|
|
25
|
+
${r}`);return a==="unknown"&&(i.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&i.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&i.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as jr}from"child_process";var ua=5e3,pa=/(\d+\.\d+\.\d+)/;function da(){let e=X()==="win32"?"where":"which",n=jr(e,["claude"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let r=(n.stdout||"").trim();return r?r.split(/\r?\n/)[0].trim():null}function ma(){let t=jr("claude",["--version"],{encoding:"utf8",timeout:ua});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return pa.exec(e)?.[1]??null}var nt=null;function Rt(){if(nt!==null)return nt;let t=da();return t?(nt={installed:!0,version:ma(),path:t},nt):(nt={installed:!1,version:null,path:null},nt)}function te(){nt=null}import{spawnSync as ga}from"child_process";f();var Lr=300*1e3,Gr="@anthropic-ai/claude-code",rt=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function fa(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new rt("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Gr} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new rt("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new rt("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Ur(){i.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=ga("npm",["install","-g",Gr],{stdio:["inherit","inherit","pipe"],timeout:Lr,encoding:"utf8"});if(t.signal==="SIGTERM")throw new rt("timeout",`npm install timeout sau ${Lr/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),fa(t.status,t.stderr||"");te();let e=Rt();if(!e.installed||!e.path)throw new rt("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `claude` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return i.success(`\u0110\xE3 c\xE0i Claude Code${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{readFileSync as ha}from"fs";import{homedir as wa}from"os";import{join as ka}from"path";import{select as Dr}from"@inquirer/prompts";function ya(){return ka(wa(),".claude","settings.json")}function an(){let t=ya(),e;try{e=ha(t,"utf8")}catch{return{exists:!1,hasBaseUrl:!1,hasToken:!1}}let n;try{n=JSON.parse(e)}catch{return{exists:!0,hasBaseUrl:!1,hasToken:!1}}let r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:void 0,s=typeof r.ANTHROPIC_AUTH_TOKEN=="string"&&r.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!o,baseUrl:o,hasToken:s,model:a,rawSettings:n}}async function Hr(t=an()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await Dr({message:`Ph\xE1t hi\u1EC7n AI config global (base URL: ${t.baseUrl}). D\xF9ng cho project n\xE0y?`,choices:[{name:"a. Yes \u2014 copy config global v\xE0o .claude/settings.json (per-project)",value:"use-global"},{name:"b. No \u2014 setup ri\xEAng (ch\u1ECDn provider kh\xE1c)",value:"setup-fresh"}]})==="use-global"?"use-global":await Dr({message:"Ch\u1ECDn provider cho AI tools:",choices:[{name:"1. Claude Code Subscription (d\xF9ng quota c\xE1 nh\xE2n Anthropic, OAuth login)",value:"subscription"},{name:"2. LLM key NAL (ai.nal.vn \u2014 gateway n\u1ED9i b\u1ED9, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}f();import{password as va,select as ba}from"@inquirer/prompts";var ee="https://api.anthropic.com",xa="2023-06-01",Kr=1e4;function ne(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Aa(t){let e=t.trim();return e.length===0?"API key b\u1EAFt bu\u1ED9c":e.startsWith("sk-ant-")?!0:"Anthropic API key th\u01B0\u1EDDng b\u1EAFt \u0111\u1EA7u b\u1EB1ng 'sk-ant-' (l\u1EA5y t\u1EEB console.anthropic.com)."}async function Sa(){return await va({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Aa})}async function re(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Kr);try{let r=await fetch(`${ee}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":xa,Accept:"application/json"},signal:e.signal});if(r.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(r.status===403)throw new Error("API key b\u1ECB reject (HTTP 403). Key c\xF3 th\u1EC3 \u0111\xE3 b\u1ECB revoke ho\u1EB7c thi\u1EBFu permission.");if(r.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let s=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(s.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return s}catch(r){throw r.name==="AbortError"?new Error(`Connect ${ee} timeout sau ${Kr/1e3}s.`):r}finally{clearTimeout(n)}}async function Pa(t){if(t.length===1){let n=t[0];return i.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let o=s=>{let a=s.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return o(n)-o(r)});return await ba({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function Vr(){let t=await Sa();i.info(`Verify key (${ne(t)}) qua ${ee}/v1/models...`);let e=await re(t);i.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Pa(e);return{apiKey:t,baseUrl:ee,model:n}}f();import{input as Ca,password as Ea,select as Ta}from"@inquirer/prompts";var $a="https://ai.nal.vn",Fr=1e4;function V(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Ra(){return await Ea({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function _a(t=$a){return(await Ca({message:"LLMLite base URL:",default:t,validate:n=>/^https?:\/\//.test(n)?!0:"Ph\u1EA3i l\xE0 URL h\u1EE3p l\u1EC7 (http/https)"})).replace(/\/+$/,"")}async function Ia(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),Fr);try{let o=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(o.status===401||o.status===403)throw new Error(`API key invalid (HTTP ${o.status}).`);if(o.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!o.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${o.status}).`);let a=((await o.json()).data||[]).map(c=>typeof c.id=="string"?c.id:null).filter(c=>c!==null);if(a.length===0)throw new Error("LLMLite tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 admin NAL.");return a}catch(o){if(o.name==="AbortError"){let c=t.includes("nal.vn")||t.includes("nal-vn")?"\n Hint: ai.nal.vn l\xE0 internal endpoint NAL \u2014 ki\u1EC3m tra VPN NAL \u0111\xE3 b\u1EADt ch\u01B0a, ho\u1EB7c tcp test b\u1EB1ng `curl -v https://ai.nal.vn`.":`
|
|
26
|
+
Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${Fr/1e3}s.${c}`)}let s=o.message||String(o);if(s.toLowerCase().includes("fetch failed")||s.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${s}${c}`)}throw o}finally{clearTimeout(r)}}async function Oa(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return i.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Ta({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Br(){let t=await Ra(),e=await _a();i.info(`Verify key (${V(t)}) qua ${e}/v1/models...`);let n=await Ia(e,t);i.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await Oa(n);return{apiKey:t,baseUrl:e,model:r}}import{promises as ce}from"fs";import{join as Qr}from"path";import{spawnSync as Na}from"child_process";import{existsSync as Ma,readFileSync as ja}from"fs";import{homedir as La}from"os";import{join as Ga}from"path";function cn(t,e){try{let n=Na(t,e,{encoding:"utf8",timeout:5e3});return{ok:n.status===0,stdout:(n.stdout??"").trim()}}catch{return{ok:!1,stdout:""}}}async function ft(){let t=cn("direnv",["--version"]);if(t.ok&&t.stdout){let n=cn("which",["direnv"]);return{installed:!0,version:t.stdout,binaryPath:n.ok?n.stdout:null,shellHookInZshrc:Wr()}}let e=cn("which",["direnv"]);return e.ok&&e.stdout?{installed:!0,version:null,binaryPath:e.stdout,shellHookInZshrc:Wr()}:{installed:!1,version:null,binaryPath:null,shellHookInZshrc:!1}}function Wr(){let t=Ga(La(),".zshrc");if(!Ma(t))return!1;try{let e=ja(t,"utf8");return/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(e)}catch{return!1}}import{spawnSync as zr}from"child_process";import{appendFileSync as Ua,existsSync as Da,readFileSync as Ha}from"fs";import{homedir as Ka,platform as Va}from"os";import{join as Fa}from"path";var ot="https://direnv.net/docs/installation.html";function oe(t){return zr("which",[t],{encoding:"utf8",timeout:3e3}).status===0}function ie(t,e){try{let n=zr(t,e,{encoding:"utf8",timeout:12e4});return{ok:n.status===0,output:((n.stdout??"")+(n.stderr??"")).trim()}}catch(n){return{ok:!1,output:n instanceof Error?n.message:String(n)}}}async function se(){let t=Va();if(t==="darwin"){if(oe("brew")){let e=ie("brew",["install","direnv"]);return{success:e.ok,method:"brew",message:e.ok?"Installed via Homebrew":`brew install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}return{success:!1,method:null,message:"macOS detected but Homebrew not found. Install brew first: https://brew.sh",manualInstructionsUrl:ot}}if(t==="linux"){if(oe("apt-get")){let e=ie("sudo",["apt-get","install","-y","direnv"]);return{success:e.ok,method:"apt-get",message:e.ok?"Installed via apt-get":`apt-get install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}if(oe("dnf")){let e=ie("sudo",["dnf","install","-y","direnv"]);return{success:e.ok,method:"dnf",message:e.ok?"Installed via dnf":`dnf install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}if(oe("pacman")){let e=ie("sudo",["pacman","-S","--noconfirm","direnv"]);return{success:e.ok,method:"pacman",message:e.ok?"Installed via pacman":`pacman install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}return{success:!1,method:null,message:"Linux detected but no supported package manager (apt-get/dnf/pacman) found.",manualInstructionsUrl:ot}}return{success:!1,method:null,message:`Unsupported platform: ${t}. Install direnv manually.`,manualInstructionsUrl:ot}}function ae(){let t=Fa(Ka(),".zshrc"),e='eval "$(direnv hook zsh)"',n="# direnv shell integration (added by avatar-cli)",r="";if(Da(t))try{r=Ha(t,"utf8")}catch{}if(/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(r))return{modified:!1,zshrcPath:t};let o=`
|
|
23
27
|
${n}
|
|
24
28
|
${e}
|
|
25
|
-
`;try{return
|
|
29
|
+
`;try{return Ua(t,o,"utf8"),{modified:!0,zshrcPath:t}}catch{return{modified:!1,zshrcPath:t}}}f();import{spawnSync as Ba}from"child_process";import{chmodSync as Wa,existsSync as Jr,readFileSync as qr,renameSync as za,writeFileSync as Ja}from"fs";import{join as qa}from"path";var Ya=".envrc",Yr="# >>> avatar-cli secrets >>>",ln="# <<< avatar-cli secrets <<<",Xa=`${Yr}
|
|
30
|
+
# Per-project secrets managed by avatar-cli.
|
|
31
|
+
# DO NOT COMMIT \u2014 must be gitignored (avatar enforces this on setup).
|
|
32
|
+
# Edit via: avatar secrets set <NAME>
|
|
33
|
+
# Remove via: avatar secrets rm <NAME>
|
|
34
|
+
# View: avatar secrets list`;function ht(t){return qa(t,Ya)}function Xr(t){let e=t.indexOf(Yr);if(e===-1)return null;let n=t.indexOf(ln,e);return n===-1?null:t.slice(e,n+ln.length)}function Zr(t){let e=new Map,n=/^export\s+([A-Z][A-Z0-9_]*)="((?:[^"\\]|\\.)*)"$/gm,r;for(;(r=n.exec(t))!==null;){let o=r[1],s=r[2];if(!o||s===void 0)continue;let a=s.replace(/\\"/g,'"').replace(/\\\\/g,"\\");e.set(o,a)}return e}function Za(t){return t.replace(/\\/g,"\\\\").replace(/"/g,'\\"')}function Qa(t){let e=[Xa],n=[...t.keys()].sort();for(let r of n){let o=t.get(r);o!==void 0&&e.push(`export ${r}="${Za(o)}"`)}return e.push(ln),e.join(`
|
|
35
|
+
`)}async function it(t,e){let n=ht(t),r="";Jr(n)&&(r=qr(n,"utf8"));let o=Xr(r),s=o?Zr(o):new Map,a=s.size;for(let[h,x]of Object.entries(e))x===null?s.delete(h):s.set(h,x);let c=Qa(s),l,u=!1;o?(l=r.replace(o,c),u=r.replace(o,"").trim().length>0):r.trim()?(u=!0,l=`${r.replace(/\s+$/,"")}
|
|
36
|
+
|
|
37
|
+
${c}
|
|
38
|
+
`):l=`${c}
|
|
39
|
+
`;let m=`${n}.avatar-tmp-${process.pid}`;Ja(m,l,{encoding:"utf8",mode:384}),za(m,n),Wa(n,384);let g=null;try{g=Ba("direnv",["allow",n],{encoding:"utf8",timeout:5e3}).status===0}catch{g=null}return{envrcPath:n,varsBefore:a,varsAfter:s.size,direnvAllowOk:g,hadExistingNonAvatarContent:u}}function F(t){let e=ht(t);if(!Jr(e))return new Map;let n=qr(e,"utf8"),r=Xr(n);return r?Zr(r):new Map}var tc=[".claude",".env"];async function ec(t,e){let n=Qr(t,...tc);await ce.mkdir(Qr(t,".claude"),{recursive:!0});let r="";try{r=await ce.readFile(n,"utf8")}catch{}let o=r?r.split(`
|
|
26
40
|
`):[],s=new Set(Object.keys(e)),a=o.filter(l=>{let u=l.match(/^([A-Z][A-Z0-9_]*)=/);return!(u?.[1]&&s.has(u[1]))});for(let[l,u]of Object.entries(e)){let m=u.replace(/'/g,"'\\''");a.push(`${l}='${m}'`)}let c=a.filter((l,u,m)=>!(l===""&&m[u-1]==="")).join(`
|
|
27
|
-
`);return await
|
|
41
|
+
`);return await ce.writeFile(n,c.endsWith(`
|
|
28
42
|
`)?c:`${c}
|
|
29
|
-
`,{encoding:"utf8",mode:384}),await fe.chmod(n,384),n}async function ln(t,e){let n=await bt(),i=!1,o=!1;if(!n.installed){r.info("\u0110ang setup direnv \u0111\u1EC3 b\u1EA3o v\u1EC7 API key (silent)...");let a=await de();if(a.success)r.dim(` \u2713 direnv installed via ${a.method}`),i=!0,n=await bt();else{r.warn(` ! direnv install fail (${a.message}). Fallback sang .claude/.env (gitignored).`);let c=await la(t,e);return r.warn(` \u26A0 Key \u0111\xE3 ghi v\xE0o ${c} (chmod 600) nh\u01B0ng KH\xD4NG auto-load \u2014 c\u1EA7n 'source ${c}' m\u1ED7i terminal m\u1EDBi.`),r.dim(` Manual install direnv sau: ${a.manualInstructionsUrl}`),{storageMethod:"claude-env-fallback",storagePath:c,autoLoad:!1,installedDirenvNow:!1,addedShellHook:!1}}}if(!n.shellHookInZshrc){let a=ge();a.modified&&(r.dim(` \u2713 Added direnv hook to ${a.zshrcPath}`),r.dim(" Reload shell: 'source ~/.zshrc' ho\u1EB7c m\u1EDF terminal m\u1EDBi"),o=!0)}let s=await st(t,e);return r.success(`\u2713 Key \u0111\xE3 l\u01B0u (${Object.keys(e).length} vars) v\xE0o ${s.envrcPath}`),s.direnvAllowOk===!1&&r.warn(" direnv allow fail \u2014 ch\u1EA1y 'direnv allow' trong workspace \u0111\u1EC3 activate."),{storageMethod:"envrc",storagePath:s.envrcPath,autoLoad:!0,installedDirenvNow:i,addedShellHook:o}}k();import{promises as ua}from"fs";import{join as pa}from"path";var un=384;function ma(t){return pa(t,".claude","settings.json")}async function da(t){if(!await d(t))return{};try{return await y(t)}catch(e){throw new Error(`Kh\xF4ng parse \u0111\u01B0\u1EE3c ${t} (JSON l\u1ED7i): ${e.message}. Backup file r\u1ED3i x\xF3a \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}function ga(t,e){let{env:n,...i}=t,o={...i,model:e};if(n){let{ANTHROPIC_BASE_URL:s,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(o.env=l)}return o}function Si(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...i}=t;return i}function fa(t,e,n,i,o){let a={...Si(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:i,avatarProvider:"llmlite"}}function ha(t,e,n,i,o){let a={...Si(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:i,avatarProvider:"anthropic"}}function ka(t,e){let n=e.env||{},i=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...i?{model:i}:{}}}async function at(t,e){let n=ma(t),i=await da(n),o;switch(e.provider){case"subscription":o=ga(i,e.model);break;case"llmlite":o=fa(i,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=ha(i,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=ka(i,e.sourceSettings);break}await $(n,o,un);try{await ua.chmod(n,un)}catch{}return{path:n,mode:un}}var j="sonnet";async function he(t){try{r.info("Setup AI provider cho workspace...");let e=Gt();if(e.installed)r.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(r.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),ci(),ae(),e=Gt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=sn();switch(await ui(n)){case"subscription":{let o=nn();if(o.state!=="authenticated"&&(on(),o=nn()),o.state==="authenticated"&&o.subscriptionType)return await at(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),r.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j};r.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=rn();if(!s.ok&&s.reason==="auth-expired"&&(r.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),on(),s=rn()),!s.ok&&(s.reason==="timeout"||s.reason==="unknown"))return r.warn(`Probe verify ${s.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),s.detail?.trim()&&r.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),await at(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),r.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j};if(!s.ok){let a=s.reason??"unknown";return await v("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),r.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),s.detail?.trim()&&r.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),r.warn(` \u2192 ${oi(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await at(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup","provider=subscription,result=ok"),r.success(`AI ready \xB7 Subscription \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j}}case"llmlite":{let o=await gi(),s=await ln(t.workspacePath,{ANTHROPIC_AUTH_TOKEN:o.apiKey});return await at(t.workspacePath,{provider:"llmlite",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:s.storageMethod==="envrc"}),await v("ai_setup",`provider=llmlite,result=ok,model=${o.model},base=${o.baseUrl},storage=${s.storageMethod}`),r.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"llmlite",model:o.model}}case"anthropic":{let o=await mi(),s=await ln(t.workspacePath,{ANTHROPIC_API_KEY:o.apiKey});return await at(t.workspacePath,{provider:"anthropic",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:s.storageMethod==="envrc"}),await v("ai_setup",`provider=anthropic,result=ok,model=${o.model},storage=${s.storageMethod}`),r.success(`AI ready \xB7 Anthropic Direct \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"anthropic",model:o.model}}case"use-global":{if(!n.rawSettings)throw new Error("use-global ch\u1ECDn nh\u01B0ng kh\xF4ng \u0111\u1ECDc \u0111\u01B0\u1EE3c global settings.");return await at(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await v("ai_setup","provider=use-global,result=ok"),r.success(`AI ready \xB7 Copy t\u1EEB global config (${n.baseUrl??"subscription"})`),{ok:!0,provider:"use-global",model:n.model}}}}catch(e){let n=e instanceof Error?e.message:String(e);return r.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),r.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await v("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}import{spawnSync as wa}from"child_process";var pn=1e4,Ai=3e4,Pi=5,mn="say ok",Ci="2023-06-01";async function ya(t,e,n){r.info(`Testing LLMLite provider: ${t} (key: ${D(e)})`);let i=new AbortController,o=setTimeout(()=>i.abort(),pn);try{let s=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:i.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(f=>typeof f.id=="string"?f.id:null).filter(f=>f!==null);if(r.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let f=c.slice(0,5).join(", "),S=c.length>5?` ...+${c.length-5} more`:"";r.dim(` Models: ${f}${S}`)}r.info(`Testing chat completion v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/chat/completions`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify({model:n,messages:[{role:"user",content:mn}],max_tokens:Pi}),signal:i.signal});if(!l.ok){let f=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${f}`)}let u=await l.json(),m=typeof u.choices?.[0]?.message?.content=="string"?u.choices[0].message.content:"(empty response)",g=u.usage?.total_tokens??"?";r.success(`Response: "${String(m).trim().slice(0,100)}"`),r.dim(` Tokens used: ${g}`)}catch(s){throw s.name==="AbortError"?new Error(`Timeout ${pn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(o)}}function va(){r.info("Testing Subscription provider qua `claude --print`...");let t=wa("claude",["--print",mn],{encoding:"utf8",timeout:Ai});if(t.signal==="SIGTERM")throw new Error(`Timeout ${Ai/1e3}s. Check m\u1EA1ng / endpoint.`);if(t.status!==0){let e=(t.stderr||"").toLowerCase();throw e.includes("401")||e.includes("invalid authentication")||e.includes("unauthorized")?new Error("Token Claude Code stale (401). Fix: `claude auth logout && claude auth login`."):new Error(`Test fail (exit ${t.status}). Stderr: ${(t.stderr||"").slice(0,200)}`)}r.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function ba(t,e,n){r.info(`Testing Anthropic Direct provider: ${t} (key: ${D(e)})`);let i=new AbortController,o=setTimeout(()=>i.abort(),pn);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":Ci},signal:i.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);r.success(`Connectivity OK \xB7 ${c.length} models available`),r.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":Ci,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:Pi,messages:[{role:"user",content:mn}]}),signal:i.signal});if(!l.ok){let g=(await l.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${l.status}): ${g}`)}let m=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);r.success(`Response: "${m}"`)}finally{clearTimeout(o)}}async function $i(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,i=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,o=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,s=typeof t.model=="string"?t.model:"default";return o&&n?(await ba(n,o,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&i?(await ya(n,i,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(va(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function ke(){let t=process.cwd(),e=vt(t);return e||(r.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
43
|
+
`,{encoding:"utf8",mode:384}),await ce.chmod(n,384),n}async function un(t,e){let n=await ft(),r=!1,o=!1;if(!n.installed){i.info("\u0110ang setup direnv \u0111\u1EC3 b\u1EA3o v\u1EC7 API key (silent)...");let a=await se();if(a.success)i.dim(` \u2713 direnv installed via ${a.method}`),r=!0,n=await ft();else{i.warn(` ! direnv install fail (${a.message}). Fallback sang .claude/.env (gitignored).`);let c=await ec(t,e);return i.warn(` \u26A0 Key \u0111\xE3 ghi v\xE0o ${c} (chmod 600) nh\u01B0ng KH\xD4NG auto-load \u2014 c\u1EA7n 'source ${c}' m\u1ED7i terminal m\u1EDBi.`),i.dim(` Manual install direnv sau: ${a.manualInstructionsUrl}`),{storageMethod:"claude-env-fallback",storagePath:c,autoLoad:!1,installedDirenvNow:!1,addedShellHook:!1}}}if(!n.shellHookInZshrc){let a=ae();a.modified&&(i.dim(` \u2713 Added direnv hook to ${a.zshrcPath}`),i.dim(" Reload shell: 'source ~/.zshrc' ho\u1EB7c m\u1EDF terminal m\u1EDBi"),o=!0)}let s=await it(t,e);return i.success(`\u2713 Key \u0111\xE3 l\u01B0u (${Object.keys(e).length} vars) v\xE0o ${s.envrcPath}`),s.direnvAllowOk===!1&&i.warn(" direnv allow fail \u2014 ch\u1EA1y 'direnv allow' trong workspace \u0111\u1EC3 activate."),{storageMethod:"envrc",storagePath:s.envrcPath,autoLoad:!0,installedDirenvNow:r,addedShellHook:o}}f();import{promises as nc}from"fs";import{join as rc}from"path";var pn=384;function oc(t){return rc(t,".claude","settings.json")}async function ic(t){if(!await d(t))return{};try{return await k(t)}catch(e){throw new Error(`Kh\xF4ng parse \u0111\u01B0\u1EE3c ${t} (JSON l\u1ED7i): ${e.message}. Backup file r\u1ED3i x\xF3a \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}function sc(t,e){let{env:n,...r}=t,o={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:s,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(o.env=l)}return o}function to(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function ac(t,e,n,r,o){let a={...to(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function cc(t,e,n,r,o){let a={...to(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function lc(t,e){let n=e.env||{},r=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...r?{model:r}:{}}}async function st(t,e){let n=oc(t),r=await ic(n),o;switch(e.provider){case"subscription":o=sc(r,e.model);break;case"llmlite":o=ac(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=cc(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=lc(r,e.sourceSettings);break}await b(n,o,pn);try{await nc.chmod(n,pn)}catch{}return{path:n,mode:pn}}var M="sonnet";async function le(t){try{i.info("Setup AI provider cho workspace...");let e=Rt();if(e.installed)i.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(i.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),Ur(),te(),e=Rt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=an();switch(await Hr(n)){case"subscription":{let o=rn();if(o.state!=="authenticated"&&(on(),o=rn()),o.state==="authenticated"&&o.subscriptionType)return await st(t.workspacePath,{provider:"subscription",model:M}),await v("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),i.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${M}`),{ok:!0,provider:"subscription",model:M};i.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=sn();if(!s.ok&&s.reason==="auth-expired"&&(i.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),on(),s=sn()),!s.ok&&(s.reason==="timeout"||s.reason==="unknown"))return i.warn(`Probe verify ${s.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),s.detail?.trim()&&i.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),await st(t.workspacePath,{provider:"subscription",model:M}),await v("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),i.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${M}`),{ok:!0,provider:"subscription",model:M};if(!s.ok){let a=s.reason??"unknown";return await v("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),i.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),s.detail?.trim()&&i.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),i.warn(` \u2192 ${Mr(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await st(t.workspacePath,{provider:"subscription",model:M}),await v("ai_setup","provider=subscription,result=ok"),i.success(`AI ready \xB7 Subscription \xB7 model=${M}`),{ok:!0,provider:"subscription",model:M}}case"llmlite":{let o=await Br(),s=await un(t.workspacePath,{ANTHROPIC_AUTH_TOKEN:o.apiKey});return await st(t.workspacePath,{provider:"llmlite",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:s.storageMethod==="envrc"}),await v("ai_setup",`provider=llmlite,result=ok,model=${o.model},base=${o.baseUrl},storage=${s.storageMethod}`),i.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"llmlite",model:o.model}}case"anthropic":{let o=await Vr(),s=await un(t.workspacePath,{ANTHROPIC_API_KEY:o.apiKey});return await st(t.workspacePath,{provider:"anthropic",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:s.storageMethod==="envrc"}),await v("ai_setup",`provider=anthropic,result=ok,model=${o.model},storage=${s.storageMethod}`),i.success(`AI ready \xB7 Anthropic Direct \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"anthropic",model:o.model}}case"use-global":{if(!n.rawSettings)throw new Error("use-global ch\u1ECDn nh\u01B0ng kh\xF4ng \u0111\u1ECDc \u0111\u01B0\u1EE3c global settings.");return await st(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await v("ai_setup","provider=use-global,result=ok"),i.success(`AI ready \xB7 Copy t\u1EEB global config (${n.baseUrl??"subscription"})`),{ok:!0,provider:"use-global",model:n.model}}}}catch(e){let n=e instanceof Error?e.message:String(e);return i.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),i.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await v("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}f();import{spawnSync as uc}from"child_process";f();var dn=1e4,eo=3e4,ro=5,mn="say ok",no="2023-06-01";async function pc(t,e,n){i.info(`Testing LLMLite provider: ${t} (key: ${V(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),dn);try{let s=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(h=>typeof h.id=="string"?h.id:null).filter(h=>h!==null);if(i.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let h=c.slice(0,5).join(", "),x=c.length>5?` ...+${c.length-5} more`:"";i.dim(` Models: ${h}${x}`)}i.info(`Testing chat completion v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/chat/completions`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify({model:n,messages:[{role:"user",content:mn}],max_tokens:ro}),signal:r.signal});if(!l.ok){let h=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${h}`)}let u=await l.json(),m=typeof u.choices?.[0]?.message?.content=="string"?u.choices[0].message.content:"(empty response)",g=u.usage?.total_tokens??"?";i.success(`Response: "${String(m).trim().slice(0,100)}"`),i.dim(` Tokens used: ${g}`)}catch(s){throw s.name==="AbortError"?new Error(`Timeout ${dn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(o)}}function dc(){i.info("Testing Subscription provider qua `claude --print`...");let t=uc("claude",["--print",mn],{encoding:"utf8",timeout:eo});if(t.signal==="SIGTERM")throw new Error(`Timeout ${eo/1e3}s. Check m\u1EA1ng / endpoint.`);if(t.status!==0){let e=(t.stderr||"").toLowerCase();throw e.includes("401")||e.includes("invalid authentication")||e.includes("unauthorized")?new Error("Token Claude Code stale (401). Fix: `claude auth logout && claude auth login`."):new Error(`Test fail (exit ${t.status}). Stderr: ${(t.stderr||"").slice(0,200)}`)}i.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function mc(t,e,n){i.info(`Testing Anthropic Direct provider: ${t} (key: ${V(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),dn);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":no},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);i.success(`Connectivity OK \xB7 ${c.length} models available`),i.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":no,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:ro,messages:[{role:"user",content:mn}]}),signal:r.signal});if(!l.ok){let g=(await l.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${l.status}): ${g}`)}let m=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);i.success(`Response: "${m}"`)}finally{clearTimeout(o)}}async function oo(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,r=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,o=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,s=typeof t.model=="string"?t.model:"default";return o&&n?(await mc(n,o,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await pc(n,r,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(dc(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function ue(){let t=process.cwd(),e=D(t);return e||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
30
44
|
Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
|
|
31
45
|
B\u1EA1n \u0111ang \u1EDF: ${t}
|
|
32
|
-
Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&r.dim(`Detected workspace root: ${e}`),e}async function dn(t){let e=Ti(t,".claude","settings.json");if(!await d(e))return{};try{return await y(e)}catch{return{}}}async function Aa(){let t=await ke();await he({workspacePath:t})}async function Ca(){let t=await ke(),e=await dn(t),n=e.env||{},i=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,o=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,s=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;s?(c=i?.includes("api.anthropic.com")||s.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=D(s)):i&&o?(c="LLMLite",l=D(o)):o?(c="Custom",l=D(o)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),r.info(`Project: ${t}`),r.info(`Provider: ${c}${i?` (${i})`:""}`),r.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),r.info(`Token: ${l}`)}async function Pa(){let t=await ke(),e=await dn(t);try{let n=await $i(e);r.success(`\u2713 ${n.message}`)}catch(n){r.error(`Test fail: ${n.message}`),process.exit(1)}}async function $a(t){let e=await ke(),n=Ti(e,".claude","settings.json"),i=await dn(e);if(!t.yes&&!await Sa({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){r.dim("\u0110\xE3 h\u1EE7y.");return}let{env:o,...s}=i,a={...s};if(o){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:u,...m}=o;Object.keys(m).length>0&&(a.env=m)}Object.keys(a).length===0?(await xa.unlink(n).catch(()=>{}),r.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await $(n,a,384),r.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function Ei(t){let e=t.command("ai").description("Qu\u1EA3n l\xFD AI provider config (M12)");e.command("setup").description("Wizard setup/re-config AI provider cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await Aa()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await Ca()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await Pa()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await $a(n)})}import{input as Ia}from"@inquirer/prompts";import{spawnSync as Ta}from"child_process";import{existsSync as gn}from"fs";import{join as we}from"path";function Ea(t){let e=we(t,"src",".git"),n=we(t,".git"),i=we(t,".claude");if(!gn(n))throw new Error(`Kh\xF4ng ph\u1EA3i workspace root: ${t}
|
|
33
|
-
Ch\u1EA1y 'avatar commit' trong workspace dir (c\xF3 .git v\xE0 .claude/).`);if(!gn(i))throw new Error(`Kh\xF4ng th\u1EA5y .claude/ trong ${t}.
|
|
34
|
-
Ch\u1EA1y 'avatar commit' trong Avatar workspace, kh\xF4ng ph\u1EA3i project b\xECnh th\u01B0\u1EDDng.`);if(!gn(e))throw new Error(`Kh\xF4ng th\u1EA5y src/.git trong ${t}.
|
|
35
|
-
Workspace thi\u1EBFu submodule src/. Ch\u1EA1y 'avatar init' l\u1EA1i?`)}function M(t,e){let n=Ta("git",["-C",t,...e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(n.status!==0){let i=(n.stderr||"").trim();throw new Error(`git ${e.join(" ")} (in ${t}) failed:
|
|
36
|
-
${i}`)}return(n.stdout||"").trim()}function Ri(t){return M(t,["status","--porcelain"]).length>0}async function Ra(t,e){let n=we(t,"src");if(!Ri(n))return r.dim("src/: nothing to commit (clean)"),{};r.info("Committing src/ ..."),M(n,["add","."]),M(n,["commit","-m",e.message]);let i=M(n,["rev-parse","HEAD"]);r.success(`src/ committed: ${i.slice(0,7)}`);let o=!1;return e.push&&(r.info("Pushing src/ ..."),M(n,["push"]),r.success("src/ pushed"),o=!0),{sha:i,pushed:o}}async function _a(t,e){if(!Ri(t))return r.dim("workspace: nothing to commit (clean)"),{};r.info("Committing workspace root ..."),M(t,["add","."]),M(t,["commit","-m",e.message]);let n=M(t,["rev-parse","HEAD"]);r.success(`workspace committed: ${n.slice(0,7)}`);let i=!1;return e.push&&(r.info("Pushing workspace ..."),M(t,["push"]),r.success("workspace pushed"),i=!0),{sha:n,pushed:i}}async function _i(t){Ea(t.workspaceRoot);let e={target:t.target,skipped:[]};if(t.target==="src"||t.target==="all"){let n=await Ra(t.workspaceRoot,t.options);e.srcCommitSha=n.sha,e.srcPushed=n.pushed,n.sha||e.skipped?.push("src")}if(t.target==="workspace"||t.target==="all"){let n=await _a(t.workspaceRoot,t.options);e.workspaceCommitSha=n.sha,e.workspacePushed=n.pushed,n.sha||e.skipped?.push("workspace")}return e}function Ii(t){t.command("commit").description("Commit code kh\xE1ch trong src/ (Avatar state do admin sync) (M07)").command("src").description("Commit src/ \u2192 push l\xEAn client repo remote").option("-m, --message <msg>","Commit message").option("--push","Auto push sau commit (default: ch\u1EC9 commit)").action(async n=>{await Oa(n)})}async function Oa(t){try{let e=t.message??await Ia({message:"Commit message:",validate:i=>i.trim().length>0?!0:"Message b\u1EAFt bu\u1ED9c"}),n=await _i({workspaceRoot:process.cwd(),target:"src",options:{message:e,push:t.push}});n.srcCommitSha&&r.success(`src/: ${n.srcCommitSha.slice(0,7)}${n.srcPushed?" (pushed)":""}`),n.skipped&&n.skipped.length>0&&r.dim(`Skipped (nothing to commit): ${n.skipped.join(", ")}`)}catch(e){r.error(e instanceof Error?e.message:String(e)),process.exit(1)}}import{spawnSync as yn}from"child_process";import{promises as vn}from"fs";import{join as ct}from"path";import rc from"boxen";import{join as Wa}from"path";k();import{promises as Ga}from"fs";import{join as Li}from"path";k();import{promises as Na}from"fs";import Oi,{join as fn}from"path";async function ja(t,e){let i=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!i?.[2])return!0;let o=i[2];if(o.startsWith("/"))return r.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let s=fn(t,o),a=Oi.resolve(t),c=Oi.resolve(s);return!c.startsWith(`${a}/`)&&c!==a?(r.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await d(s)}function hn(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)+String(e.getMonth()+1).padStart(2,"0")+String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-${n}`}function Dt(t,e){let n=new Set,i=[];for(let o of[...t,...e]){let s=typeof o=="string"?o:JSON.stringify(o);n.has(s)||(n.add(s),i.push(o))}return i}function Ma(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function La(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],i=[];for(let o of n)if(typeof o=="object"&&o!==null){let s=o.command;typeof s=="string"&&i.push(s)}return i}function Ni(t){let e=t.map((s,a)=>{let c=La(s),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:s,commands:c,hasVar:l}}),n=new Map;for(let s of e){let a=s.commands.map(Ma).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(s),n.set(a,c)}let i=new Set;for(let[,s]of n)if(!(s.length<2||!s.some(c=>c.hasVar)))for(let c of s)c.hasVar||i.add(c.index);return i.size===0?{entries:t,droppedCount:0}:{entries:t.filter((s,a)=>!i.has(a)),droppedCount:i.size}}function Ha(t,e){let n=[],i={...e},o=0;for(let s of Object.keys(i)){let a=i[s]||[],{entries:c,droppedCount:l}=Ni(a);l>0&&(i[s]=c,o+=l,n.includes(s)||n.push(s))}for(let[s,a]of Object.entries(t)){let c=i[s]||[],l=Dt(c,a),{entries:u,droppedCount:m}=Ni(l);m>0&&(o+=m),u.length!==c.length&&(n.includes(s)||n.push(s)),i[s]=u}return{merged:i,touchedEvents:n,migratedCount:o}}async function ye(t){let e=fn(t,".claude","pack","templates","settings.json.tpl"),n=fn(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let i;try{let u=await G(e);i=JSON.parse(u)}catch(u){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${u.message}. Path: ${e}`)}let o={},s=!1;if(await d(n)){s=!0;try{o=await y(n)}catch(u){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${u.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...o};if(i.statusLine&&!o.statusLine&&(await ja(t,i.statusLine.command)?(c.statusLine=i.statusLine,a.push("statusLine added")):a.push(`statusLine SKIPPED (file ref '${i.statusLine.command}' kh\xF4ng t\u1ED3n t\u1EA1i)`)),typeof i.includeCoAuthoredBy=="boolean"&&typeof o.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=i.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),i.model&&!o.model&&(c.model=i.model,a.push("model added")),i.env||o.env){let u={...o.env||{}},m=!1;for(let[g,f]of Object.entries(u))typeof f=="string"&&f.includes("llm.nal.vn")&&(u[g]=f.replace(/llm\.nal\.vn/g,"ai.nal.vn"),m=!0);if(m&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),i.env)for(let[g,f]of Object.entries(i.env))g in u||(u[g]=f,m=!0);m&&(c.env=u,a.some(g=>g.includes("env vars"))||a.push("env vars added from pack"))}if(i.permissions){let u=o.permissions?.allow||[],m=o.permissions?.deny||[],g=i.permissions.allow||[],f=i.permissions.deny||[],S=Dt(u,g),w=Dt(m,f);(S.length!==u.length||w.length!==m.length)&&(c.permissions={allow:S,deny:w},a.push(`permissions union (+${S.length-u.length} allow, +${w.length-m.length} deny)`))}if(i.hooks){let u=o.hooks||{},{merged:m,touchedEvents:g,migratedCount:f}=Ha(i.hooks,u);(g.length>0||f>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),f>0&&a.push(`migrated ${f} stale relative-path hook entries to use \${CLAUDE_PROJECT_DIR} version (fixes hook failure when shell cwd changes)`))}if(a.length===0)return{action:"no-change",changes:[]};let l;return s&&(l=hn(n),await Na.copyFile(n,l)),await $(n,c),{action:"merged",backupPath:l,changes:a}}function Ua(t,e){return Li(t,".claude","pack","tools",e,"tool.json")}async function Vt(t,e){let n=Ua(t,e);return await d(n)?await y(n):null}var Da=[".claude","settings.json"];function Hi(t){return Li(t,...Da)}function ve(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function Gi(t){let e=Hi(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await y(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable tool.`)}}async function Ui(t,e,n){let i=Hi(t),o;return n&&(o=hn(i),await Ga.copyFile(i,o)),await $(i,e),o}function ji(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Mi(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Va(t,e){let n=new Set;for(let s of e)Mi(s)&&n.add(ji(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let i=0;return{migratedUser:t.filter(s=>Mi(s)?!0:n.has(ji(s))?(i++,!1):!0),droppedCount:i}}async function be(t,e){let{settings:n,existed:i}=await Gi(t),o={...n},s=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},m={...u},g=[],f=0;for(let[S,w]of Object.entries(a)){let P=u[S]??[],{migratedUser:I,droppedCount:A}=Va(P,w);A>0&&(f+=A);let O=new Set(I.map(ve)),_=w.filter(ht=>!O.has(ve(ht)));(_.length>0||A>0)&&(m[S]=[...I,..._],g.push(S))}g.length>0&&(o.hooks=m,s.push(`hooks added: ${g.join(", ")}`),f>0&&s.push(`migrated ${f} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],m=Dt(u,c);m.length!==u.length&&(o.permissions={...n.permissions,deny:m},s.push(`deny +${m.length-u.length}`))}return s.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await Ui(t,o,i),changes:s}}async function Di(t,e){let{settings:n,existed:i}=await Gi(t);if(!i)return{action:"no-change",changes:[]};let{hooks:o,permissions:s,...a}=n,c={...a},l=[],u=e.settings.hooks??{},m=n.hooks??{},g={},f=[];for(let[A,O]of Object.entries(m)){let _=u[A];if(!_){g[A]=O;continue}let ht=new Set(_.map(ve)),kt=O.filter(oe=>!ht.has(ve(oe)));kt.length!==O.length&&f.push(A),kt.length>0&&(g[A]=kt)}Object.keys(g).length>0&&(c.hooks=g),f.length>0&&l.push(`hooks removed: ${f.join(", ")}`);let S=new Set(e.settings.permissions?.deny??[]),w=n.permissions?.deny??[],P=S.size>0?w.filter(A=>!S.has(A)):w;if(n.permissions){let{deny:A,...O}=n.permissions,_={...O,...P.length>0?{deny:P}:{}};Object.keys(_).length>0&&(c.permissions=_)}return P.length!==w.length&&l.push(`deny -${w.length-P.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await Ui(t,c,i),changes:l}}k();import{join as Ka}from"path";var Ba=".claude/avatar-tools.json";function Vi(){return{tools:{}}}function Ki(t){return Ka(t,Ba)}async function St(t){let e=Ki(t);if(!await d(e))return Vi();try{return{tools:(await y(e)).tools??{}}}catch{return Vi()}}async function Fa(t,e){await $(Ki(t),e)}async function xe(t,e,n){let i=await St(t);return i.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Fa(t,i),i}async function At(t){let e=await St(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}k();function za(t,e){let n=e.settings.hooks??{},i=t.hooks??{};for(let[o,s]of Object.entries(n)){let a=i[o]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(u=>u.command??"")));for(let l of s)for(let u of l.hooks??[])if(u.command&&!c.has(u.command))return!1}return!0}async function Bi(t){let e=await At(t);if(e.length===0)return[];let n=Wa(t,".claude","settings.json"),i={};if(await d(n))try{i=await y(n)}catch{i={}}let o=[];for(let s of e){let a=await Vt(t,s);if(!a){o.push({name:`Tool: ${s}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}za(i,a)?o.push({name:`Tool: ${s}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):o.push({name:`Tool: ${s}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await be(t,a)}})}return o}k();k();import{join as kn}from"path";import{simpleGit as qa}from"simple-git";function b(t=process.cwd()){return qa({baseDir:t,binary:"git"})}async function Ct(t=process.cwd()){return await d(kn(t,".git"))}async function Fi(t,e,n=process.cwd()){await b(n).subModule(["add",t,e])}async function Kt(t,e,n=process.cwd()){let i=kn(n,t);await b(i).fetch(["--tags"]),await b(i).checkout(e)}async function Se(t,e,n=process.cwd()){let i=kn(n,t);await b(i).fetch(["origin"]),await b(i).checkout(["-B",e,`origin/${e}`])}async function K(t=process.cwd()){return(await b(t).tags()).all}async function Ae(t=process.cwd()){try{return(await b(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Pt(t=process.cwd()){return(await b(t).revparse(["HEAD"])).trim()}k();import{promises as Yi}from"fs";import{join as B}from"path";k();import{existsSync as Ya}from"fs";import{dirname as zi,join as Bt}from"path";import{fileURLToPath as Xa}from"url";var Ja=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g;function Wi(t,e){return t.replace(Ja,(n,i)=>{let o=e[i];return o===void 0?n:String(o)})}var Za=zi(Xa(import.meta.url)),qi=ec(Za),Qa=Bt(qi,"src","templates"),tc=Bt(qi,"src","hooks");function ec(t){let e=t;for(;;){if(Ya(Bt(e,"package.json")))return e;let n=zi(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function nc(t){return await G(Bt(Qa,`${t}.tpl`))}async function Ce(t,e){let n=await nc(t);return Wi(n,e)}async function Ji(t){return await G(Bt(tc,`${t}.sh.tpl`))}async function ic(t){if(!await d(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,i=n,o=1;for(;await d(i);)if(i=`${n}-${o}`,o++,o>5)throw new Error(`Could not find free backup name for ${t}`);return await Yi.rename(t,i),i}async function Xi(t,e,n){let i=await ic(t);return await yt(t,e,n),i}var oc=["state","_pending","_backup"];async function Zi(t){let e=B(t,".claude");await T(e);for(let n of oc){let i=B(e,n);await T(i),await yt(B(i,".gitkeep"),"")}}async function Qi(t,e){return[]}async function wn(t,e){let n=await Ce("CLAUDE.md",e);return await Xi(B(t,"CLAUDE.md"),n)}async function to(t,e){let n=await Ce("settings.json",e);return await Xi(B(t,".claude","settings.json"),n)}async function eo(t){let e=B(t,".gitignore"),n=await Ce("gitignore",{}),i="# Avatar \u2014 git-ignored entries injected on `avatar init`",o="";if(await d(e)&&(o=await Yi.readFile(e,"utf8"),o.includes(i)))return;let s=o.endsWith(`
|
|
37
|
-
`)||o.length===0?"":`
|
|
38
|
-
`;await yt(e,`${o}${s}
|
|
39
|
-
${n}`)}async function Ft(t,e){let n=await Ji(e),i=B(t,"hooks");await T(i);let o=B(i,e);await yt(o,n,493)}function no(t){t.command("doctor").description("Ch\u1EA9n \u0111o\xE1n c\xE0i \u0111\u1EB7t Avatar: hooks, MCP, login, submodule, ...").option("--fix","T\u1EF1 \u0111\u1ED9ng fix c\xE1c issue c\xF3 th\u1EC3 fix t\u1EF1 \u0111\u1ED9ng").action(async e=>{try{let n=await sc(process.cwd());ac(n),e.fix&&await cc(n)}catch(n){r.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function sc(t){let e=[],n=process.versions.node,[i,o]=n.split(".").map(N=>Number.parseInt(N,10)),s=(i??0)>18||(i??0)===18&&(o??0)>=17;e.push({name:"Node.js version",status:s?"ok":"fail",detail:`v${n}${s?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await tt();a?et(a)?e.push({name:"Login status",status:"warn",detail:`Token h\u1EBFt h\u1EA1n (${a.email}) \u2014 ch\u1EA1y 'avatar login'`,fixable:!1}):e.push({name:"Login status",status:"ok",detail:`Logged in: ${a.email}`,fixable:!1}):e.push({name:"Login status",status:"fail",detail:"Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y 'avatar login'",fixable:!1});let c=ct(t,".claude","pack"),l=ct(t,"CLAUDE.md"),u=ct(t,".git","hooks","post-merge"),[m,g,f,S]=await Promise.all([Ct(t),d(c),d(l),d(u)]);e.push({name:"Git repository",status:m?"ok":"warn",detail:m?t:"Kh\xF4ng ph\u1EA3i git repo (c\u1EA7n cho 'avatar init')",fixable:!1}),e.push({name:"team-ai-pack submodule",status:g?"ok":"warn",detail:g?c:"Avatar ch\u01B0a init \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),e.push({name:"CLAUDE.md",status:f?"ok":"warn",detail:f?"t\u1ED3n t\u1EA1i \u1EDF project root":"thi\u1EBFu \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),m&&g&&e.push({name:"Git hook post-merge",status:S?"ok":"fail",detail:S?"installed":"missing \u2014 fixable",fixable:!S,fix:S?void 0:async()=>{await Ft(ct(t,".git"),"post-merge")}});let w=ct(t,".gitignore"),P="";if(m){let N=!1;await d(w)&&(P=await vn.readFile(w,"utf8"),N=P.includes(".claude/_pending/")),e.push({name:".gitignore Avatar entries",status:N?"ok":g?"fail":"warn",detail:N?"c\xF3 .claude/_pending/, .claude/_backup/":"thi\u1EBFu entries",fixable:!1});let Z=P.includes(".claude/settings.json")||P.includes("/.claude/settings.json")||P.includes(".claude/*.json");e.push({name:"\u{1F512} settings.json gitignored",status:Z?"ok":"fail",detail:Z?"an to\xE0n \u2014 settings.json kh\xF4ng commit nh\u1EA7m":"CRITICAL: settings.json ch\u1EE9a API key \u2014 ch\u1EA1y 'avatar doctor --fix' \u0111\u1EC3 add gitignore",fixable:!Z,fix:Z?void 0:async()=>{await vn.appendFile(w,`
|
|
46
|
+
Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&i.dim(`Detected workspace root: ${e}`),e}async function gn(t){let e=io(t,".claude","settings.json");if(!await d(e))return{};try{return await k(e)}catch{return{}}}async function hc(){let t=await ue();await le({workspacePath:t})}async function wc(){let t=await ue(),e=await gn(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,o=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,s=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;s?(c=r?.includes("api.anthropic.com")||s.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=V(s)):r&&o?(c="LLMLite",l=V(o)):o?(c="Custom",l=V(o)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),i.info(`Project: ${t}`),i.info(`Provider: ${c}${r?` (${r})`:""}`),i.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),i.info(`Token: ${l}`)}async function kc(){let t=await ue(),e=await gn(t);try{let n=await oo(e);i.success(`\u2713 ${n.message}`)}catch(n){i.error(`Test fail: ${n.message}`),process.exit(1)}}async function yc(t){let e=await ue(),n=io(e,".claude","settings.json"),r=await gn(e);if(!t.yes&&!await fc({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){i.dim("\u0110\xE3 h\u1EE7y.");return}let{env:o,...s}=r,a={...s};if(o){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:u,...m}=o;Object.keys(m).length>0&&(a.env=m)}Object.keys(a).length===0?(await gc.unlink(n).catch(()=>{}),i.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await b(n,a,384),i.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function so(t){let e=t.command("ai").description("Qu\u1EA3n l\xFD AI provider config (M12)");e.command("setup").description("Wizard setup/re-config AI provider cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await hc()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await wc()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await kc()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await yc(n)})}import{spawnSync as wn}from"child_process";import{promises as kn}from"fs";import{join as Ot}from"path";import Nc from"boxen";import{join as Ic}from"path";import{promises as Pc}from"fs";import{join as po}from"path";import{promises as vc}from"fs";import ao,{join as fn}from"path";f();async function bc(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let o=r[2];if(o.startsWith("/"))return i.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let s=fn(t,o),a=ao.resolve(t),c=ao.resolve(s);return!c.startsWith(`${a}/`)&&c!==a?(i.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await d(s)}function hn(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)+String(e.getMonth()+1).padStart(2,"0")+String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-${n}`}function _t(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let s=typeof o=="string"?o:JSON.stringify(o);n.has(s)||(n.add(s),r.push(o))}return r}function xc(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Ac(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let o of n)if(typeof o=="object"&&o!==null){let s=o.command;typeof s=="string"&&r.push(s)}return r}function co(t){let e=t.map((s,a)=>{let c=Ac(s),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:s,commands:c,hasVar:l}}),n=new Map;for(let s of e){let a=s.commands.map(xc).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(s),n.set(a,c)}let r=new Set;for(let[,s]of n)if(!(s.length<2||!s.some(c=>c.hasVar)))for(let c of s)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((s,a)=>!r.has(a)),droppedCount:r.size}}function Sc(t,e){let n=[],r={...e},o=0;for(let s of Object.keys(r)){let a=r[s]||[],{entries:c,droppedCount:l}=co(a);l>0&&(r[s]=c,o+=l,n.includes(s)||n.push(s))}for(let[s,a]of Object.entries(t)){let c=r[s]||[],l=_t(c,a),{entries:u,droppedCount:m}=co(l);m>0&&(o+=m),u.length!==c.length&&(n.includes(s)||n.push(s)),r[s]=u}return{merged:r,touchedEvents:n,migratedCount:o}}async function pe(t){let e=fn(t,".claude","pack","templates","settings.json.tpl"),n=fn(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await N(e);r=JSON.parse(u)}catch(u){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${u.message}. Path: ${e}`)}let o={},s=!1;if(await d(n)){s=!0;try{o=await k(n)}catch(u){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${u.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...o};if(r.statusLine&&!o.statusLine&&(await bc(t,r.statusLine.command)?(c.statusLine=r.statusLine,a.push("statusLine added")):a.push(`statusLine SKIPPED (file ref '${r.statusLine.command}' kh\xF4ng t\u1ED3n t\u1EA1i)`)),typeof r.includeCoAuthoredBy=="boolean"&&typeof o.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=r.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),r.model&&!o.model&&(c.model=r.model,a.push("model added")),r.env||o.env){let u={...o.env||{}},m=!1;for(let[g,h]of Object.entries(u))typeof h=="string"&&h.includes("llm.nal.vn")&&(u[g]=h.replace(/llm\.nal\.vn/g,"ai.nal.vn"),m=!0);if(m&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[g,h]of Object.entries(r.env))g in u||(u[g]=h,m=!0);m&&(c.env=u,a.some(g=>g.includes("env vars"))||a.push("env vars added from pack"))}if(r.permissions){let u=o.permissions?.allow||[],m=o.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],x=_t(u,g),y=_t(m,h);(x.length!==u.length||y.length!==m.length)&&(c.permissions={allow:x,deny:y},a.push(`permissions union (+${x.length-u.length} allow, +${y.length-m.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:m,touchedEvents:g,migratedCount:h}=Sc(r.hooks,u);(g.length>0||h>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),h>0&&a.push(`migrated ${h} stale relative-path hook entries to use \${CLAUDE_PROJECT_DIR} version (fixes hook failure when shell cwd changes)`))}if(a.length===0)return{action:"no-change",changes:[]};let l;return s&&(l=hn(n),await vc.copyFile(n,l)),await b(n,c),{action:"merged",backupPath:l,changes:a}}function Cc(t,e){return po(t,".claude","pack","tools",e,"tool.json")}async function It(t,e){let n=Cc(t,e);return await d(n)?await k(n):null}var Ec=[".claude","settings.json"];function mo(t){return po(t,...Ec)}function de(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function go(t){let e=mo(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await k(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable tool.`)}}async function fo(t,e,n){let r=mo(t),o;return n&&(o=hn(r),await Pc.copyFile(r,o)),await b(r,e),o}function lo(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function uo(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Tc(t,e){let n=new Set;for(let s of e)uo(s)&&n.add(lo(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(s=>uo(s)?!0:n.has(lo(s))?(r++,!1):!0),droppedCount:r}}async function me(t,e){let{settings:n,existed:r}=await go(t),o={...n},s=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},m={...u},g=[],h=0;for(let[x,y]of Object.entries(a)){let E=u[x]??[],{migratedUser:I,droppedCount:P}=Tc(E,y);P>0&&(h+=P);let O=new Set(I.map(de)),$=y.filter(T=>!O.has(de(T)));($.length>0||P>0)&&(m[x]=[...I,...$],g.push(x))}g.length>0&&(o.hooks=m,s.push(`hooks added: ${g.join(", ")}`),h>0&&s.push(`migrated ${h} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],m=_t(u,c);m.length!==u.length&&(o.permissions={...n.permissions,deny:m},s.push(`deny +${m.length-u.length}`))}return s.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await fo(t,o,r),changes:s}}async function ho(t,e){let{settings:n,existed:r}=await go(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:s,...a}=n,c={...a},l=[],u=e.settings.hooks??{},m=n.hooks??{},g={},h=[];for(let[P,O]of Object.entries(m)){let $=u[P];if(!$){g[P]=O;continue}let T=new Set($.map(de)),R=O.filter(Me=>!T.has(de(Me)));R.length!==O.length&&h.push(P),R.length>0&&(g[P]=R)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&l.push(`hooks removed: ${h.join(", ")}`);let x=new Set(e.settings.permissions?.deny??[]),y=n.permissions?.deny??[],E=x.size>0?y.filter(P=>!x.has(P)):y;if(n.permissions){let{deny:P,...O}=n.permissions,$={...O,...E.length>0?{deny:E}:{}};Object.keys($).length>0&&(c.permissions=$)}return E.length!==y.length&&l.push(`deny -${y.length-E.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await fo(t,c,r),changes:l}}import{join as $c}from"path";var Rc=".claude/avatar-tools.json";function wo(){return{tools:{}}}function ko(t){return $c(t,Rc)}async function wt(t){let e=ko(t);if(!await d(e))return wo();try{return{tools:(await k(e)).tools??{}}}catch{return wo()}}async function _c(t,e){await b(ko(t),e)}async function ge(t,e,n){let r=await wt(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await _c(t,r),r}async function kt(t){let e=await wt(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function Oc(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[o,s]of Object.entries(n)){let a=r[o]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(u=>u.command??"")));for(let l of s)for(let u of l.hooks??[])if(u.command&&!c.has(u.command))return!1}return!0}async function yo(t){let e=await kt(t);if(e.length===0)return[];let n=Ic(t,".claude","settings.json"),r={};if(await d(n))try{r=await k(n)}catch{r={}}let o=[];for(let s of e){let a=await It(t,s);if(!a){o.push({name:`Tool: ${s}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}Oc(r,a)?o.push({name:`Tool: ${s}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):o.push({name:`Tool: ${s}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await me(t,a)}})}return o}f();function vo(t){t.command("doctor").description("Ch\u1EA9n \u0111o\xE1n c\xE0i \u0111\u1EB7t Avatar: hooks, MCP, login, submodule, ...").option("--fix","T\u1EF1 \u0111\u1ED9ng fix c\xE1c issue c\xF3 th\u1EC3 fix t\u1EF1 \u0111\u1ED9ng").action(async e=>{try{let n=await Mc(process.cwd());jc(n),e.fix&&await Lc(n)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Mc(t){let e=[],n=process.versions.node,[r,o]=n.split(".").map(T=>Number.parseInt(T,10)),s=(r??0)>18||(r??0)===18&&(o??0)>=17;e.push({name:"Node.js version",status:s?"ok":"fail",detail:`v${n}${s?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await K();a?et(a)?e.push({name:"Login status",status:"warn",detail:`Token h\u1EBFt h\u1EA1n (${a.email}) \u2014 ch\u1EA1y 'avatar login'`,fixable:!1}):e.push({name:"Login status",status:"ok",detail:`Logged in: ${a.email}`,fixable:!1}):e.push({name:"Login status",status:"fail",detail:"Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y 'avatar login'",fixable:!1});let c=Ot(t,".claude","pack"),l=Ot(t,"CLAUDE.md"),[u,m]=await Promise.all([d(c),d(l)]);e.push({name:"team-ai-pack installation",status:u?"ok":"warn",detail:u?c:"Avatar ch\u01B0a init \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),e.push({name:"CLAUDE.md",status:m?"ok":"warn",detail:m?"t\u1ED3n t\u1EA1i \u1EDF project root":"thi\u1EBFu \u2014 ch\u1EA1y 'avatar init'",fixable:!1});let g=Ot(t,".gitignore"),h="";if(u){let T=!1;await d(g)&&(h=await kn.readFile(g,"utf8"),T=h.includes(".claude/_pending/")),e.push({name:".gitignore Avatar entries",status:T?"ok":u?"fail":"warn",detail:T?"c\xF3 .claude/_pending/, .claude/_backup/":"thi\u1EBFu entries",fixable:!1});let R=h.includes(".claude/settings.json")||h.includes("/.claude/settings.json")||h.includes(".claude/*.json");e.push({name:"\u{1F512} settings.json gitignored",status:R?"ok":"fail",detail:R?"an to\xE0n \u2014 settings.json kh\xF4ng commit nh\u1EA7m":"CRITICAL: settings.json ch\u1EE9a API key \u2014 ch\u1EA1y 'avatar doctor --fix' \u0111\u1EC3 add gitignore",fixable:!R,fix:R?void 0:async()=>{await kn.appendFile(g,`
|
|
40
47
|
# Avatar v1.7.0 \u2014 Security: settings.json ch\u1EE9a raw API key, KH\xD4NG commit.
|
|
41
48
|
.claude/settings.json
|
|
42
49
|
.claude/settings.json.backup-*
|
|
43
|
-
`)}})}let
|
|
50
|
+
`)}})}let x=wn("which",["python"]),y=wn("which",["python3"]),E=x.status===0,I=y.status===0;I&&!E?e.push({name:"Python binary alias",status:"warn",detail:`Ch\u1EC9 c\xF3 python3 (modern macOS). Pack scripts th\u01B0\u1EDDng ref 'python' \u2192 suggest: ln -s ${y.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):E?e.push({name:"Python binary",status:"ok",detail:`python: ${x.stdout.toString().trim()}`,fixable:!1}):I&&e.push({name:"Python binary",status:"ok",detail:`python3: ${y.stdout.toString().trim()}`,fixable:!1});let P=Ot(t,".claude","settings.json");if(await d(P))try{let T=await kn.readFile(P,"utf8"),R=JSON.parse(T);if(R.statusLine?.command){let On=R.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(On?.[2]){let At=On[2],Gi=At.startsWith("/")?At:Ot(t,At),Nn=await d(Gi);e.push({name:"statusLine command",status:Nn?"ok":"fail",detail:Nn?`ref OK: ${At}`:`BROKEN: settings.json ref '${At}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let O=wn("which",["claude"]),$=O.status===0;if(e.push({name:"Claude Code CLI",status:$?"ok":"warn",detail:$?O.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),u){let T=await yo(t);e.push(...T)}return e}function jc(t){let e=[p.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,o=0;for(let s of t){let a=s.status==="ok"?p.green("\u2713"):s.status==="warn"?p.yellow("\u26A0"):p.red("\u2717");e.push(`${a} ${s.name.padEnd(28)} ${p.dim(s.detail)}`),s.status==="ok"?n+=1:(r+=1,s.fixable&&(o+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${o>0?` (${o} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${Nc(e.join(`
|
|
44
51
|
`),{padding:1,borderStyle:"round"})}
|
|
45
|
-
`)}async function
|
|
46
|
-
`).slice(-e).join(`
|
|
47
|
-
`)}function so(){let t=Ht("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=oo("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:uc,encoding:"utf8"});if(e.status!==0||e.signal==="SIGTERM"){t.fail("GitNexus setup failed");let n=(e.stderr||"").trim(),i=(e.stdout||"").trim();throw n?process.stderr.write(`${Pe(n,30)}
|
|
48
|
-
`):i&&process.stderr.write(`${Pe(i,30)}
|
|
49
|
-
`),ro("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function $e(t){let e=Ht(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=oo("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:pc,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let o=(n.stderr||"").trim(),s=(n.stdout||"").trim();throw o?process.stderr.write(`${Pe(o,30)}
|
|
50
|
-
`):s&&process.stderr.write(`${Pe(s,30)}
|
|
51
|
-
`),ro("analyze",n.status,n.signal,o)}let i=io(t,".gitnexus","meta.json");if(!lc(i))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new F("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${i}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);e.succeed(`Analyze OK (index t\u1EA1i ${io(t,".gitnexus")})`)}import{confirm as Gc}from"@inquirer/prompts";import Uc from"boxen";import{spawnSync as ao}from"child_process";var mc=5e3,dc=/(\d+\.\d+\.\d+)/;function gc(){let e=nt()==="win32"?"where":"which",n=ao(e,["gitnexus"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let i=(n.stdout||"").trim();return i?i.split(/\r?\n/)[0].trim():null}function fc(){let t=ao("gitnexus",["--version"],{encoding:"utf8",timeout:mc});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return dc.exec(e)?.[1]??null}var lt=null;function Wt(){if(lt!==null)return lt;let t=gc();return t?(lt={installed:!0,version:fc(),path:t},lt):(lt={installed:!1,version:null,path:null},lt)}function Te(){lt=null}import{spawnSync as hc}from"child_process";var co=300*1e3,lo="gitnexus",L=class extends Error{reason;exitCode;constructor(e,n,i=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=i}};function kc(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new L("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${lo} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new L("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new L("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function uo(){r.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=hc("npm",["install","-g",lo],{stdio:["inherit","inherit","pipe"],timeout:co,encoding:"utf8"});if(t.signal==="SIGTERM")throw new L("timeout",`npm install timeout sau ${co/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),kc(t.status,t.stderr||"");Te();let e=Wt();if(!e.installed||!e.path)throw new L("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return r.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{input as Qg,select as wc}from"@inquirer/prompts";var x=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function R(t){r.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&r.info(t.hint);let e=[{name:"Th\u1EED l\u1EA1i (Retry)",value:"retry"}];return t.allowSkip&&e.push({name:"B\u1ECF qua b\u01B0\u1EDBc n\xE0y v\xE0 ti\u1EBFp t\u1EE5c (Skip)",value:"skip"}),e.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i sau (Abort)",value:"abort"}),await wc({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}k();import{promises as go}from"fs";import{homedir as yc}from"os";import{join as vc}from"path";var po=384,mo={command:"gitnexus",args:["mcp"]};function bc(){return vc(yc(),".claude","mcp_servers.json")}function xc(t,e){return t===e?!0:typeof t!="object"||typeof e!="object"||t===null||e===null?!1:JSON.stringify(t)===JSON.stringify(e)}async function Sc(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await go.copyFile(t,n),n}async function fo(){let t=bc(),e={},n=!1;if(await d(t)){n=!0;try{e=await y(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let i=e.mcp_servers?.gitnexus;if(i&&xc(i,mo))return r.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await Sc(t),r.dim(`Backup ${t} \u2192 ${o}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:mo}};await $(t,s,po);try{await go.chmod(t,po)}catch{}return r.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}import{spawnSync as Rc}from"child_process";import{existsSync as _c}from"fs";import{join as vo}from"path";import{confirm as Ic}from"@inquirer/prompts";var Ac=[/^claude-(opus|sonnet)-4/i,/^claude-(opus|sonnet|haiku)-[5-9]/i,/^o1(-|$)/i,/^o3(-|$)/i,/^o4(-|$)/i,/nal-claude-(opus|sonnet)-?4/i];function ho(t){return t?Ac.some(e=>e.test(t)):!1}k();Ut();import{promises as zt}from"fs";import{homedir as Cc}from"os";import{dirname as Pc,join as ko}from"path";var $c=ko(Cc(),".gitnexus"),qt=ko($c,"config.json");async function Tc(){try{let t=await zt.readFile(qt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function Ec(t){await zt.mkdir(Pc(qt),{recursive:!0});let e=`${qt}.tmp-${process.pid}`;await zt.writeFile(e,t,{mode:384}),await zt.rename(e,qt)}async function wo(t){let n={...await Tc(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await Ec(JSON.stringify(n,null,2)),await zt.chmod(qt,384).catch(()=>{})}var Oc=900*1e3,Nc="nal-claude",jc="claude-sonnet-4-5";function Mc(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function Lc(t){let e=vo(t,".claude","settings.json");if(!await d(e))return null;try{let n=await y(e),i=n.env||{},o=typeof i.ANTHROPIC_BASE_URL=="string"?i.ANTHROPIC_BASE_URL:null;if(!o)return null;let s=typeof n.model=="string"?n.model:"",a=typeof i.ANTHROPIC_MODEL=="string"?i.ANTHROPIC_MODEL:"",c=s.length>0?s:a,l=null,u=()=>{if(l!==null)return l;try{l=V(t)}catch{l=new Map}return l},m=w=>{let P=process.env[w];if(typeof P=="string"&&P.length>0)return P;let I=i[w];if(typeof I=="string"&&I.length>0)return I;let A=u().get(w);return typeof A=="string"&&A.length>0?A:null},g=typeof n.avatarProvider=="string"?n.avatarProvider:null,f=!1;try{f=new URL(o).hostname==="api.anthropic.com"}catch{f=!1}if((g==="anthropic"?"anthropic":g==="llmlite"?"llmlite":f?"anthropic":"llmlite")==="anthropic"){let w=m("ANTHROPIC_API_KEY");if(w)return{provider:"anthropic",apiKey:w,baseUrl:Mc(o),model:c.length>0?c:jc}}else{let w=m("ANTHROPIC_AUTH_TOKEN");if(w)return{provider:"llmlite",apiKey:w,baseUrl:o,model:c.length>0?c:Nc}}return null}catch{return null}}async function Hc(t,e){return await Ic({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function yo(t,e){return t.split(`
|
|
52
|
+
`)}async function Lc(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),i.success(`Fixed: ${n.name}`),e+=1}catch(r){i.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&i.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}import{spawnSync as pl}from"child_process";import{promises as dl}from"fs";import{join as No}from"path";import{confirm as sl}from"@inquirer/prompts";import al from"boxen";import{spawnSync as Gc}from"child_process";f();var bo=300*1e3,xo="gitnexus",j=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function Uc(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new j("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${xo} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new j("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new j("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Ao(){i.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Gc("npm",["install","-g",xo],{stdio:["inherit","inherit","pipe"],timeout:bo,encoding:"utf8"});if(t.signal==="SIGTERM")throw new j("timeout",`npm install timeout sau ${bo/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Uc(t.status,t.stderr||"");Ft();let e=Q();if(!e.installed||!e.path)throw new j("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return i.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{promises as Co}from"fs";import{homedir as Dc}from"os";import{join as Hc}from"path";f();var So=384,Po={command:"gitnexus",args:["mcp"]};function Kc(){return Hc(Dc(),".claude","mcp_servers.json")}function Vc(t,e){return t===e?!0:typeof t!="object"||typeof e!="object"||t===null||e===null?!1:JSON.stringify(t)===JSON.stringify(e)}async function Fc(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await Co.copyFile(t,n),n}async function Eo(){let t=Kc(),e={},n=!1;if(await d(t)){n=!0;try{e=await k(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let r=e.mcp_servers?.gitnexus;if(r&&Vc(r,Po))return i.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await Fc(t),i.dim(`Backup ${t} \u2192 ${o}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:Po}};await b(t,s,So);try{await Co.chmod(t,So)}catch{}return i.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}import{spawnSync as Xc}from"child_process";import{existsSync as Zc}from"fs";import{join as Io}from"path";import{confirm as Qc}from"@inquirer/prompts";var Bc=[/^claude-(opus|sonnet)-4/i,/^claude-(opus|sonnet|haiku)-[5-9]/i,/^o1(-|$)/i,/^o3(-|$)/i,/^o4(-|$)/i,/nal-claude-(opus|sonnet)-?4/i];function To(t){return t?Bc.some(e=>e.test(t)):!1}f();import{promises as Nt}from"fs";import{homedir as Wc}from"os";import{dirname as zc,join as $o}from"path";var Jc=$o(Wc(),".gitnexus"),Mt=$o(Jc,"config.json");async function qc(){try{let t=await Nt.readFile(Mt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function Yc(t){await Nt.mkdir(zc(Mt),{recursive:!0});let e=`${Mt}.tmp-${process.pid}`;await Nt.writeFile(e,t,{mode:384}),await Nt.rename(e,Mt)}async function Ro(t){let n={...await qc(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await Yc(JSON.stringify(n,null,2)),await Nt.chmod(Mt,384).catch(()=>{})}var tl=900*1e3,el="nal-claude",nl="claude-sonnet-4-5";function rl(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function ol(t){let e=Io(t,".claude","settings.json");if(!await d(e))return null;try{let n=await k(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!o)return null;let s=typeof n.model=="string"?n.model:"",a=typeof r.ANTHROPIC_MODEL=="string"?r.ANTHROPIC_MODEL:"",c=s.length>0?s:a,l=null,u=()=>{if(l!==null)return l;try{l=F(t)}catch{l=new Map}return l},m=y=>{let E=process.env[y];if(typeof E=="string"&&E.length>0)return E;let I=r[y];if(typeof I=="string"&&I.length>0)return I;let P=u().get(y);return typeof P=="string"&&P.length>0?P:null},g=typeof n.avatarProvider=="string"?n.avatarProvider:null,h=!1;try{h=new URL(o).hostname==="api.anthropic.com"}catch{h=!1}if((g==="anthropic"?"anthropic":g==="llmlite"?"llmlite":h?"anthropic":"llmlite")==="anthropic"){let y=m("ANTHROPIC_API_KEY");if(y)return{provider:"anthropic",apiKey:y,baseUrl:rl(o),model:c.length>0?c:nl}}else{let y=m("ANTHROPIC_AUTH_TOKEN");if(y)return{provider:"llmlite",apiKey:y,baseUrl:o,model:c.length>0?c:el}}return null}catch{return null}}async function il(t,e){return await Qc({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function _o(t,e){return t.split(`
|
|
52
53
|
`).slice(-e).join(`
|
|
53
|
-
`)}async function
|
|
54
|
+
`)}async function Oo(t){let e=await ol(t);if(!e)return i.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
|
|
54
55
|
\u0110\xE3 ki\u1EC3m tra: process.env \u2192 .envrc Avatar block \u2192 settings.json.env
|
|
55
56
|
Nguy\xEAn nh\xE2n c\xF3 th\u1EC3:
|
|
56
57
|
\u2022 Subscription mode (OAuth, kh\xF4ng c\xF3 key)
|
|
57
58
|
\u2022 Key ch\u01B0a setup \u2014 ch\u1EA1y 'avatar ai setup' ho\u1EB7c 'avatar secrets set ANTHROPIC_API_KEY'
|
|
58
|
-
\u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),
|
|
59
|
-
gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await
|
|
60
|
-
`):m&&process.stderr.write(`${
|
|
61
|
-
`),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${l} (exit ${a.status??"null"})`}}let c=
|
|
59
|
+
\u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),i.dim(`\u0110\u1EC3 gen wiki sau khi \u0111\xE3 c\xF3 key, ch\u1EA1y manual:
|
|
60
|
+
gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await il(e.baseUrl,e.model))return i.dim("User decline wiki gen \u2014 workspace OK kh\xF4ng c\xF3 wiki. Ch\u1EA1y `gitnexus wiki` manual sau khi c\u1EA7n."),{ran:!1,skipped:!0,reason:"user-declined"};let r=To(e.model);await Ro({apiKey:e.apiKey,baseUrl:e.baseUrl,model:e.model,isReasoningModel:r});let o=["wiki",".","--base-url",e.baseUrl,"--model",e.model];r&&o.push("--reasoning-model");let s=Pt(`Generating wiki via ${e.baseUrl} (${e.provider}) model=${e.model}${r?" [reasoning]":""}`),a=Xc("gitnexus",o,{cwd:t,stdio:["ignore","pipe","pipe"],timeout:tl,encoding:"utf8"});if(a.status!==0||a.signal==="SIGTERM"){let l=a.signal==="SIGTERM"?"timeout":"non-zero-exit";s.fail(`Wiki gen ${l} (exit ${a.status??"null"})`);let u=(a.stderr||"").trim(),m=(a.stdout||"").trim();return u?process.stderr.write(`${_o(u,30)}
|
|
61
|
+
`):m&&process.stderr.write(`${_o(m,30)}
|
|
62
|
+
`),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${l} (exit ${a.status??"null"})`}}let c=Io(t,".gitnexus","wiki","index.html");return Zc(c)?(s.succeed(`Wiki ready: ${c}`),{ran:!0,skipped:!1,wikiPath:c}):(s.fail("Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y index.html"),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y ${c}`})}f();async function cl(){let t=[p.bold("\u{1F9E0} GitNexus ch\u01B0a c\xE0i"),"","GitNexus = code intelligence layer cho Claude Code:"," \u2022 Architectural awareness (impact analysis)"," \u2022 Call chain debug + blast radius tr\u01B0\u1EDBc refactor"," \u2022 Wiki HTML t\u1EF1 gen m\xF4 t\u1EA3 codebase","",`S\u1EBD c\xE0i: ${p.cyan("npm install -g gitnexus")} (global)`];return process.stdout.write(`${al(t.join(`
|
|
62
63
|
`),{padding:1,borderStyle:"round",borderColor:"cyan"})}
|
|
63
|
-
`),await
|
|
64
|
+
`),await sl({message:"C\xE0i GitNexus global?",default:!0})}async function ll(){for(;;)try{return Ao(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof j&&t.reason==="permission-denied"?"Th\u1EED l\u1EA1i v\u1EDBi sudo, ho\u1EB7c fix npm prefix: npm config set prefix ~/.npm-global":"Check log npm ph\xEDa tr\xEAn + th\u1EED l\u1EA1i.",r=await _({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function ul(t){for(;;)try{return gt(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof H&&e.reason==="missing-output"?"Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus version mismatch. Check `gitnexus --version`.":"Network glitch? Retry th\u01B0\u1EDDng work.",o=await _({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function fe(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{i.info("=== Phase 10: GitNexus Setup ===");let n=Q();if(!n.installed){if(!await cl())return await v("gitnexus_setup","result=skipped,reason=user-declined"),i.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await ll())return await v("gitnexus_setup","result=skipped,reason=install-skipped"),i.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(Ft(),n=Q(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,i.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{xr()}catch(r){i.warn(`gitnexus setup fail: ${r.message}`),i.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(t.skipAnalyze)i.dim("GitNexus: c\xE0i + MCP xong. Index s\u1EBD ch\u1EA1y per-repo khi `avatar add repo`.");else{if(!await ul(t.workspacePath))return await v("gitnexus_setup","result=skipped,reason=analyze-skipped"),i.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let o=await Oo(t.workspacePath);e.wikiGenerated=o.ran,o.skipped&&o.reason==="fail"&&i.warn(`Wiki gen fail (workspace v\u1EABn OK): ${o.detail??"unknown"}`)}try{let r=await Eo();e.mcpRegistered=!0,r.wasUpdated||i.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(r){i.warn(`MCP server register fail: ${r.message}`),i.dim("Workspace OK nh\u01B0ng Claude Code kh\xF4ng t\u1EF1 attach MCP server. Manual add v\xE0o ~/.claude/mcp_servers.json.")}return e.ok=!0,await v("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),i.success("GitNexus ready"),e}catch(n){if(n instanceof S)throw n;let r=n instanceof Error?n.message:String(n);return i.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),i.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await v("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}f();function yn(){let t=process.cwd(),e=D(t);return e||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
64
65
|
Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
|
|
65
66
|
B\u1EA1n \u0111ang \u1EDF: ${t}
|
|
66
|
-
Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&
|
|
67
|
-
`);let
|
|
68
|
-
`)}function
|
|
69
|
-
${
|
|
70
|
-
|
|
71
|
-
`)}import{spawnSync as $o}from"child_process";import{input as cl,select as ll}from"@inquirer/prompts";import{spawnSync as $n}from"child_process";import{promises as Qc}from"fs";import{basename as tl,join as Co}from"path";import{confirm as el,select as nl}from"@inquirer/prompts";import{spawnSync as Yc}from"child_process";var Cn=class extends Error{constructor(e){super(`Repo "${e}" \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. \u0110\u1ED5i t\xEAn ho\u1EB7c x\xF3a repo c\u0169.`),this.name="RepoAlreadyExistsError"}};function Ao(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],i=Yc("gh",n,{stdio:"inherit"});if(i.status!==0)throw i.status===1?new Cn(e):new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${i.status})`);return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}import{spawnSync as Xc}from"child_process";function Ie(){let t=Xc("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(t.status!==0)throw new Error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c GitHub username: ${t.stderr?.trim()}`);return t.stdout.trim()}var Zc=/^[a-zA-Z0-9._-]{1,100}$/,Pn=class extends Error{constructor(e){super(`T\xEAn repo "${e}" kh\xF4ng h\u1EE3p l\u1EC7. Ch\u1EC9 d\xF9ng ch\u1EEF/s\u1ED1/d\u1EA5u ch\u1EA5m/g\u1EA1ch/underscore, d\xE0i 1-100 k\xFD t\u1EF1.`),this.name="InvalidRepoNameError"}};function Oe(t){if(!Zc.test(t))throw new Pn(t)}function Ne(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function Tt(t){Oe(t.name),Ne(t.visibility);let e=t.org??Ie();r.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=Ao({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return r.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}k();function il(){let t=new Date;return`${t.getFullYear().toString().slice(-2)}${String(t.getMonth()+1).padStart(2,"0")}${String(t.getDate()).padStart(2,"0")}-${String(t.getHours()).padStart(2,"0")}${String(t.getMinutes()).padStart(2,"0")}`}async function ol(t){let e=Co(t,".git");if(!await d(e))throw new Error(`.git kh\xF4ng t\u1ED3n t\u1EA1i \u1EDF ${t} \u2014 kh\xF4ng c\u1EA7n reset.`);let n=`.git.backup-${il()}`,i=Co(t,n);return await Qc.rename(e,i),r.success(`Backup .git \u2192 ${n}`),i}function rl(t){let e=$n("git",["-C",t,"init","-b","main"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(e.status!==0)throw new Error(`git init th\u1EA5t b\u1EA1i: ${e.stderr||e.stdout}`);r.success("Git init m\u1EDBi (branch main)"),$n("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=$n("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0?r.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(n.stderr||"").slice(0,200)}`):r.success("Initial commit")}async function Po(t){let e=tl(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await el({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
|
|
72
|
-
1. Backup .git \u2192 .git.backup-{timestamp}
|
|
73
|
-
2. Git init m\u1EDBi (branch main, initial commit)
|
|
74
|
-
3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
|
|
75
|
-
Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let i=t.visibility??(t.autoYes?"private":await nl({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await ol(t.folderPath);return rl(t.folderPath),{newRemoteUrl:Tt({folder:t.folderPath,name:n,visibility:i,org:t.org}).httpsUrl,backupPath:o}}import{spawnSync as sl}from"child_process";var Tn=5e3;function al(t){let e=t.toLowerCase();return e.includes("authentication")||e.includes("could not read username")||e.includes("permission denied")||e.includes("403")||e.includes("access denied")||e.includes("repository not found")?"no-access":e.includes("404")||e.includes("does not exist")?"not-found":e.includes("could not resolve host")||e.includes("network")||e.includes("connection refused")||e.includes("connection timed out")?"network":"unknown"}function Et(t){let e=sl("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:Tn,stdio:["ignore","pipe","pipe"]});if(e.error){let o=e.error;return o.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:o.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${Tn/1e3}s`}:{ok:!1,reason:"unknown",detail:o.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${Tn/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:al(n),detail:n.slice(0,300)}}var Jt=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function ul(){let t=$o("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function pl(){r.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=$o("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&r.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function ml(t,e,n){switch(t){case"no-access":return n?`Repo c\xF3 th\u1EC3 private v\xE0 account '${n}' kh\xF4ng c\xF3 quy\u1EC1n access. Switch sang account c\xF3 quy\u1EC1n, ho\u1EB7c xin invite t\u1EEB owner repo.`:"Repo c\xF3 th\u1EC3 private v\xE0 gh CLI ch\u01B0a login. Login tr\u01B0\u1EDBc r\u1ED3i retry.";case"not-found":return`URL c\xF3 th\u1EC3 sai ch\xEDnh t\u1EA3, ho\u1EB7c repo \u0111\xE3 b\u1ECB x\xF3a/rename. Nh\u1EADp l\u1EA1i URL \u0111\xFAng. Check: ${e}`;case"network":return"Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c GitHub. Check m\u1EA1ng / VPN / firewall.";case"timeout":return"Network ch\u1EADm > 5s. Check m\u1EA1ng r\u1ED3i retry.";default:return"L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh. URL c\xF3 th\u1EC3 sai \u2014 nh\u1EADp l\u1EA1i, ho\u1EB7c check gh auth status."}}function dl(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function je(t){let e=t.url,n=t.initialReason,i=t.initialDetail;for(;;){let o=ul();r.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),r.dim(` L\xFD do: ${n}${i?` \u2014 ${i.slice(0,150)}`:""}`),r.info(ml(n,e,o)),o&&r.dim(` gh CLI hi\u1EC7n \u0111ang login: ${o}`);let s=[{name:"Nh\u1EADp l\u1EA1i URL \u0111\xFAng (recommended khi URL sai ch\xEDnh t\u1EA3)",value:"re-input-url"},{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"T\xF4i v\u1EEBa fix (accept invite / s\u1EEDa permission) \u2014 retry verify",value:"retry"}];t.folderPath&&s.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),s.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let a=await ll({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new Jt(`User ch\u1ECDn t\u1EA1m ng\u01B0ng. Fix access ${e} r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.`);if(a==="reset-folder"){if(!t.folderPath){r.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await Po({folderPath:t.folderPath,visibility:t.defaultVisibility});return r.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),r.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){r.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await cl({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>dl(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&pl(),r.info(`Verify remote l\u1EA1i: ${e}...`);let c=Et(e);if(c.ok)return r.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",i=c.detail}}import{readdirSync as El}from"fs";import{select as Rl}from"@inquirer/prompts";import{simpleGit as Io}from"simple-git";import{existsSync as gl,statSync as fl}from"fs";import{join as hl}from"path";function Me(t){let e=hl(t,".git");if(!gl(e))return!1;let n=fl(e);return n.isDirectory()||n.isFile()}import{simpleGit as kl}from"simple-git";var To="chore: initial commit";async function Yt(t){let e=kl({baseDir:t});await e.checkIsRepo().catch(()=>!1)||await e.init();try{await e.branch(["-M","main"])}catch{}await e.add(".");let i=await e.status();(await e.raw(["rev-list","-n","1","--all"]).catch(()=>"")).trim()||(i.files.length===0?await e.commit(To,void 0,{"--allow-empty":null}):await e.commit(To))}import{existsSync as wl}from"fs";import{join as yl}from"path";var vl={node:["package.json"],python:["pyproject.toml","requirements.txt","setup.py","Pipfile"],go:["go.mod"],rust:["Cargo.toml"],java:["pom.xml","build.gradle","build.gradle.kts"],ruby:["Gemfile"]};function Eo(t){let e=[];for(let[n,i]of Object.entries(vl))i.some(o=>wl(yl(t,o)))&&e.push(n);return e.length>0?e:["generic"]}import{readFileSync as bl}from"fs";import{dirname as xl,join as Xt}from"path";import{fileURLToPath as Sl}from"url";var Le=xl(Sl(import.meta.url)),Al=[Xt(Le,"templates","gitignore"),Xt(Le,"..","templates","gitignore"),Xt(Le,"..","..","src","templates","gitignore"),Xt(Le,"..","src","templates","gitignore")],Zt="# === avatar ===",ut="# === /avatar ===";function Cl(t){for(let e of Al)try{return bl(Xt(e,`${t}.txt`),"utf8")}catch{}throw new Error(`Kh\xF4ng t\xECm th\u1EA5y template gitignore cho stack "${t}"`)}function Ro(t){let n=["generic",...t.filter(i=>i!=="generic")].map(i=>`# --- ${i} ---
|
|
76
|
-
${Cl(i).trim()}`);return[Zt,...n,ut,""].join(`
|
|
77
|
-
`)}import{existsSync as Pl,readFileSync as $l,writeFileSync as En}from"fs";import{join as Tl}from"path";function _o(t,e){let n=Tl(t,".gitignore");if(!Pl(n)){En(n,e,"utf8");return}let i=$l(n,"utf8"),o=i.indexOf(Zt),s=i.indexOf(ut);if(o!==-1&&s!==-1&&s>o){let a=i.slice(0,o),c=i.slice(s+ut.length);En(n,`${a.trimEnd()}
|
|
78
|
-
|
|
79
|
-
${e}${c.trimStart()}`,"utf8");return}En(n,`${i.trimEnd()}
|
|
67
|
+
Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&i.dim(`Detected workspace root: ${e}`),e}async function ml(){let t=yn(),e=await fe({workspacePath:t});e.ok?(i.success("GitNexus setup complete"),i.dim("Update CLAUDE.md \u0111\u1EC3 re-render section GitNexus: re-run avatar init ho\u1EB7c ch\u1EC9nh tay.")):i.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function gl(){let t=yn(),e=No(t,".gitnexus","meta.json");if(!await d(e)){i.warn(`Ch\u01B0a c\xF3 ${e}. Ch\u1EA1y: avatar gitnexus install`);return}try{let n=await k(e);if(i.info(`Project: ${t}`),i.info(`Last commit: ${n.lastCommit?.slice(0,7)??"(unknown)"}`),i.info(`Indexed at: ${n.indexedAt??"(unknown)"}`),n.stats&&i.info(`Stats: ${n.stats.files??"?"} files \xB7 ${n.stats.nodes??"?"} nodes \xB7 ${n.stats.edges??"?"} edges`),n.lastCommit){let o=pl("git",["rev-parse","HEAD"],{cwd:t,encoding:"utf8"});o.status===0&&(o.stdout.trim()!==n.lastCommit?i.warn("\u26A0 Index stale \u2014 HEAD \u0111\xE3 ti\u1EBFn t\u1EEB lastCommit. Ch\u1EA1y: avatar gitnexus analyze"):i.dim("Index fresh (HEAD === lastCommit)"))}let r=No(t,".gitnexus","wiki","index.html");if(await d(r)){let o=await dl.stat(r);i.info(`Wiki: ${o.mtime.toISOString()}`)}else i.dim("Wiki: ch\u01B0a generate (ch\u1EA1y gitnexus wiki manual)")}catch(n){i.error(`Read meta.json fail: ${n.message}`),process.exit(1)}}async function fl(){let t=yn();try{gt(t),i.success("Index refreshed. Ch\u1EA1y `avatar gitnexus status` xem chi ti\u1EBFt.")}catch(e){i.error(`Analyze fail: ${e.message}`),process.exit(1)}}function Mo(t){let e=t.command("gitnexus").description("Qu\u1EA3n l\xFD GitNexus code intelligence (M10)");e.command("install").description("C\xE0i + setup GitNexus cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await ml()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await gl()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await fl()})}import{resolve as Ql}from"path";import{input as tu}from"@inquirer/prompts";import vn from"chalk";var he=[" \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2557 ","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u255A\u2550\u2550\u2588\u2588\u2554\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2554\u255D","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u255A\u2588\u2588\u2557 \u2588\u2588\u2554\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2551 \u2588\u2588\u2551 \u255A\u2588\u2588\u2588\u2588\u2554\u255D \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551","\u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u2550\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D"],we=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function bn(t,e,n){return Math.round(t+(e-t)*n)}function hl(t){let n=Math.max(0,Math.min(1,t))*(we.length-1),r=Math.floor(n),o=Math.min(we.length-1,r+1),s=n-r,a=we[r],c=we[o];return[bn(a[0],c[0],s),bn(a[1],c[1],s),bn(a[2],c[2],s)]}function xn(t){if(!((process.stdout.isTTY??!1)&&vn.level>0))return[...he,...t?.tagline?["",t.tagline]:[]].join(`
|
|
68
|
+
`);let r=he.map((o,s)=>{let a=he.length===1?0:s/(he.length-1),[c,l,u]=hl(a);return vn.rgb(c,l,u).bold(o)});return t?.tagline&&(r.push(""),r.push(vn.dim(t.tagline))),r.join(`
|
|
69
|
+
`)}function yt(t){process.stdout.write(`
|
|
70
|
+
${xn(t)}
|
|
80
71
|
|
|
81
|
-
|
|
82
|
-
`),{padding:1,
|
|
83
|
-
`)}function
|
|
84
|
-
`)
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
1. T\u1EA1o repo: gh repo create <owner>/team-ai-pack --private --add-readme
|
|
91
|
-
2. Ho\u1EB7c override URL: export AVATAR_TEAM_PACK_REPO_URL=<url-repo-c\u1EE7a-b\u1EA1n>
|
|
92
|
-
3. Ho\u1EB7c d\xF9ng flag --skip-team-pack`),u}if(e)return await Kt(E,e,t),{pinnedTag:e};if(i)return await Se(E,Ho,t),{pinnedTag:`${Ho} (HEAD)`};let a=Go(t,E),c=await K(a),l=W(c);return l&&await Kt(E,l,t),{pinnedTag:l}}async function Ge(t){let e=Go(t,E),n=await Ae(e);return n||(await Pt(e)).slice(0,7)}import{basename as Gu,join as Uu,resolve as qe}from"path";import{input as ne,select as Du}from"@inquirer/prompts";import{spawnSync as Do}from"child_process";import{select as zl}from"@inquirer/prompts";function ql(t){let e=t.toLowerCase();return e.includes("permission denied (publickey)")||e.includes("publickey)")||e.includes("ssh: could not resolve")||e.includes("host key verification failed")}function Jl(){r.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Do("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&r.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Yl(){r.info("M\u1EDF trang GitHub Settings \u2192 SSH Keys..."),Do("open",["https://github.com/settings/keys"],{stdio:"ignore"}).status!==0&&r.info("URL: https://github.com/settings/keys")}async function Xl(){return await zl({message:"SSH permission denied. C\xE1ch x\u1EED l\xFD?",choices:[{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"D\xF9ng HTTPS thay SSH (override URL b\u1EB1ng env AVATAR_TEAM_PACK_REPO_URL)",value:"https"},{name:"T\xF4i v\u1EEBa add SSH key l\xEAn GitHub \u2014 retry",value:"retry"},{name:"B\u1ECF qua team-ai-pack (d\xF9ng avatar sync sau)",value:"skip"},{name:"T\u1EA1m ng\u01B0ng init \u2014 fix SSH key tay r\u1ED3i ch\u1EA1y l\u1EA1i",value:"abort"}]})}async function Ue(t,e,n,i=!1){for(;;)try{return{pinnedTag:(await Uo(t,e,n,i)).pinnedTag,skipped:!1}}catch(o){if(o instanceof pt)throw o;let s=o instanceof Error?o.message:String(o);if(ql(s)){r.warn("Pull team-ai-pack th\u1EA5t b\u1EA1i: SSH permission denied (publickey)."),r.dim(" \u2192 M\xE1y n\xE0y ch\u01B0a c\xF3 SSH key \u0111\u01B0\u1EE3c register tr\xEAn GitHub, ho\u1EB7c key thu\u1ED9c account kh\xE1c.");let c=await Xl();if(c==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc pull team-ai-pack. Fix SSH key (https://github.com/settings/keys) r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.");if(c==="skip")return r.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng kh\xF4ng c\xF3 shared knowledge. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};if(c==="switch"){Jl();continue}if(c==="https"){process.env.AVATAR_TEAM_PACK_REPO_URL="https://github.com/nalvn/team-ai-pack.git",r.info("Override URL sang HTTPS. L\u01B0u \xFD: HTTPS c\xF3 th\u1EC3 fail 403 n\u1EBFu read-only role."),Yl();continue}continue}let a=await R({taskName:"Pull team-ai-pack submodule",reason:s,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. N\u1EBFu skip, d\xF9ng `avatar sync` sau \u0111\u1EC3 pull pack."});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc pull team-ai-pack.");if(a==="skip")return r.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng kh\xF4ng c\xF3 shared knowledge. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0}}}k();import{spawnSync as Zl}from"child_process";function De(){let t=Zl("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as Ql}from"child_process";function tu(t){let e=nt();return Ql(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function Vo(){let t=nt(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(tu(n))return n;return null}import{spawnSync as eu}from"child_process";var nu={brew:{cmd:"brew",args:["install","gh"]},apt:{cmd:"sudo",args:["apt-get","install","-y","gh"]},dnf:{cmd:"sudo",args:["dnf","install","-y","gh"]},pacman:{cmd:"sudo",args:["pacman","-S","--noconfirm","github-cli"]},winget:{cmd:"winget",args:["install","--id","GitHub.cli","-e","--silent"]}};function Ko(t){let e=nu[t];r.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=eu(e.cmd,e.args,{stdio:"inherit"});if(n.status!==0)throw new Error(`C\xE0i gh CLI th\u1EA5t b\u1EA1i qua ${t} (exit ${n.status}). C\xE0i tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);r.success("\u0110\xE3 c\xE0i gh CLI")}import{spawnSync as iu}from"child_process";function Bo(){if(iu("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){r.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}r.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}import{spawnSync as ou}from"child_process";function Fo(){r.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=ou("gh",["auth","login","--hostname","github.com","--web","--git-protocol","ssh"],{stdio:"inherit"});if(t.status!==0)throw new Error(`gh auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'gh auth login' tay.`);r.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}async function mt(t){for(;De()==="not-installed";){r.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=Vo();if(!e){if(await R({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{Ko(e)}catch(n){if(await R({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;De()==="not-authenticated";){r.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{Fo()}catch(e){if(await R({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(De()!=="authenticated"&&await R({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(r.success("gh CLI s\u1EB5n s\xE0ng"),Bo(),t){let e=Et(t);return e.ok?(r.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await je({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}function jn(t){if(t.preserveUncommitted)return"stash";if(!t.bootstrapStrategy)return;let e=["stash","commit-all","skip","branch"];if(e.includes(t.bootstrapStrategy))return t.bootstrapStrategy;throw new Error(`--bootstrap-strategy kh\xF4ng h\u1EE3p l\u1EC7: ${t.bootstrapStrategy}. Ch\u1ECDn: ${e.join(" | ")}`)}import{readFileSync as ru}from"fs";import{dirname as su,resolve as au}from"path";import{fileURLToPath as cu}from"url";var Rt=null;function H(){if(Rt!==null)return Rt;let t=su(cu(import.meta.url));for(let e=0;e<5;e++){let n=au(t,...Array(e).fill(".."),"package.json");try{let i=ru(n,"utf8"),o=JSON.parse(i);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return Rt=o.version,Rt}catch{}}return Rt="unknown",Rt}function Wo(t){let i=(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"");return i?`avatar-${i.replace(/^avatar-/,"")}-workspace`:"avatar-client-workspace"}function lu(t){return t?`
|
|
72
|
+
`)}import{readdirSync as Bh}from"fs";import{select as zh}from"@inquirer/prompts";import{simpleGit as qh}from"simple-git";import{existsSync as dh,statSync as mh}from"fs";import{join as fh}from"path";import{simpleGit as kh}from"simple-git";import{existsSync as bh}from"fs";import{join as Ah}from"path";import{readFileSync as Ch}from"fs";import{dirname as wl,join as ke}from"path";import{fileURLToPath as kl}from"url";var ye=wl(kl(import.meta.url)),$h=[ke(ye,"templates","gitignore"),ke(ye,"..","templates","gitignore"),ke(ye,"..","..","src","templates","gitignore"),ke(ye,"..","src","templates","gitignore")],An="# === avatar ===",ve="# === /avatar ===";f();import{existsSync as Ih,readFileSync as Oh,writeFileSync as Nh}from"fs";import{join as jh}from"path";var be=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as Do}from"path";import{join as jo}from"path";var yl=".avatar-pack-manifest.json",vl=".pack-version";function Lo(t){return jo(t,yl)}async function xe(t,e){await b(Lo(t),e)}async function bl(t){let e=Lo(t);if(!await d(e))return null;try{return await k(e)}catch{return null}}async function Ae(t){let e=await bl(t);if(e?.version)return e.version;let n=jo(t,vl);if(await d(n)){let r=(await N(n)).trim();if(r)return r}return null}import{promises as Se}from"fs";import{join as Go}from"path";var xl="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack";function Al(){return process.env.AVATAR_GET_PACK_ENDPOINT??xl}var ct=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},lt=class extends Error{constructor(e){super(e),this.name="PackNetworkError"}},at=class extends Error{constructor(e){super(e),this.name="PackParseError"}};async function Pe(t,e){let n;try{n=await fetch(Al(),{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({id_token:t,version:e})})}catch(o){throw new lt(`Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Supabase get-pack: ${o instanceof Error?o.message:o}`)}if(n.status===401||n.status===403){let o=await Uo(n);throw new ct(`Supabase t\u1EEB ch\u1ED1i (${n.status}): ${o}. Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn.`)}if(n.status>=500)throw new lt(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let o=await Uo(n);throw new at(`get-pack tr\u1EA3 ${n.status}: ${o}`)}let r;try{r=await n.json()}catch{throw new at("get-pack tr\u1EA3 response kh\xF4ng ph\u1EA3i JSON.")}if(r.error)throw new at(`get-pack b\xE1o l\u1ED7i: ${r.error}`);if(!r.url||!r.object)throw new at("get-pack thi\u1EBFu url/object trong response.");return{url:r.url,object:r.object,expiresIn:r.expires_in??0,requestedBy:r.requested_by??""}}async function Ce(t,e){let n=Go(e,"..");await A(n);let r=Go(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await Ln(t,r)}catch(c){throw new lt(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await Ht(o);try{await Gn(r,o)}catch(c){throw new at(`Gi\u1EA3i n\xE9n tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}let s=`${e}.old-${process.pid}`,a=await d(e);a&&await Se.rename(e,s);try{await Se.rename(o,e)}catch(c){throw a&&await Se.rename(s,e).catch(()=>{}),c}a&&await Ht(s).catch(()=>{})}finally{await Se.rm(r,{force:!0}).catch(()=>{}),await Ht(o).catch(()=>{})}}async function Uo(t){try{let e=await t.json();return e.error??JSON.stringify(e).slice(0,200)}catch{return(await t.text().catch(()=>"")).slice(0,200)}}var B=".claude/pack",vt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function Ho(t,e){let n=await Qt(),r=await Pe(n,e),o=Do(t,B);await Ce(r.url,o);let s=e??Sl(r.object);return await xe(o,{version:s,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:s}}function Sl(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function Ee(t){let e=Do(t,B);return await Ae(e)??"(unknown)"}f();import{join as $e,relative as Tl,resolve as Bo}from"path";import{input as zo,select as $l}from"@inquirer/prompts";import Rl from"boxen";f();import Pl from"boxen";var Ko=[{cmd:"/avatar:help",desc:"H\u01B0\u1EDBng d\u1EABn s\u1EED d\u1EE5ng Avatar \u2014 ch\u1EC9 c\u1EA7n g\xF5 t\u1EF1 nhi\xEAn"},{cmd:"/avatar:brainstorm",desc:"Brainstorm \xFD t\u01B0\u1EDFng cho m\u1ED9t tool"},{cmd:"/avatar:plan",desc:"T\u1EA1o plan th\xF4ng minh v\u1EDBi prompt enhancement"},{cmd:"/avatar:scout",desc:"T\xECm file li\xEAn quan trong codebase, ti\u1EBFt ki\u1EC7m token"},{cmd:"/avatar:implement",desc:"B\u1EAFt \u0111\u1EA7u code & test theo plan c\xF3 s\u1EB5n"},{cmd:"/avatar:build-full-flow",desc:"Implement m\u1ED9t tool t\u1EEBng b\u01B0\u1EDBc (end-to-end)"},{cmd:"/avatar:fix",desc:"Ph\xE2n t\xEDch v\xE0 fix v\u1EA5n \u0111\u1EC1 t\u1EF1 \u0111i\u1EC1u h\u01B0\u1EDBng"},{cmd:"/avatar:debug",desc:"Debug v\u1EA5n \u0111\u1EC1 k\u1EF9 thu\u1EADt + \u0111\u01B0a ra gi\u1EA3i ph\xE1p"},{cmd:"/avatar:test",desc:"Ch\u1EA1y test tr\xEAn m\xE1y + ph\xE2n t\xEDch b\xE1o c\xE1o t\u1ED5ng h\u1EE3p"},{cmd:"/avatar:design:good",desc:"T\u1EA1o thi\u1EBFt k\u1EBF ch\u1EC9n chu, s\u1ED1ng \u0111\u1ED9ng"},{cmd:"/avatar:docs:init",desc:"Ph\xE2n t\xEDch codebase + t\u1EA1o t\xE0i li\u1EC7u kh\u1EDFi \u0111\u1EA7u"},{cmd:"/avatar:status",desc:"Xem l\u1EA1i thay \u0111\u1ED5i g\u1EA7n \u0111\xE2y + t\u1ED5ng k\u1EBFt c\xF4ng vi\u1EC7c"},{cmd:"/avatar:journal",desc:"Ghi nh\u1EADt k\xFD session"}];function Vo(){let t=Math.max(...Ko.map(a=>a.cmd.length)),e=p.bold("\u{1F3AF} Slash commands t\u1EEB team-ai-pack"),n=p.dim("G\xF5 trong Claude Code session \u0111\u1EC3 g\u1ECDi capability c\u1EE7a pack:"),r=Ko.map(a=>` ${p.cyan(a.cmd.padEnd(t))} ${p.dim(a.desc)}`),o=p.dim("Catalog \u0111\u1EA7y \u0111\u1EE7 46 commands: cat .claude/pack/scripts/commands_data.yaml"),s=[e,n,"",...r,"",o].join(`
|
|
73
|
+
`);return Pl(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}f();import{readdir as Cl}from"fs/promises";import{join as El}from"path";async function Te(t){if(!await d(t))return!0;try{return(await Cl(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function Fo(t,e,n=10){for(let r=2;r<n;r++){let o=El(t,`${e}-${r}`);if(await Te(o))return o}return null}function Wo(t,e){if(/[\/\\]/.test(e)||e==="."||e===".."||e.includes(".."))throw new Error(`T\xEAn workspace kh\xF4ng an to\xE0n (ch\u1EE9a path separator ho\u1EB7c '..'): "${e}".
|
|
74
|
+
Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=Bo($e(t,e)),r=Bo(t);if(!n.startsWith(`${r}/`)&&n!==r)throw new Error(`T\xEAn workspace "${e}" resolve ra ngo\xE0i parent dir (path traversal). Block.`)}async function Jo(t,e,n){Wo(t,e);let r=$e(t,e);if(await Te(r))return r;for(i.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await Fo(t,e);if(n&&o)return i.info(`--force: d\xF9ng ${o}`),o;let s=[];o&&s.push({name:`D\xF9ng "${o}" (suggest)`,value:"use-alt"}),s.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),s.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await $l({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await zo({message:"T\xEAn workspace m\u1EDBi:",validate:u=>{let m=u.trim();return m.length===0?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng":/[\/\\]/.test(m)||m==="."||m===".."||m.includes("..")?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..' (path traversal)":!0}});Wo(t,c.trim());let l=$e(t,c.trim());if(await Te(l))return l;i.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function qo(t){return await zo({message:"Team owner email:",default:t})}async function Yo(t){try{await A(t)}catch(e){let n=e;throw n.code==="EACCES"||n.code==="EPERM"?new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": kh\xF4ng c\xF3 quy\u1EC1n ghi.`):n.code==="ENAMETOOLONG"?new Error(`\u0110\u01B0\u1EDDng d\u1EABn workspace qu\xE1 d\xE0i: "${t}".`):n.code==="ENOSPC"?new Error("H\u1EBFt dung l\u01B0\u1EE3ng \u0111\u0129a, kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace."):new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": ${n.message}`)}}function _l(t){if(t===null)return` ${p.yellow("AI:")} skipped \xB7 ${p.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${p.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${p.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${p.cyan("avatar ai setup")}`}function Il(t){if(t===null)return` ${p.yellow("GitNexus:")} skipped \xB7 ${p.cyan("avatar gitnexus install")} \u0111\u1EC3 setup sau`;if(t.ok){let e=["ready"];return t.analyzed&&e.push("indexed"),t.wikiGenerated&&e.push("wiki"),t.mcpRegistered&&e.push("mcp"),` ${p.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${p.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${p.cyan("avatar gitnexus install")}`}async function Xo(t,e=null,n=null){let r=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Tl(process.cwd(),t)||t}`,_l(e),Il(n),"",` ${p.cyan(`cd ${t}`)}`,` ${p.cyan("avatar add repo")} Th\xEAm repo code v\xE0o src/`,` ${p.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${p.cyan("avatar sync")} C\u1EADp nh\u1EADt team-ai-pack m\u1EDBi`,` ${p.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${Rl(r.join(`
|
|
75
|
+
`),{padding:1,borderStyle:"round"})}
|
|
76
|
+
`);let o=$e(t,B);await d(o)&&process.stdout.write(`
|
|
77
|
+
${Vo()}
|
|
78
|
+
`)}import Ol from"boxen";import Nl from"open";Zt();f();function Zo(t){t.command("login").description("\u0110\u0103ng nh\u1EADp Google SSO (workspace @nal.vn)").option("--reset","X\xF3a credential c\u0169 v\xE0 \u0111\u0103ng nh\u1EADp l\u1EA1i").action(async e=>{try{await Sn(e)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Sn(t){if(yt({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await Ir(),await v("login_reset");else{let g=await K();if(g&&!et(g)){i.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=mt("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await Be(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=Xe(n),o=[`1. Truy c\u1EADp: ${p.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${p.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${p.green("Allow")}...`].join(`
|
|
79
|
+
`);process.stdout.write(`${Ol(o,{padding:1,borderStyle:"round"})}
|
|
80
|
+
`),Nl(r).catch(()=>{i.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=mt("\u0110ang ch\u1EDD x\xE1c nh\u1EADn trong browser..."),a=n.interval*1e3,c=Date.now()+n.expires_in*1e3,l=null;for(;Date.now()<c;){await Ml(a);try{if(l=await We(n.device_code),l)break}catch(g){throw s.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(s.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),s.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let u=ze(l.id_token);try{Je(u)}catch(g){throw await Ye(l.access_token),g}let m=qe(l,u);await en(m),await v("login",m.email),i.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${m.email}`),i.success(`Verify hosted domain: ${u.hd} \u2713`),i.success(`L\u01B0u credential v\xE0o ${tt} (chmod 600)`)}function Ml(t){return new Promise(e=>setTimeout(e,t))}import{join as Ie}from"path";f();var Qo=3;async function ti(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await Ho(t,e)).pinnedTag,skipped:!1}}catch(s){if(s instanceof vt)throw s;if(s instanceof ct){i.error(s.message);let l=await _({taskName:"T\u1EA3i team-ai-pack (x\xE1c th\u1EF1c NAL)",reason:s.message,allowSkip:!0,hint:"Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn r\u1ED3i th\u1EED l\u1EA1i. Ho\u1EB7c skip + 'avatar sync' sau."});if(l==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc x\xE1c th\u1EF1c t\u1EA3i pack.");if(l==="skip")return i.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};continue}if(s instanceof lt&&o<Qo){o+=1,i.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${Qo}): ${s.message}`),await jl(1e3*o);continue}let a=s instanceof Error?s.message:String(s),c=await _({taskName:"T\u1EA3i team-ai-pack tarball",reason:a,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. Skip \u2192 d\xF9ng `avatar sync` sau."});if(c==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA3i team-ai-pack.");if(c==="skip")return i.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};o=0}}function jl(t){return new Promise(e=>setTimeout(e,t))}import{confirm as Hl}from"@inquirer/prompts";import{promises as Ll}from"fs";import{join as Pn}from"path";var Gl=[".claude","pack","tools"];function ei(t){return Pn(t,...Gl)}async function Cn(t){let e=ei(t);if(!await d(e))return[];let n=await Ll.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await d(Pn(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function ni(t){let e=Pn(ei(t),"defaults.json");if(!await d(e))return[];try{return(await k(e)).defaultTools??[]}catch{return[]}}async function jt(t){let[e,n]=await Promise.all([Cn(t),kt(t)]);return{available:e,enabled:n}}f();import{spawnSync as Ul}from"child_process";f();function Dl(t){if(!t)return;let e=Ul(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&i.warn(`Tool y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function ri(t,e){switch(e.action){case"enabled":i.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&i.dim(` Backup: ${e.backupPath}`);break;case"disabled":i.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&i.dim(` Backup: ${e.backupPath}`);break;case"no-change":i.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function ut(t,e,n={}){let r=await It(t,e);if(!r)return n.silent||i.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;Dl(r.requires?.runtime);let o=await me(t,r);return ri(e,o),await ge(t,e,{enabled:!0,version:r.version}),!0}async function En(t,e){let n=await It(t,e);if(!n){i.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let s=(await wt(t)).tools[e]?.version??"unknown";return await ge(t,e,{enabled:!1,version:s}),!1}let r=await ho(t,n);return ri(e,r),await ge(t,e,{enabled:!1,version:n.version}),!0}var Kl={"prompt-scoring":"prompt scoring"};async function oi(t,e={}){let n=await ni(t);if(n.length!==0)for(let r of n){let o=Kl[r]??r,s;if(e.autoYes?s=!0:s=await Hl({message:`Do you want to setup ${o} (y/n)`,default:!0}),!s){i.dim(` Skip tool '${r}' (user opt-out).`);continue}await ut(t,r,{silent:!0})}}import{promises as Re}from"fs";import{dirname as Tn,join as si,relative as $n}from"path";import{promises as Vl}from"fs";function Fl(){let t=new Date,e=n=>n.toString().padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}-${e(t.getHours())}${e(t.getMinutes())}`}async function ii(t){let e=`${t}.backup-${Fl()}`;return await Vl.rename(t,e),e}var Rn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function Bl(t){try{return(await Re.lstat(t)).isSymbolicLink()}catch{return!1}}async function Wl(t,e,n){let r=$n(Tn(e),e)||e;if(!await d(t))return{dir:r,action:"source-missing"};if(await d(e))if(await Bl(e))await Re.unlink(e);else if(n){let s=await ii(e),a=$n(Tn(e),t);return await Re.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:s}}else return{dir:r,action:"skipped-conflict"};let o=$n(Tn(e),t);return await Re.symlink(o,e),{dir:r,action:"created"}}async function _e(t,e,n){let r=[];for(let o of Rn){let s=si(t,o),a=si(e,o);r.push(await Wl(s,a,n))}return r}f();import{readFileSync as zl}from"fs";import{dirname as Jl,resolve as ql}from"path";import{fileURLToPath as Yl}from"url";var bt=null;function L(){if(bt!==null)return bt;let t=Jl(Yl(import.meta.url));for(let e=0;e<5;e++){let n=ql(t,...Array(e).fill(".."),"package.json");try{let r=zl(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return bt=o.version,bt}catch{}}return bt="unknown",bt}function Xl(t){return t?`
|
|
93
81
|
### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
|
|
94
82
|
|
|
95
83
|
Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
|
|
@@ -107,7 +95,7 @@ cho Claude Code (impact analysis, call chains, blast radius).
|
|
|
107
95
|
**Folders Claude scan auto cho skills:**
|
|
108
96
|
|
|
109
97
|
- \`~/.claude/skills/gitnexus-*/\` \u2014 GitNexus global skills (exploring, debugging, ...)
|
|
110
|
-
- \`.claude/pack/skills/\` \u2014 NAL team-shared skills (qua team-ai-pack
|
|
98
|
+
- \`.claude/pack/skills/\` \u2014 NAL team-shared skills (qua team-ai-pack)
|
|
111
99
|
- C\u1EA3 2 \u0111\u1EC1u \u0111\u01B0\u1EE3c scan, kh\xF4ng xung \u0111\u1ED9t (different naming prefix)
|
|
112
100
|
|
|
113
101
|
**Manual wiki update:**
|
|
@@ -117,47 +105,32 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
|
|
|
117
105
|
\`\`\`bash
|
|
118
106
|
gitnexus wiki . --api-key <key> --base-url <url>
|
|
119
107
|
\`\`\`
|
|
120
|
-
`:""}function
|
|
121
|
-
`);return uu(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}k();import{readdir as pu}from"fs/promises";import{join as mu}from"path";async function Ve(t){if(!await d(t))return!0;try{return(await pu(t)).filter(i=>!i.startsWith(".")&&i!=="Thumbs.db").length===0}catch{return!1}}async function Jo(t,e,n=10){for(let i=2;i<n;i++){let o=mu(t,`${e}-${i}`);if(await Ve(o))return o}return null}function Xo(t,e){if(/[\/\\]/.test(e)||e==="."||e===".."||e.includes(".."))throw new Error(`T\xEAn workspace kh\xF4ng an to\xE0n (ch\u1EE9a path separator ho\u1EB7c '..'): "${e}".
|
|
122
|
-
Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=Yo(Ke(t,e)),i=Yo(t);if(!n.startsWith(`${i}/`)&&n!==i)throw new Error(`T\xEAn workspace "${e}" resolve ra ngo\xE0i parent dir (path traversal). Block.`)}async function te(t,e,n){Xo(t,e);let i=Ke(t,e);if(await Ve(i))return i;for(r.warn(`Workspace path "${i}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await Jo(t,e);if(n&&o)return r.info(`--force: d\xF9ng ${o}`),o;let s=[];o&&s.push({name:`D\xF9ng "${o}" (suggest)`,value:"use-alt"}),s.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),s.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await gu({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await Zo({message:"T\xEAn workspace m\u1EDBi:",validate:u=>{let m=u.trim();return m.length===0?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng":/[\/\\]/.test(m)||m==="."||m===".."||m.includes("..")?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..' (path traversal)":!0}});Xo(t,c.trim());let l=Ke(t,c.trim());if(await Ve(l))return l;r.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function Be(t){return await Zo({message:"Team owner email:",default:t})}async function Qo(t,e){if(e){r.warn("Skip commit (--no-commit). Ch\u1EA1y 'git status' + commit th\u1EE7 c\xF4ng sau.");return}let n=b(t);await n.add(["CLAUDE.md",".claude/",".gitignore",".gitmodules","notes/","scripts/"]),await n.commit("chore: initialize Avatar workspace"),r.success("\u0110\xE3 commit workspace")}function hu(t){if(t===null)return` ${p.yellow("AI:")} skipped \xB7 ${p.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${p.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${p.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${p.cyan("avatar ai setup")}`}function ku(t){if(t===null)return` ${p.yellow("GitNexus:")} skipped \xB7 ${p.cyan("avatar gitnexus install")} \u0111\u1EC3 setup sau`;if(t.ok){let e=["ready"];return t.analyzed&&e.push("indexed"),t.wikiGenerated&&e.push("wiki"),t.mcpRegistered&&e.push("mcp"),` ${p.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${p.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${p.cyan("avatar gitnexus install")}`}async function tr(t,e,n=null,i=null){let o=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${du(process.cwd(),t)||t}`,` ${p.dim(`(flow: ${e})`)}`,hu(n),ku(i),"",` ${p.cyan(`cd ${t}`)}`,` ${p.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${p.cyan("avatar commit src")} Commit code l\xEAn client remote`,` ${p.cyan("avatar sync")} Pull team-ai-pack m\u1EDBi`,` ${p.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${fu(o.join(`
|
|
123
|
-
`),{padding:1,borderStyle:"round"})}
|
|
124
|
-
`);let s=Ke(t,E);await d(s)&&process.stdout.write(`
|
|
125
|
-
${qo()}
|
|
126
|
-
`)}import{join as Ot}from"path";import{basename as Mu}from"path";import{confirm as pr,input as mr,select as Bn}from"@inquirer/prompts";import{spawnSync as dt}from"child_process";var _t=class extends Error{reason;fullName;stderr;constructor(e,n,i,o){super(i),this.name="CreateWorkspaceRemoteError",this.reason=e,this.fullName=n,this.stderr=o}};function wu(t){let e=t.toLowerCase();return e.includes("name already exists")||e.includes("already exists on this account")||e.includes("repository already exists")?"repo-exists":e.includes("403")||e.includes("permission")||e.includes("not authorized")||e.includes("forbidden")?"no-permission":e.includes("invalid")&&e.includes("name")?"name-invalid":e.includes("could not resolve")||e.includes("network")||e.includes("connection refused")?"network":"unknown"}function yu(t){return dt("gh",["repo","view",t,"--json","name"],{stdio:"ignore"}).status===0}function vu(t,e){return t.toLowerCase()===e.toLowerCase()?{ok:!0}:dt("gh",["api",`orgs/${t}/members/${e}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!0}:dt("gh",["api",`orgs/${t}`,"--silent"],{stdio:"ignore"}).status!==0?dt("gh",["api",`users/${t}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!1,reason:`'${t}' l\xE0 personal account kh\xE1c (kh\xF4ng ph\u1EA3i org). B\u1EA1n (${e}) kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi account c\u1EE7a user kh\xE1c. Pass --repo-org=${e} ho\u1EB7c switch gh: gh auth login`}:{ok:!1,reason:`'${t}' kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn GitHub. Check ch\xEDnh t\u1EA3 ho\u1EB7c t\u1EA1o org tr\u01B0\u1EDBc.`}:{ok:!1,reason:`'${e}' kh\xF4ng ph\u1EA3i member c\u1EE7a org '${t}'. Li\xEAn h\u1EC7 admin org \u0111\u1EC3 \u0111\u01B0\u1EE3c invite, ho\u1EB7c pass --repo-org=${e} t\u1EA1o d\u01B0\u1EDBi personal account.`}}async function er(t){Oe(t.workspaceName),Ne(t.visibility),await mt();let e=Ie(),n=t.org??e,i=vu(n,e);if(!i.ok)throw new Error(`Kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi '${n}/': ${i.reason}`);let o=`${n}/${t.workspaceName}`;if(yu(o))throw new _t("repo-exists",o,`Repo '${o}' \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. C\xF3 th\u1EC3 b\u1EA1n \u0111\xE3 init workspace n\xE0y tr\u01B0\u1EDBc \u0111\xF3.`);r.info(`T\u1EA1o GitHub repo cho workspace: ${o} (${t.visibility})...`);let s=dt("gh",["repo","create",o,`--${t.visibility}`,"--source",t.workspacePath,"--remote","origin","--push"],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(s.status!==0){let l=(s.stderr||"").trim(),u=(s.stdout||"").trim(),m=[l,u].filter(Boolean).join(`
|
|
127
|
-
`),g=wu(m);throw m&&process.stderr.write(`
|
|
128
|
-
${m}
|
|
129
|
-
|
|
130
|
-
`),new _t(g,o,`T\u1EA1o workspace remote th\u1EA5t b\u1EA1i (exit ${s.status}): ${g}.`,m)}let a=`git@github.com:${o}.git`,c=`https://github.com/${o}.git`;return r.success(`Workspace remote: ${a}`),{sshUrl:a,httpsUrl:c}}function nr(t){let e=`git@github.com:${t.fullName}.git`,n=`https://github.com/${t.fullName}.git`;return dt("git",["-C",t.workspacePath,"remote","add","origin",e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]}).status!==0&&dt("git",["-C",t.workspacePath,"remote","set-url","origin",e],{stdio:"ignore"}),r.success(`Linked existing remote: ${e}`),{sshUrl:e,httpsUrl:n}}k();import{confirm as Cu}from"@inquirer/prompts";import{promises as bu}from"fs";import{join as Ln}from"path";k();var xu=[".claude","pack","tools"];function ir(t){return Ln(t,...xu)}async function Hn(t){let e=ir(t);if(!await d(e))return[];let n=await bu.readdir(e,{withFileTypes:!0}),i=[];for(let o of n)o.isDirectory()&&await d(Ln(e,o.name,"tool.json"))&&i.push(o.name);return i.sort()}async function or(t){let e=Ln(ir(t),"defaults.json");if(!await d(e))return[];try{return(await y(e)).defaultTools??[]}catch{return[]}}async function ee(t){let[e,n]=await Promise.all([Hn(t),At(t)]);return{available:e,enabled:n}}import{spawnSync as Su}from"child_process";function Au(t){if(!t)return;let e=Su(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&r.warn(`Tool y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function rr(t,e){switch(e.action){case"enabled":r.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&r.dim(` Backup: ${e.backupPath}`);break;case"disabled":r.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&r.dim(` Backup: ${e.backupPath}`);break;case"no-change":r.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function gt(t,e,n={}){let i=await Vt(t,e);if(!i)return n.silent||r.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;Au(i.requires?.runtime);let o=await be(t,i);return rr(e,o),await xe(t,e,{enabled:!0,version:i.version}),!0}async function Gn(t,e){let n=await Vt(t,e);if(!n){r.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let s=(await St(t)).tools[e]?.version??"unknown";return await xe(t,e,{enabled:!1,version:s}),!1}let i=await Di(t,n);return rr(e,i),await xe(t,e,{enabled:!1,version:n.version}),!0}var Pu={"prompt-scoring":"prompt scoring"};async function sr(t,e={}){let n=await or(t);if(n.length!==0)for(let i of n){let o=Pu[i]??i,s;if(e.autoYes?s=!0:s=await Cu({message:`Do you want to setup ${o} (y/n)`,default:!0}),!s){r.dim(` Skip tool '${i}' (user opt-out).`);continue}await gt(t,i,{silent:!0})}}import{promises as Fe}from"fs";import{dirname as Un,join as cr,relative as Dn}from"path";import{promises as $u}from"fs";function Tu(){let t=new Date,e=n=>n.toString().padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}-${e(t.getHours())}${e(t.getMinutes())}`}async function ar(t){let e=`${t}.backup-${Tu()}`;return await $u.rename(t,e),e}k();var Vn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function Eu(t){try{return(await Fe.lstat(t)).isSymbolicLink()}catch{return!1}}async function Ru(t,e,n){let i=Dn(Un(e),e)||e;if(!await d(t))return{dir:i,action:"source-missing"};if(await d(e))if(await Eu(e))await Fe.unlink(e);else if(n){let s=await ar(e),a=Dn(Un(e),t);return await Fe.symlink(a,e),{dir:i,action:"backed-up-and-linked",backupPath:s}}else return{dir:i,action:"skipped-conflict"};let o=Dn(Un(e),t);return await Fe.symlink(o,e),{dir:i,action:"created"}}async function We(t,e,n){let i=[];for(let o of Vn){let s=cr(t,o),a=cr(e,o);i.push(await Ru(s,a,n))}return i}async function dr(t,e){let i=(await b(t).getRemotes(!0)).find(l=>l.name==="origin");if(i?.refs.push)return r.success(`Folder \u0111\xE3 c\xF3 remote origin: ${i.refs.push}`),i.refs.push;if(!(e.createRemote??await pr({message:"Folder ch\u01B0a c\xF3 remote. T\u1EA1o GitHub repo ngay \u0111\u1EC3 share team?",default:!0}))){r.warn("Ti\u1EBFp t\u1EE5c v\u1EDBi local path. Workspace ch\u1EC9 ch\u1EA1y \u0111\u01B0\u1EE3c tr\xEAn m\xE1y b\u1EA1n.");return}await mt();let s=e.repoVisibility??await Bn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),a=await mr({message:"T\xEAn repo:",default:Mu(t)});return Tt({folder:t,name:a,visibility:s,org:e.repoOrg}).sshUrl}async function Fn(t){await T(t.workspacePath),await b(t.workspacePath).init();let e=U(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await b(t.workspacePath).subModule(["add",t.srcRemoteUrl,"src"]);let n="HEAD";t.skipTeamPack?e.succeed("Skip team-ai-pack (--skip-team-pack)"):(e.stop(),n=(await Ue(t.workspacePath,t.packVersion,t.ssoEmail,t.packLatest===!0&&!t.packVersion)).pinnedTag??"HEAD",e.succeed(`Pin team-ai-pack v\xE0o ${n}`)),await Wn({workspacePath:t.workspacePath,workspaceName:t.workspaceName,teamOwner:t.teamOwner,description:t.description,packVersion:n,autoYes:t.autoYes,skipCommit:t.skipCommit,createWorkspaceRemote:t.createWorkspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:t.flow,aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip})}catch(n){throw e.fail("Init workspace th\u1EA5t b\u1EA1i"),n}}async function Wn(t){let e=Mn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client"});await Zi(t.workspacePath),await Qi(t.workspacePath,e),await wn(t.workspacePath,e),await to(t.workspacePath,e),await eo(t.workspacePath),await T(Ot(t.workspacePath,"notes")),await T(Ot(t.workspacePath,"scripts")),await Ft(Ot(t.workspacePath,".git"),"post-merge"),await Ft(Ot(t.workspacePath,".git","modules","src"),"pre-push"),r.success("C\xE0i post-merge (workspace) + pre-push (src/)"),await Lu(t.workspacePath,t.autoYes),await v("init",`flow=${t.flow},workspace=${t.workspaceName}`),await Qo(t.workspacePath,t.skipCommit),await Hu(t);let n=null;t.aiSkip?r.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau qua: avatar ai setup"):n=await he({workspacePath:t.workspacePath});let i=null;if(t.aiSkip||t.gitnexusSkip?t.gitnexusSkip?r.dim("B\u1ECF qua GitNexus setup (--gitnexus-skip). Setup sau: avatar gitnexus install"):r.dim("B\u1ECF qua GitNexus setup (auto-skip do --ai-skip)."):i=await Ee({workspacePath:t.workspacePath}),i?.ok){let s=Mn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client",gitnexusReady:!0});await wn(t.workspacePath,s),r.dim("Updated CLAUDE.md v\u1EDBi GitNexus section")}await tr(t.workspacePath,t.flow,n,i)}async function Lu(t,e){let n=Ot(t,E);if(!await d(n)){r.dim("Pack submodule kh\xF4ng t\u1ED3n t\u1EA1i (skip auto-sync). C\xF3 th\u1EC3 ch\u1EA1y `avatar sync` sau.");return}let i=Ot(t,".claude");r.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings merge)...");try{let o=await We(n,i,!1),s=o.filter(l=>l.action==="created"||l.action==="updated").length,a=o.filter(l=>l.action==="source-missing").length;r.success(` \u2713 Symlinks: ${s} created${a>0?`, ${a} source-missing (pack thi\u1EBFu dir)`:""}`);let c=await ye(t);switch(c.action){case"merged":r.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":r.dim(" - settings.json \u0111\xE3 sync, kh\xF4ng c\u1EA7n thay \u0111\u1ED5i.");break;case"no-pack-template":r.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}await sr(t,{autoYes:e});try{let{detectPlaintextSecretsInSettings:l,migrateSecretsFromSettingsToEnvrc:u}=await Promise.resolve().then(()=>(Kn(),ur)),m=await l(t);m.keys.length>0&&(r.info(`Auto-migrating ${m.keys.length} plaintext secrets t\u1EEB settings.json \u2192 .envrc...`),await u(t))}catch(l){r.warn(`Auto-migrate secrets fail: ${l instanceof Error?l.message:l}.`)}}catch(o){r.warn(`Auto-sync pack fail: ${o instanceof Error?o.message:o}. Ch\u1EA1y \`avatar sync\` th\u1EE7 c\xF4ng \u0111\u1EC3 retry.`)}}async function Hu(t){if(t.skipCommit){r.dim("Skip workspace remote (ch\u01B0a commit). Setup sau qua: gh repo create ...");return}let e=t.createWorkspaceRemote;if(e===void 0){if(t.autoYes)return;e=await pr({message:"T\u1EA1o remote GitHub cho workspace \u0111\u1EC3 share team? (Avatar state)",default:!1})}if(!e)return;let n=t.repoVisibility??(t.autoYes?"private":await Bn({message:"Workspace visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh, an to\xE0n)",value:"private"},{name:"public",value:"public"}]}));for(;;)try{await er({workspacePath:t.workspacePath,workspaceName:t.workspaceName,visibility:n,org:t.repoOrg});return}catch(i){if(i instanceof _t&&i.reason==="repo-exists"){let s=i.fullName,a=await Bn({message:`Repo '${s}' \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. C\xE1ch x\u1EED l\xFD?`,choices:[{name:"D\xF9ng remote \u0111\xE3 c\xF3 (link workspace local v\xE0o repo n\xE0y)",value:"reuse"},{name:"Nh\u1EADp t\xEAn workspace kh\xE1c (t\u1EA1o repo m\u1EDBi)",value:"rename"},{name:"B\u1ECF qua (workspace local-only)",value:"skip"},{name:"T\u1EA1m ng\u01B0ng init",value:"abort"}]});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(a==="skip"){r.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}if(a==="reuse"){nr({workspacePath:t.workspacePath,fullName:s});return}let c=await mr({message:"T\xEAn workspace m\u1EDBi (s\u1EBD t\u1EA1o repo new):",validate:l=>l.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"});t.workspaceName=c.trim();continue}let o=await R({taskName:"T\u1EA1o workspace remote tr\xEAn GitHub",reason:i instanceof Error?i.message:String(i),allowSkip:!0,hint:"Tip: sai org? Pass --repo-org=<your-gh-user>. Ho\u1EB7c switch gh: gh auth login."});if(o==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(o==="skip"){r.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}}}async function gr(t,e){let n=t.clientRepo??await ne({message:"URL git c\u1EE7a repo:",validate:m=>m.length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),{resolvedRemoteUrl:i}=await mt(n),o=i??n,s=t.teamOwner??await Be(e),a=Wo(o),c=t.workspaceName??await ne({message:"T\xEAn workspace:",default:a}),l=qe(t.workspaceParent??"."),u=await te(l,c,t.force);await Fn({workspacePath:u,workspaceName:c,srcRemoteUrl:o,teamOwner:s,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho ${o}`,packVersion:t.packVersion,packLatest:t.latest,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"existing-remote",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip,ssoEmail:e})}async function fr(t,e){let n=qe(t.folderPath??await ne({message:"\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3:",validate:u=>u.length>0?!0:"Path b\u1EAFt bu\u1ED9c"}));await Rn(n,{presetStrategy:jn(t),autoYes:t.yes});let i=await dr(n,t);if(i){let u=Et(i);u.ok||(r.warn(`Remote ${i} kh\xF4ng accessible (${u.reason??"unknown"}).`),i=(await je({url:i,initialReason:u.reason??"unknown",initialDetail:u.detail,folderPath:n,defaultVisibility:t.repoVisibility})).resolvedUrl)}let o=t.teamOwner??await Be(e),s=t.workspaceName??`${Gu(n)}-avatar-workspace`,a=t.workspaceName??await ne({message:"T\xEAn workspace:",default:s}),c=qe(t.workspaceParent??"."),l=await te(c,a,t.force);await Fn({workspacePath:l,workspaceName:a,srcRemoteUrl:i??n,teamOwner:o,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho folder ${n}`,packVersion:t.packVersion,packLatest:t.latest,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"existing-folder",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip,ssoEmail:e})}async function hr(t,e){await mt();let n=t.workspaceName??await ne({message:"T\xEAn d\u1EF1 \xE1n:",validate:m=>m.length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),i=t.repoVisibility??await Du({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),o=t.teamOwner??await Be(e),s=qe(t.workspaceParent??"."),a=await te(s,n,t.force),c=Uu(a,"src");await T(a),await T(c),await Rn(c,{autoYes:!0});let l=Tt({folder:c,name:n,visibility:i,org:t.repoOrg});await b(a).init();let u=U(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await b(a).subModule(["add",l.sshUrl,"src"]);let m="HEAD";t.skipTeamPack?u.succeed("Skip team-ai-pack (--skip-team-pack)"):(u.stop(),m=(await Ue(a,t.packVersion,e,t.latest===!0&&!t.packVersion)).pinnedTag??"HEAD",u.succeed(`Pin team-ai-pack v\xE0o ${m}`)),await Wn({workspacePath:a,workspaceName:n,teamOwner:o,description:t.description??`D\u1EF1 \xE1n m\u1EDBi: ${n}`,packVersion:m,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"new-project",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip})}catch(m){throw u.fail("Init workspace th\u1EA5t b\u1EA1i"),m}}import Yu from"boxen";import Xu from"open";var qn="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Vu="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",zn="nal.vn",Ku=["openid","email","profile"],Bu="https://oauth2.googleapis.com/device/code",Fu="https://oauth2.googleapis.com/token",Wu="https://oauth2.googleapis.com/revoke";async function kr(){let t=new URLSearchParams({client_id:qn,scope:Ku.join(" ")}),e=await fetch(Bu,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:t});if(!e.ok){let n=await e.text();throw new Error(`Device code request failed (${e.status}): ${n}`)}return await e.json()}async function wr(t){let e=new URLSearchParams({client_id:qn,client_secret:Vu,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Fu,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(n.ok)return await n.json();let i="";try{i=(await n.json()).error??""}catch{i=""}if(i==="authorization_pending"||i==="slow_down")return null;throw i==="access_denied"?new Error("User t\u1EEB ch\u1ED1i quy\u1EC1n truy c\u1EADp"):i==="expired_token"?new Error("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."):new Error(`OAuth token endpoint tr\u1EA3 l\u1ED7i: ${i||n.status}`)}function yr(t){let e=t.split(".");if(e.length!==3)throw new Error("id_token format kh\xF4ng h\u1EE3p l\u1EC7");let n=e[1];if(!n)throw new Error("id_token thi\u1EBFu payload");let i=n.replace(/-/g,"+").replace(/_/g,"/"),o=Buffer.from(i,"base64").toString("utf8");return JSON.parse(o)}var zu=new Set(["https://accounts.google.com","accounts.google.com"]),qu=60;function Ju(t){if(!zu.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==qn)throw new Error("id_token audience kh\xF4ng kh\u1EDBp client ID Avatar. Token c\xF3 th\u1EC3 \u0111\u01B0\u1EE3c sign cho app kh\xE1c.");let e=Math.floor(Date.now()/1e3);if(t.exp+qu<e){let n=e-t.exp;throw new Error(`id_token \u0111\xE3 h\u1EBFt h\u1EA1n ${n}s tr\u01B0\u1EDBc. Login l\u1EA1i \u0111\u1EC3 l\u1EA5y token m\u1EDBi.`)}if(t.hd!==zn)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${zn}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}var vr=Ju;function br(t,e){let n=new Date(Date.now()+t.expires_in*1e3).toISOString();return{email:e.email,name:e.name??e.email,access_token:t.access_token,refresh_token:t.refresh_token,expires_at:n,id_token:t.id_token}}async function xr(t){let e=new URLSearchParams({token:t});await fetch(Wu,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function Sr(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",zn),e.toString()}function Ar(t){t.command("login").description("\u0110\u0103ng nh\u1EADp Google SSO (workspace @nal.vn)").option("--reset","X\xF3a credential c\u0169 v\xE0 \u0111\u0103ng nh\u1EADp l\u1EA1i").action(async e=>{try{await Jn(e)}catch(n){r.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Jn(t){if($t({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await ei(),await v("login_reset");else{let g=await tt();if(g&&!et(g)){r.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=U("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await kr(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let i=Sr(n),o=[`1. Truy c\u1EADp: ${p.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${p.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${p.green("Allow")}...`].join(`
|
|
131
|
-
`);process.stdout.write(`${Yu(o,{padding:1,borderStyle:"round"})}
|
|
132
|
-
`),Xu(i).catch(()=>{r.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=U("\u0110ang ch\u1EDD x\xE1c nh\u1EADn trong browser..."),a=n.interval*1e3,c=Date.now()+n.expires_in*1e3,l=null;for(;Date.now()<c;){await Zu(a);try{if(l=await wr(n.device_code),l)break}catch(g){throw s.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(s.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),s.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let u=yr(l.id_token);try{vr(u)}catch(g){throw await xr(l.access_token),g}let m=br(l,u);await ti(m),await v("login",m.email),r.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${m.email}`),r.success(`Verify hosted domain: ${u.hd} \u2713`),r.success(`L\u01B0u credential v\xE0o ${Q} (chmod 600)`)}function Zu(t){return new Promise(e=>setTimeout(e,t))}function Cr(t){t.command("init").description("Kh\u1EDFi t\u1EA1o Avatar \u2014 3 flow t\u1EF1 nh\u1EADn di\u1EC7n (repo / folder / new)").option("--project-status <val>","existing-remote | existing-folder | new-project").option("--folder-path <path>","\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3 (flow existing-folder)").option("--create-remote","Force t\u1EA1o remote qua gh (flow existing-folder ho\u1EB7c new-project)").option("--repo-visibility <val>","private (m\u1EB7c \u0111\u1ECBnh) | public").option("--repo-org <name>","GitHub org/owner cho repo m\u1EDBi").option("--client-repo <url>","URL git remote (flow existing-remote)").option("--workspace-name <name>","T\xEAn workspace").option("--workspace-parent <path>","Th\u01B0 m\u1EE5c cha t\u1EA1o workspace (m\u1EB7c \u0111\u1ECBnh . \u2014 CWD)").option("--pack-version <tag>","Pin team-ai-pack v\xE0o tag c\u1EE5 th\u1EC3").option("--latest","Pull HEAD c\u1EE7a team-ai-pack main branch (b\u1ECF qua tag SemVer, bleeding-edge)").option("--team-owner <email>","Email team owner (b\u1ECF qua prompt)").option("--description <text>","M\xF4 t\u1EA3 1 d\xF2ng c\u1EE7a d\u1EF1 \xE1n").option("--skip-scan","B\u1ECF qua project-scanner sau scaffold").option("--skip-team-pack","B\u1ECF qua submodule team-ai-pack (test mode)").option("--force","B\u1ECF qua prompt khi workspace path \u0111\xE3 t\u1ED3n t\u1EA1i").option("--yes","Auto-confirm t\u1EA5t c\u1EA3 prompt").option("--no-commit","Skip commit workspace initial state (m\u1EB7c \u0111\u1ECBnh LU\xD4N commit)").option("--workspace-remote","T\u1EA1o GitHub remote cho workspace root (default: prompt)").option("--ai-skip","B\u1ECF qua phase AI setup (CI/test mode \u2014 ch\u1EA1y `avatar ai setup` sau)").option("--gitnexus-skip","B\u1ECF qua phase GitNexus setup (M10 \u2014 ch\u1EA1y `avatar gitnexus install` sau)").option("--bootstrap-strategy <s>","X\u1EED l\xFD folder dirty: stash | commit-all | skip | branch (default: prompt)").option("--preserve-uncommitted","Alias cho --bootstrap-strategy=stash (gi\u1EEF changes user)").option("--mode <mode>","[DEPRECATED] D\xF9ng --project-status thay th\u1EBF").action(async e=>{try{await tp(e)}catch(n){n instanceof Qt&&(r.dim(n.message),process.exit(0)),n instanceof pt&&(r.dim(n.message),process.exit(0)),n instanceof Jt&&(r.dim(n.message),process.exit(0)),n instanceof x&&(r.dim(n.message),process.exit(0)),r.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function tp(t){t.yes||$t({tagline:"Kh\u1EDFi t\u1EA1o Avatar trong d\u1EF1 \xE1n c\u1EE7a b\u1EA1n"}),t.mode&&r.warn("Flag --mode \u0111\xE3 deprecated t\u1EEB v1.1. D\xF9ng --project-status thay th\u1EBF.");let e=await tt();for(;!e||et(e);){r.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y login flow tr\u01B0\u1EDBc khi init...");try{await Jn({})}catch(o){r.warn(`Login fail: ${o.message}`)}if(e=await tt(),e&&!et(e))break;if(await R({taskName:"\u0110\u0103ng nh\u1EADp SSO Google",reason:"Token ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o ho\u1EB7c \u0111\xE3 h\u1EBFt h\u1EA1n.",allowSkip:!1,hint:"\u0110\u1EA3m b\u1EA3o b\u1EA1n ch\u1ECDn 'Allow' tr\xEAn browser v\xE0 d\xF9ng email @nal.vn."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc login. Ch\u1EA1y 'avatar login' tay r\u1ED3i 'avatar init' l\u1EA1i.")}switch(t.projectStatus??await ep()){case"existing-remote":await gr(t,e.email);break;case"existing-folder":await fr(t,e.email);break;case"new-project":await hr(t,e.email);break}}async function ep(){return await Qu({message:"T\xECnh tr\u1EA1ng d\u1EF1 \xE1n c\u1EE7a b\u1EA1n?",choices:[{name:"1. \u0110\xE3 c\xF3 repo git remote (URL c\xF3 s\u1EB5n)",value:"existing-remote"},{name:"2. \u0110\xE3 c\xF3 folder code local",value:"existing-folder"},{name:"3. D\u1EF1 \xE1n m\u1EDBi ho\xE0n to\xE0n",value:"new-project"}]})}function z(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
|
|
108
|
+
`:""}function ai(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:L(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:Xl(t.gitnexusReady??!1)}}async function ci(t){await Yo(t.workspacePath),await A(Ie(t.workspacePath,"src")),await A(Ie(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)i.dim("Skip team-ai-pack (--skip-team-pack).");else{mt("T\u1EA3i team-ai-pack t\u1EEB Supabase...").stop();let c=await ti(t.workspacePath,t.packVersion);e=c.pinnedTag??"(skipped)",c.skipped||i.success(`team-ai-pack: ${e}`)}let n=ai({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:e});await Bn(t.workspacePath),await Wn(t.workspacePath,n),await zn(t.workspacePath,n),await Jn(t.workspacePath,n),await qn(t.workspacePath),await Zl(t.workspacePath,t.autoYes),await v("init",`workspace=${t.workspaceName}`);let r=null;t.aiSkip?i.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau: avatar ai setup"):r=await le({workspacePath:t.workspacePath});let o=null;t.aiSkip||t.gitnexusSkip?i.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await fe({workspacePath:t.workspacePath,skipAnalyze:!0}),await Xo(t.workspacePath,r,o)}async function Zl(t,e){let n=Ie(t,B);if(!await d(n)){i.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=Ie(t,".claude");i.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");try{let o=await _e(n,r,!1),s=o.filter(l=>l.action==="created"||l.action==="updated").length,a=o.filter(l=>l.action==="source-missing").length;i.success(` \u2713 Symlinks: ${s} created${a>0?`, ${a} source-missing`:""}`);let c=await pe(t);c.action==="merged"&&i.success(` \u2713 settings.json merged (${c.changes.join("; ")})`),await oi(t,{autoYes:e})}catch(o){i.warn(`Auto-sync pack fail: ${o instanceof Error?o.message:o}. Ch\u1EA1y \`avatar sync\` \u0111\u1EC3 retry.`)}}function li(t){t.command("init").description("Kh\u1EDFi t\u1EA1o Avatar workspace (m\xF4i tr\u01B0\u1EDDng) \u2014 repo c\xE0i sau b\u1EB1ng 'avatar add repo'").argument("[workspace-name]","T\xEAn workspace (b\u1EAFt bu\u1ED9c, h\u1ECFi n\u1EBFu thi\u1EBFu)").option("--workspace-parent <path>","Th\u01B0 m\u1EE5c cha t\u1EA1o workspace (m\u1EB7c \u0111\u1ECBnh . \u2014 CWD)").option("--pack-version <tag>","Pin team-ai-pack v\xE0o tag c\u1EE5 th\u1EC3 (vd v0.2.0-beta.0)").option("--team-owner <email>","Email team owner (b\u1ECF qua prompt)").option("--description <text>","M\xF4 t\u1EA3 1 d\xF2ng c\u1EE7a workspace").option("--skip-team-pack","B\u1ECF qua t\u1EA3i team-ai-pack (test mode)").option("--force","B\u1ECF qua prompt khi workspace path \u0111\xE3 t\u1ED3n t\u1EA1i").option("--yes","Auto-confirm t\u1EA5t c\u1EA3 prompt").option("--ai-skip","B\u1ECF qua phase AI setup (ch\u1EA1y 'avatar ai setup' sau)").option("--gitnexus-skip","B\u1ECF qua phase GitNexus setup").action(async(e,n)=>{try{await eu(e,n)}catch(r){(r instanceof be||r instanceof vt||r instanceof Ct||r instanceof S)&&(i.dim(r.message),process.exit(0)),i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function eu(t,e){e.yes||yt({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await K();for(;!n||et(n);){i.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp (ho\u1EB7c token h\u1EBFt h\u1EA1n) \u2014 ch\u1EA1y login tr\u01B0\u1EDBc...");try{await Sn({})}catch(c){i.warn(`Login fail: ${c.message}`)}if(n=await K(),n&&!et(n))break;throw new S("C\u1EA7n \u0111\u0103ng nh\u1EADp @nal.vn \u0111\u1EC3 t\u1EA3i team-ai-pack. Ch\u1EA1y 'avatar login' r\u1ED3i 'avatar init' l\u1EA1i.")}let r=t??await tu({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await qo(n.email),s=Ql(e.workspaceParent??"."),a=await Jo(s,r.trim(),e.force);await ci({workspacePath:a,workspaceName:r.trim(),teamOwner:o,description:e.description??`Avatar workspace: ${r.trim()}`,packVersion:e.packVersion,autoYes:e.yes,skipTeamPack:e.skipTeamPack,aiSkip:e.aiSkip,gitnexusSkip:e.gitnexusSkip})}f();function W(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
|
|
133
109
|
`),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${p.cyan(e)}
|
|
134
110
|
`),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
|
|
135
|
-
`),process.exit(0)}}function
|
|
136
|
-
`):
|
|
111
|
+
`),process.exit(0)}}function ui(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(W("mcp-run","Milestone 09"))}import{join as nu}from"path";import pi from"boxen";f();var ru=".claude/pack";function di(t){t.command("pack").description("Qu\u1EA3n l\xFD team-ai-pack (status, ...)").command("status").description("Hi\u1EC3n th\u1ECB version pack \u0111ang c\xE0i").option("--json","Output JSON cho script").action(async n=>{try{let r=await ou(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
|
|
112
|
+
`):iu(r)}catch(r){i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function ou(t){let e=nu(t,ru);return await d(e)?{installed:!0,currentVersion:await Ee(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function iu(t){if(!t.installed){let r=[`${p.bold("team-ai-pack")} \xB7 ${p.yellow("ch\u01B0a c\xE0i")}`,"\u2500".repeat(48),p.dim("Ch\u1EA1y `avatar init` ho\u1EB7c `avatar sync` \u0111\u1EC3 c\xE0i pack.")];process.stdout.write(`${pi(r.join(`
|
|
137
113
|
`),{padding:1,borderStyle:"round"})}
|
|
138
|
-
`);return}let e=t.
|
|
114
|
+
`);return}let e=t.currentVersion??p.yellow("(unknown)"),n=[`${p.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${p.dim("Version hi\u1EC7n t\u1EA1i:")} ${e}`,"",p.dim("C\u1EADp nh\u1EADt m\u1EDBi nh\u1EA5t: avatar sync")];process.stdout.write(`${pi(n.join(`
|
|
139
115
|
`),{padding:1,borderStyle:"round"})}
|
|
140
|
-
`)}function
|
|
141
|
-
`);let e=await
|
|
142
|
-
`)
|
|
143
|
-
`).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function
|
|
116
|
+
`)}function mi(t){t.command("restore").description("Kh\xF4i ph\u1EE5c .claude/pack/ t\u1EEB backup (M08)").option("--backup <name>","T\xEAn backup folder trong .claude/_backup/").option("--list","Li\u1EC7t k\xEA c\xE1c backup hi\u1EC7n c\xF3").action(W("restore","Milestone 08"))}function gi(t){t.command("review").description("Review pending proposals t\u1EEB avatar scan (M08)").option("--accept-all","Approve m\u1ECDi pending kh\xF4ng h\u1ECFi (CI mode)").option("--reject-all","X\xF3a m\u1ECDi pending kh\xF4ng h\u1ECFi").action(W("review","Milestone 08"))}function fi(t){t.command("scan").description("Ch\u1EA1y project scanner v\xE0 \u0111\u1EC1 xu\u1EA5t knowledge update (M06)").option("--incremental","Ch\u1EC9 scan c\xE1c file thay \u0111\u1ED5i t\u1EEB commit cu\u1ED1i").option("--full","Scan to\xE0n b\u1ED9 d\u1EF1 \xE1n (default)").option("--scanners <list>","tech-stack,conventions,architecture,domain,git-pattern").option("--quiet","Ch\u1EA1y ng\u1EA7m, \xEDt output (d\xF9ng cho git hook)").action(W("scan","Milestone 06"))}import{confirm as xt,password as du}from"@inquirer/prompts";import{promises as su}from"fs";import{join as au}from"path";var cu=["ANTHROPIC_API_KEY","ANTHROPIC_AUTH_TOKEN","OPENAI_API_KEY","GEMINI_API_KEY","DEEPSEEK_API_KEY","GROQ_API_KEY","PERPLEXITY_API_KEY","ANTHROPIC_VERTEX_PROJECT_ID","GOOGLE_APPLICATION_CREDENTIALS"],lu=[".claude","settings.json"];function uu(t){return au(t,...lu)}function pu(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)}${String(e.getMonth()+1).padStart(2,"0")}${String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-secrets-migration-${n}`}async function Lt(t){let e=uu(t);if(!await d(e))return{keys:[],settingsPath:e};let n;try{n=await k(e)}catch{return{keys:[],settingsPath:e}}let r=n.env??{},o=[];for(let s of cu)typeof r[s]=="string"&&r[s].length>0&&o.push(s);return{keys:o,settingsPath:e}}async function _n(t){let{keys:e,settingsPath:n}=await Lt(t);if(e.length===0)return{migratedKeys:[],settingsJsonBackupPath:null,noOp:!0};let r=await k(n),o=r.env??{},s={};for(let c of e){let l=o[c];typeof l=="string"&&l.length>0&&(s[c]=l)}await it(t,s);let a=pu(n);await su.copyFile(n,a);for(let c of e)delete o[c];return Object.keys(o).length===0?r.env=void 0:r.env=o,await b(n,r),{migratedKeys:e,settingsJsonBackupPath:a,noOp:!1}}f();async function pt(){let t=D(process.cwd());return t||(i.error("Kh\xF4ng t\xECm th\u1EA5y Avatar workspace. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc, ho\u1EB7c cd v\xE0o workspace."),process.exit(1)),t}function hi(t){return t.length<=8?"****":`${t.slice(0,4)}...${t.slice(-4)}`}async function mu(t){i.info(`Workspace: ${t}`);let e=await ft();if(e.installed)i.success(`\u2713 direnv ${e.version??"(installed)"}`);else if(i.warn("direnv ch\u01B0a c\xE0i. Avatar secrets d\xF9ng direnv \u0111\u1EC3 auto-load .envrc khi cd v\xE0o project."),await xt({message:"C\xE0i direnv ngay? (qua brew/apt/dnf/pacman t\xF9y OS)",default:!0})){i.info("\u0110ang c\xE0i direnv...");let s=await se();s.success?i.success(`\u2713 ${s.message}`):(i.warn(`! ${s.message}`),i.info(`Manual install: ${s.manualInstructionsUrl}`),i.info("Ti\u1EBFp t\u1EE5c \u2014 .envrc s\u1EBD ghi nh\u01B0ng c\u1EA7n direnv \u0111\u1EC3 auto-load."))}if(e.shellHookInZshrc)i.dim("\u2713 direnv shell hook \u0111\xE3 trong ~/.zshrc");else if(i.warn("~/.zshrc ch\u01B0a c\xF3 direnv shell hook \u2192 .envrc s\u1EBD kh\xF4ng t\u1EF1 load."),await xt({message:'Add eval "$(direnv hook zsh)" v\xE0o ~/.zshrc?',default:!0})){let s=ae();s.modified?(i.success(`\u2713 Added direnv hook to ${s.zshrcPath}`),i.info("Reload shell: 'source ~/.zshrc' ho\u1EB7c m\u1EDF terminal m\u1EDBi")):i.dim("Skip \u2014 hook \u0111\xE3 t\u1ED3n t\u1EA1i ho\u1EB7c kh\xF4ng write \u0111\u01B0\u1EE3c")}let n=await Lt(t);if(n.keys.length>0&&(i.warn(`Ph\xE1t hi\u1EC7n ${n.keys.length} secret plaintext trong .claude/settings.json: ${n.keys.join(", ")}`),i.warn("Risk: leak n\u1EBFu file commit l\xEAn git."),await xt({message:"Migrate sang .envrc (direnv) ngay?",default:!0}))){let s=await _n(t);i.success(`\u2713 Migrated ${s.migratedKeys.length} secrets: ${s.migratedKeys.join(", ")}`),s.settingsJsonBackupPath&&i.dim(` settings.json backup: ${s.settingsJsonBackupPath}`)}F(t).has("ANTHROPIC_API_KEY")||await xt({message:"Set ANTHROPIC_API_KEY cho project n\xE0y ngay?",default:!0})&&await wi(t,"ANTHROPIC_API_KEY",{validateAnthropic:!0}),i.success("\u2713 Secrets setup done."),i.dim(` .envrc: ${ht(t)}`),i.dim(" View: avatar secrets list")}async function gu(){let t=await pt();await mu(t)}async function wi(t,e,n={}){let r=await du({message:`Nh\u1EADp value cho ${e} (input \u1EA9n):`,mask:"*",validate:s=>s.trim().length>0?!0:"Value b\u1EAFt bu\u1ED9c"});if(n.validateAnthropic&&e==="ANTHROPIC_API_KEY"){i.info(`Verify key (${ne(r)}) qua Anthropic /v1/models...`);try{let s=await re(r);i.success(`\u2713 Key valid (${s.length} models available)`)}catch(s){if(i.error(`Key kh\xF4ng valid: ${s instanceof Error?s.message:s}`),!await xt({message:"V\u1EABn ghi key (b\u1ECF qua validation)?",default:!1}))return}}let o=await it(t,{[e]:r});i.success(`\u2713 Set ${e} (.envrc: ${o.varsBefore} \u2192 ${o.varsAfter} vars)`),o.direnvAllowOk===!0?i.dim(" direnv allow ok \u2014 s\u1EBD auto-load l\u1EA7n cd k\u1EBF ti\u1EBFp"):o.direnvAllowOk===null?i.warn(" direnv missing \u2192 file ghi OK nh\u01B0ng kh\xF4ng auto-load. C\xE0i direnv ho\u1EB7c source th\u1EE7 c\xF4ng."):i.warn(" direnv allow failed \u2014 ch\u1EA1y 'direnv allow' th\u1EE7 c\xF4ng trong workspace")}async function fu(t){let e=await pt();await wi(e,t,{validateAnthropic:t==="ANTHROPIC_API_KEY"})}async function hu(t){let e=await pt(),r=F(e).get(t);r===void 0&&(i.error(`Secret '${t}' kh\xF4ng t\u1ED3n t\u1EA1i trong .envrc.`),process.exit(1)),process.stdout.isTTY?(console.log(hi(r)),i.dim(`(masked. \u0110\u1EC3 in raw: 'avatar secrets get ${t} | cat')`)):console.log(r)}async function wu(){let t=await pt(),e=F(t);if(e.size===0){i.info("Ch\u01B0a c\xF3 secret n\xE0o trong .envrc Avatar block."),i.dim("Set: avatar secrets set <NAME>");return}i.info(`${e.size} secrets in ${ht(t)}:`);for(let[n,r]of[...e.entries()].sort())console.log(` ${n.padEnd(30)} = ${hi(r)}`)}async function ku(t){let e=await pt();if(!F(e).has(t)){i.warn(`Secret '${t}' kh\xF4ng t\u1ED3n t\u1EA1i \u2014 no-op.`);return}if(!await xt({message:`X\xF3a secret '${t}' kh\u1ECFi .envrc?`,default:!1})){i.dim("H\u1EE7y.");return}let o=await it(e,{[t]:null});i.success(`\u2713 Removed ${t} (.envrc: ${o.varsBefore} \u2192 ${o.varsAfter} vars)`)}async function yu(){let t=await pt();i.info(`Workspace: ${t}
|
|
117
|
+
`);let e=await ft();i.info(`direnv installed: ${e.installed?`\u2713 ${e.version??""}`:"\u2717"}`),i.info(`direnv shell hook (~/.zshrc): ${e.shellHookInZshrc?"\u2713":"\u2717"}`);let n=ht(t),r=F(t);i.info(`.envrc: ${r.size>0?`\u2713 ${r.size} secrets at ${n}`:"\u2717 (no Avatar block)"}`);let o=await Lt(t);o.keys.length>0&&i.warn(`\u26A0 Plaintext secrets in settings.json: ${o.keys.join(", ")} \u2014 run 'avatar secrets migrate'`);let s=r.get("ANTHROPIC_API_KEY");if(s){i.info(`Verifying ANTHROPIC_API_KEY (${ne(s)})...`);try{let a=await re(s);i.success(`\u2713 Anthropic key valid (${a.length} models)`)}catch(a){i.error(`\u2717 Anthropic key invalid: ${a instanceof Error?a.message:a}`)}}}async function vu(){let t=await pt(),e=await Lt(t);if(e.keys.length===0){i.info("Kh\xF4ng c\xF3 plaintext secret trong settings.json \u2014 nothing to migrate.");return}i.warn(`Migrating ${e.keys.length} secrets: ${e.keys.join(", ")}`);let n=await _n(t);i.success(`\u2713 Migrated ${n.migratedKeys.length} secrets to .envrc`),n.settingsJsonBackupPath&&i.dim(` Backup: ${n.settingsJsonBackupPath}`)}function ki(t){let e=t.command("secrets").description("Qu\u1EA3n l\xFD per-project secrets via direnv .envrc (v1.19.0)"),n=o=>async()=>{try{await o()}catch(s){i.error(`secrets failed: ${s instanceof Error?s.message:s}`),process.exit(1)}},r=o=>async s=>{try{await o(s)}catch(a){i.error(`secrets failed: ${a instanceof Error?a.message:a}`),process.exit(1)}};e.command("setup").description("Interactive wizard: install direnv + migrate + prompt new key").action(n(gu)),e.command("set <NAME>").description("Set/update one secret in .envrc (hidden input prompt)").action(r(fu)),e.command("get <NAME>").description("Print one secret value (masked on TTY, raw when piped)").action(r(hu)),e.command("list").description("List Avatar-managed secret names (values masked)").action(n(wu)),e.command("rm <NAME>").description("Remove one secret from .envrc").action(r(ku)),e.command("check").description("Health: direnv installed, .envrc exists, Anthropic key validates").action(n(yu)),e.command("migrate").description("Migrate plaintext secrets from settings.json \u2192 .envrc").action(n(vu))}import{promises as Su}from"fs";import{join as Oe}from"path";import Pu from"boxen";import{promises as bu}from"fs";import{join as xu}from"path";var Au="_backup";async function yi(t){let e=xu(t,".claude",Au);return await d(e)?(await bu.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}f();function vi(t){t.command("status").description("Snapshot t\u1EE9c th\xEC: project, pack version, pending, backup").option("--json","Output JSON cho script").action(async e=>{try{let n=await Cu(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
|
|
118
|
+
`):$u(n)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Cu(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=Oe(t,".claude");if(!await d(n))return{projectName:e,cliVersion:L(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,toolsEnabled:[],toolsAvailableCount:0};let o=Oe(n,"_pending"),[s,a,c,l,u]=await Promise.all([(async()=>await d(Oe(n,"pack"))?Ee(t).catch(()=>null):null)(),(async()=>await d(o)?(await Su.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),yi(t).then(m=>m.length).catch(()=>0),Eu(n).catch(()=>"(read error)"),jt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:L(),packVersion:s,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,toolsEnabled:u.enabled,toolsAvailableCount:u.available.length}}async function Eu(t){let e=Oe(t,"project","tech-stack.md");return await d(e)?(await N(e)).split(`
|
|
119
|
+
`).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function Tu(t){return t.toolsEnabled.length>0?`${t.toolsEnabled.join(", ")} (${t.toolsEnabled.length}/${t.toolsAvailableCount} available)`:t.toolsAvailableCount>0?`none enabled (${t.toolsAvailableCount} available \u2014 avatar tool list)`:"none"}function $u(t){let e=[`${p.bold("Avatar Status")} \xB7 ${p.cyan(t.projectName)}`,"\u2500".repeat(48),`${p.dim("CLI version:")} ${t.cliVersion}`,`${p.dim("Pack version:")} ${t.packVersion??p.yellow("not installed")}`,`${p.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?p.dim(" (avatar review)"):""}`,`${p.dim("Backups:")} ${t.backupCount}`,`${p.dim("Tech stack:")} ${t.techStackSummary}`,`${p.dim("Tools:")} ${Tu(t)}`];process.stdout.write(`${Pu(e.join(`
|
|
144
120
|
`),{padding:1,borderStyle:"round"})}
|
|
145
|
-
`)}
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
`,"utf8")}async function
|
|
154
|
-
`),o=[],s=!1;for(let c of i){if(c.trim().startsWith("[submodule")&&c.includes(e)){s=!0;continue}s&&c.trim().startsWith("[submodule")&&(s=!1),s||o.push(c)}let a=o.join(`
|
|
155
|
-
`).trim();a.length===0?await X(t,{force:!0}):await Xr(t,`${a}
|
|
156
|
-
`,"utf8")}function Qr(t){t.command("uninstall").description("G\u1EE1 Avatar kh\u1ECFi project \u2014 backup t\u1EF1 \u0111\u1ED9ng (M11)").option("--yes","Skip confirm prompt").option("--no-backup","Kh\xF4ng t\u1EA1o backup tr\u01B0\u1EDBc khi x\xF3a (nguy hi\u1EC3m)").option("--keep-submodule","Gi\u1EEF submodule .claude/pack/").option("--keep-hooks","Gi\u1EEF git hooks post-merge, pre-push").option("--dry-run","Hi\u1EC3n th\u1ECB danh s\xE1ch s\u1EBD x\xF3a, kh\xF4ng th\u1EF1c thi").action(async e=>{try{await Bp(e)}catch(n){r.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Bp(t){let e=process.cwd(),n=Jr(e);if(!n.hasAnyArtifact){r.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Fp(e,n,t),t.dryRun){r.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Vp({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){r.info("\u0110\xE3 h\u1EE7y.");return}let i=null;t.noBackup||(i=await qr(e,n,H()),r.success(`Backup t\u1EA1o t\u1EA1i: ${i}`)),await Zr(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await v("uninstall",`project=${e},backup=${i??"skipped"}`),Wp(i)}function Fp(t,e,n){r.info(`Project: ${t}`),r.plain(""),r.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&r.plain(` ${p.red("\u2717")} ${Dp(t,e.claudeDir)||".claude/"}`),e.claudeMd&&r.plain(` ${p.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&r.plain(` ${p.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&r.plain(` ${p.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&r.plain(` ${p.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&r.plain(` ${p.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),r.plain(""),r.plain("Kh\xF4ng \u0111\u1EE5ng:"),r.plain(` ${p.green("\u2713")} src/ (code kh\xE1ch)`),r.plain(` ${p.green("\u2713")} Git history`),r.plain(` ${p.green("\u2713")} ~/.avatar/config.json (token SSO)`),r.plain(` ${p.green("\u2713")} Secrets trong keychain`),r.plain("")}function Wp(t){let e=[`${p.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${p.dim("Backup:")} ${t}`),e.push(` ${p.dim("Restore:")} ${p.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${Kp(e.join(`
|
|
121
|
+
`)}import{join as Si}from"path";import{join as bi}from"path";async function Ru(t,e,n){let r=bi(t,n),o=bi(e,n);if(!await d(r))return"source-missing";if(!await d(o))return"needs-creation";let{promises:s}=await import("fs");return(await s.lstat(o)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function xi(t,e,n){let r=await Ae(t)??"(ch\u01B0a c\xE0i)",o=n??"stable m\u1EDBi nh\u1EA5t (server resolve)",s=[];for(let a of Rn)s.push({dir:a,status:await Ru(t,e,a)});return{currentVersion:r,targetVersion:o,mountDirStatuses:s}}f();async function Ai(t){let e=await kt(t);if(e.length!==0){i.info(`Re-apply ${e.length} tool(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await ut(t,n)}}f();function _u(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function Iu(t){let e=process.cwd(),n=Si(e,".claude"),r=Si(e,B);if(await d(n)||(i.error(`Kh\xF4ng th\u1EA5y .claude/ \u2014 kh\xF4ng ph\u1EA3i Avatar workspace.
|
|
122
|
+
Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await xi(r,n,t.version);i.info(`Pack version hi\u1EC7n t\u1EA1i: ${c.currentVersion}`),i.info(`Target version: ${c.targetVersion}`),i.info(`
|
|
123
|
+
Mount dir statuses:`);for(let l of c.mountDirStatuses)console.log(` ${l.dir.padEnd(12)} ${l.status}`);i.info(`
|
|
124
|
+
Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await Qt()}catch(c){i.error(c instanceof Error?c.message:String(c)),process.exit(1)}i.info(t.version?`T\u1EA3i team-ai-pack ${t.version} t\u1EEB Supabase...`:"T\u1EA3i team-ai-pack (stable m\u1EDBi nh\u1EA5t) t\u1EEB Supabase...");let s;try{let c=await Pe(o,t.version);await Ce(c.url,r),s=t.version??_u(c.object),await xe(r,{version:s,downloadedObject:c.object,extractedAt:new Date().toISOString()}),i.success(`\u0110\xE3 c\xE0i pack ${s}`)}catch(c){c instanceof ct?i.error(`${c.message}`):i.error(`T\u1EA3i pack th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`),process.exit(1)}i.info("T\u1EA1o symlink farm...");let a=await _e(r,n,t.force===!0);Ou(a,t.force===!0),i.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await pe(e);switch(c.action){case"merged":i.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":i.info(" - settings.json \u0111\xE3 sync.");break;case"no-pack-template":i.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip.");break}}catch(c){i.warn(` ! Merge settings.json fail: ${c instanceof Error?c.message:c}.`)}try{await Ai(e)}catch(c){i.warn(` ! Re-apply tools fail: ${c instanceof Error?c.message:c}.`)}i.success(`Synced team-ai-pack \u2192 ${s}.`)}function Ou(t,e){for(let r of t)switch(r.action){case"created":i.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":i.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":i.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":i.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":i.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override.`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&i.warn(`${n} mount dir(s) skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force.`)}function Pi(t){t.command("sync").description("T\u1EA3i team-ai-pack m\u1EDBi nh\u1EA5t (tarball Supabase) + t\u1EA1o symlink farm").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--version <tag>","Pin version c\u1EE5 th\u1EC3 (vd: v0.2.0 ho\u1EB7c v0.2.0-beta.0)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply").action(Iu)}import{resolve as Nu}from"path";import{checkbox as Mu,confirm as ju}from"@inquirer/prompts";f();function Ci(t,e){let n=new Set(e);return t.map(r=>({name:n.has(r)?`${r} (installed)`:r,value:r,checked:n.has(r)}))}function Ei(t){return t.map(e=>({name:e,value:e,checked:!1}))}function Ti(t){return[...t]}function Gt(t){return Nu(t.target??process.cwd())}async function $i(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return Ti(t);let s=r==="add"?Ci(t,e):Ei(t);return await Mu({message:r==="add"?"Ch\u1ECDn tool \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn tool \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:s})}async function Lu(t){let e=Gt(t),{available:n,enabled:r}=await jt(e);if(n.length===0){i.dim("Kh\xF4ng c\xF3 tool available (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync).");return}let o=await $i(n,r,t,"add");if(o.length===0){i.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let s of o)await ut(e,s)}async function Gu(t){let e=Gt(t),{enabled:n}=await jt(e);if(n.length===0){i.dim("Kh\xF4ng c\xF3 tool n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await $i(n,[],t,"remove");if(r.length===0){i.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await ju({message:`G\u1EE1 ${r.length} tool: ${r.join(", ")}?`,default:!0})){i.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 tool n\xE0o.");return}for(let s of r)await En(e,s)}async function Uu(t){let e=Gt(t),n=await Cn(e),r=await wt(e);if(n.length===0&&Object.keys(r.tools).length===0){i.dim("Kh\xF4ng c\xF3 tool n\xE0o (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let o=[...new Set([...n,...Object.keys(r.tools)])].sort();i.info("Tools:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let s of o){let a=n.includes(s)?"yes":"no",c=r.tools[s],l=c?.enabled?"yes":"no",u=c?.version??"-";console.log(` ${s.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${u}`)}}function Ri(t){let e=t.command("tools").description("Qu\u1EA3n l\xFD tool toggle (prompt-scoring, ...) b\u1EADt/t\u1EAFt nguy\xEAn kh\u1ED1i");e.command("enable <name>").description("B\u1EADt tool: merge manifest v\xE0o settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let o=Gt(r);await ut(o,n)||process.exit(1)}),e.command("disable <name>").description("T\u1EAFt tool: r\xFAt \u0111\xFAng entry manifest ra kh\u1ECFi settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let o=Gt(r);await En(o,n)}),e.command("list").description("Li\u1EC7t k\xEA tool: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await Uu(n)}),e.command("add").description("Ch\u1ECDn (multi-select) tool available \u0111\u1EC3 c\xE0i").option("--target <path>","Workspace root (default: cwd)").option("--all","C\xE0i h\u1EBFt tool available (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Lu(n)}),e.command("remove").description("Ch\u1ECDn (multi-select) tool \u0111ang b\u1EADt \u0111\u1EC3 g\u1EE1").option("--target <path>","Workspace root (default: cwd)").option("--all","G\u1EE1 h\u1EBFt tool \u0111ang b\u1EADt (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Gu(n)})}import{relative as zu}from"path";import{confirm as Ju}from"@inquirer/prompts";import qu from"boxen";import{cp as Ne,mkdir as _i,writeFile as Du}from"fs/promises";import{homedir as Hu}from"os";import{basename as Ku,join as z}from"path";var Vu=z(Hu(),".avatar","uninstall-backups");async function Ii(t,e,n){let r=Ku(t),o=new Date().toISOString().replace(/[:.]/g,"-"),s=z(Vu,`${r}-${o}`);if(await _i(s,{recursive:!0,mode:448}),e.claudeDir&&await Ne(e.claudeDir,z(s,".claude"),{recursive:!0}),e.claudeMd&&await Ne(e.claudeMd,z(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=z(s,"hooks");await _i(c,{recursive:!0}),e.postMergeHook&&await Ne(e.postMergeHook,z(c,"post-merge")),e.prePushHook&&await Ne(e.prePushHook,z(c,"pre-push"))}let a={projectName:r,projectPath:t,timestamp:o,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await Du(z(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as Fu}from"fs";import{join as J}from"path";function q(t){return Fu(t)?t:null}function Oi(t){let e=q(J(t,".claude")),n=q(J(t,"CLAUDE.md")),r=q(J(t,".git","hooks","post-merge")),o=q(J(t,".git","modules","src","hooks","pre-push")),s=q(J(t,".gitignore")),a=q(J(t,".gitmodules")),c=q(J(t,"notes")),l=q(J(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||o),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:o,gitignorePath:s,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as Ni,rm as Y,writeFile as Mi}from"fs/promises";async function ji(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),o=await n(t.claudeDir);for(let s of o)s!=="pack"&&await Y(r(t.claudeDir,s),{recursive:!0,force:!0})}else await Y(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await Y(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await Y(t.postMergeHook,{force:!0}),t.prePushHook&&await Y(t.prePushHook,{force:!0})),t.gitignorePath&&await Bu(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Wu(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:r}=await import("fs/promises");(await r(n)).length===0&&await Y(n,{recursive:!0,force:!0})}}async function Bu(t){let e=await Ni(t,"utf8"),n=e.indexOf(An),r=e.indexOf(ve);if(n===-1||r===-1)return;let o=e.slice(0,n),s=e.slice(r+ve.length),a=`${o.trimEnd()}
|
|
125
|
+
${s.trimStart()}`.trim();a.length===0?await Y(t,{force:!0}):await Mi(t,`${a}
|
|
126
|
+
`,"utf8")}async function Wu(t,e){let r=(await Ni(t,"utf8")).split(`
|
|
127
|
+
`),o=[],s=!1;for(let c of r){if(c.trim().startsWith("[submodule")&&c.includes(e)){s=!0;continue}s&&c.trim().startsWith("[submodule")&&(s=!1),s||o.push(c)}let a=o.join(`
|
|
128
|
+
`).trim();a.length===0?await Y(t,{force:!0}):await Mi(t,`${a}
|
|
129
|
+
`,"utf8")}f();function Li(t){t.command("uninstall").description("G\u1EE1 Avatar kh\u1ECFi project \u2014 backup t\u1EF1 \u0111\u1ED9ng (M11)").option("--yes","Skip confirm prompt").option("--no-backup","Kh\xF4ng t\u1EA1o backup tr\u01B0\u1EDBc khi x\xF3a (nguy hi\u1EC3m)").option("--keep-submodule","Gi\u1EEF submodule .claude/pack/").option("--keep-hooks","Gi\u1EEF git hooks post-merge, pre-push").option("--dry-run","Hi\u1EC3n th\u1ECB danh s\xE1ch s\u1EBD x\xF3a, kh\xF4ng th\u1EF1c thi").action(async e=>{try{await Yu(e)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Yu(t){let e=process.cwd(),n=Oi(e);if(!n.hasAnyArtifact){i.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Xu(e,n,t),t.dryRun){i.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Ju({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){i.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await Ii(e,n,L()),i.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await ji(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await v("uninstall",`project=${e},backup=${r??"skipped"}`),Zu(r)}function Xu(t,e,n){i.info(`Project: ${t}`),i.plain(""),i.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&i.plain(` ${p.red("\u2717")} ${zu(t,e.claudeDir)||".claude/"}`),e.claudeMd&&i.plain(` ${p.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&i.plain(` ${p.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&i.plain(` ${p.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&i.plain(` ${p.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&i.plain(` ${p.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),i.plain(""),i.plain("Kh\xF4ng \u0111\u1EE5ng:"),i.plain(` ${p.green("\u2713")} src/ (code kh\xE1ch)`),i.plain(` ${p.green("\u2713")} Git history`),i.plain(` ${p.green("\u2713")} ~/.avatar/config.json (token SSO)`),i.plain(` ${p.green("\u2713")} Secrets trong keychain`),i.plain("")}function Zu(t){let e=[`${p.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${p.dim("Backup:")} ${t}`),e.push(` ${p.dim("Restore:")} ${p.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${qu(e.join(`
|
|
157
130
|
`),{padding:1,borderStyle:"round"})}
|
|
158
|
-
`)}var
|
|
159
|
-
${
|
|
131
|
+
`)}var In=L(),C=new Qu;C.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(In,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
|
|
132
|
+
${xn({tagline:`v${In} \xB7 AI harness CLI for NAL Vietnam`})}
|
|
160
133
|
|
|
161
|
-
`);var
|
|
134
|
+
`);var tp=process.argv.includes("-v")||process.argv.includes("--version");tp&&(yt({tagline:`v${In} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));Zo(C);li(C);Pi(C);fi(C);gi(C);vi(C);vo(C);mi(C);Ri(C);ki(C);ui(C);so(C);Mo(C);di(C);Cr(C);Li(C);C.parseAsync(process.argv).catch(t=>{let e=t instanceof Error?t.message:String(t);process.stderr.write(`\u2717 L\u1ED7i kh\xF4ng x\u1EED l\xFD \u0111\u01B0\u1EE3c: ${e}
|
|
162
135
|
`),process.exit(1)});
|
|
163
136
|
//# sourceMappingURL=index.js.map
|