@nalvietnam/avatar-cli 1.13.0 → 1.14.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.js +66 -65
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -1,75 +1,75 @@
|
|
|
1
1
|
// @nalvietnam/avatar-cli — built with tsup
|
|
2
|
-
import{Command as
|
|
3
|
-
`,n)}import{existsSync as Nt,readFileSync as
|
|
4
|
-
`;await
|
|
2
|
+
import{Command as Tl}from"commander";import{promises as Ho}from"fs";import{join as Un}from"path";import{confirm as Do}from"@inquirer/prompts";import{constants as Dr,promises as ot}from"fs";import{dirname as Vr,join as Il,relative as Ol}from"path";async function d(t){try{return await ot.access(t,Dr.F_OK),!0}catch{return!1}}async function x(t){await ot.mkdir(t,{recursive:!0})}async function I(t){return await ot.readFile(t,"utf8")}async function C(t){return JSON.parse(await I(t))}async function st(t,e,n){await x(Vr(t));let i=`${t}.tmp-${process.pid}-${Date.now()}`;await ot.writeFile(i,e,"utf8"),n!==void 0&&await ot.chmod(i,n),await ot.rename(i,t)}async function $(t,e,n){await st(t,`${JSON.stringify(e,null,2)}
|
|
3
|
+
`,n)}import{existsSync as Nt,readFileSync as Br}from"fs";import{dirname as Kr,join as Gt}from"path";var Fr=5;function Wr(t){let e=Nt(Gt(t,".claude")),n=Nt(Gt(t,"CLAUDE.md"));if(!e||!n)return!1;let i=Nt(Gt(t,"src")),r=Gt(t,".gitmodules");if(i)return!0;if(Nt(r))try{let s=Br(r,"utf8");return s.includes("path = src")||s.includes("path = ./src")}catch{return!1}return!1}function Lt(t){let e=t;for(let n=0;n<Fr;n++){if(Wr(e))return e;let i=Kr(e);if(i===e)return null;e=i}return null}import{promises as bn}from"fs";import{homedir as zr}from"os";import{join as kt}from"path";import{z as h}from"zod";var kn=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)}),qr=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({})}),jl=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()}),Hl=h.enum(["internal","client","library"]);var wt=kt(zr(),".avatar"),W=kt(wt,"config.json"),Wl=kt(wt,"state.json"),we=kt(wt,"audit.log"),ql=kt(wt,"backups"),Yr=384;async function ye(){await x(wt)}async function q(){if(!await d(W))return null;let t=await C(W),e=kn.safeParse(t);return e.success?e.data:null}async function wn(t){await ye(),await $(W,t,Yr)}async function yn(){if(await d(W)){let{promises:t}=await import("fs");await t.unlink(W)}}function z(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}async function Jr(){try{await bn.chmod(we,384)}catch{}}async function f(t,e){await ye();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},i=`${JSON.stringify(n)}
|
|
4
|
+
`;await bn.appendFile(we,i,{encoding:"utf8",mode:384}),await Jr()}import{spawnSync as be}from"child_process";import u from"chalk";import Xr from"ora";var o={info:t=>process.stdout.write(`${u.blue("\u2139")} ${t}
|
|
5
5
|
`),success:t=>process.stdout.write(`${u.green("\u2713")} ${t}
|
|
6
6
|
`),warn:t=>process.stdout.write(`${u.yellow("\u26A0")} ${t}
|
|
7
7
|
`),error:t=>process.stderr.write(`${u.red("\u2717")} ${t}
|
|
8
8
|
`),dim:t=>process.stdout.write(`${u.dim(t)}
|
|
9
9
|
`),plain:t=>process.stdout.write(`${t}
|
|
10
|
-
`)};function O(t){return
|
|
11
|
-
${i}`);return a==="unknown"&&(o.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&o.warn(`[debug] stderr: ${n.slice(0,500)}`),i.trim()&&o.warn(`[debug] stdout: ${i.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||i.slice(0,500)}}import{spawnSync as
|
|
10
|
+
`)};function O(t){return Xr({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function yt(t){let e=Date.now(),n=O(`${t} (0:00)`),i=()=>{let s=Math.floor((Date.now()-e)/1e3),a=Math.floor(s/60),c=s%60;return`${a}:${String(c).padStart(2,"0")}`},r=setInterval(()=>{n.text=`${t} (${i()})`},1e3);return{succeed:s=>{clearInterval(r),n.succeed(`${s} (${i()})`)},fail:s=>{clearInterval(r),n.fail(`${s} (${i()})`)},stop:()=>{clearInterval(r),n.stop()}}}var vn=6e4,Qr="ok";function ve(){let t=be("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 xe(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=be("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.`);o.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function Zr(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 xn(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 Se(){let t=be("claude",["--print",Qr],{encoding:"utf8",timeout:vn,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${vn/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",i=t.stdout||"";if(t.status===0)return{ok:!0};let r=i.trim(),s=n.toLowerCase();if(r.length>20&&!s.includes("error")&&!s.includes("limit")&&!s.includes("quota")&&!s.includes("401"))return o.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${r.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=Zr(`${n}
|
|
11
|
+
${i}`);return a==="unknown"&&(o.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&o.warn(`[debug] stderr: ${n.slice(0,500)}`),i.trim()&&o.warn(`[debug] stdout: ${i.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||i.slice(0,500)}}import{spawnSync as Sn}from"child_process";import{platform as to}from"os";function Y(){let t=to();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var eo=5e3,no=/(\d+\.\d+\.\d+)/;function io(){let e=Y()==="win32"?"where":"which",n=Sn(e,["claude"],{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 ro(){let t=Sn("claude",["--version"],{encoding:"utf8",timeout:eo});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return no.exec(e)?.[1]??null}var J=null;function bt(){if(J!==null)return J;let t=io();return t?(J={installed:!0,version:ro(),path:t},J):(J={installed:!1,version:null,path:null},J)}function Mt(){J=null}import{spawnSync as oo}from"child_process";var Cn=300*1e3,An="@anthropic-ai/claude-code",X=class extends Error{reason;exitCode;constructor(e,n,i=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=i}};function so(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new X("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${An} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new X("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new X("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Pn(){o.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=oo("npm",["install","-g",An],{stdio:["inherit","inherit","pipe"],timeout:Cn,encoding:"utf8"});if(t.signal==="SIGTERM")throw new X("timeout",`npm install timeout sau ${Cn/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),so(t.status,t.stderr||"");Mt();let e=bt();if(!e.installed||!e.path)throw new X("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 o.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 ao}from"fs";import{homedir as co}from"os";import{join as lo}from"path";import{select as $n}from"@inquirer/prompts";function uo(){return lo(co(),".claude","settings.json")}function Ce(){let t=uo(),e;try{e=ao(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 i=n.env||{},r=typeof i.ANTHROPIC_BASE_URL=="string"?i.ANTHROPIC_BASE_URL:void 0,s=typeof i.ANTHROPIC_AUTH_TOKEN=="string"&&i.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!r,baseUrl:r,hasToken:s,model:a,rawSettings:n}}async function Tn(t=Ce()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await $n({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 $n({message:"Ch\u1ECDn provider cho AI features:",choices:[{name:"1. Claude Code Subscription (d\xF9ng quota c\xE1 nh\xE2n Anthropic, OAuth login)",value:"subscription"},{name:"2. LLMLite API key (llm.nal.vn \u2014 NAL gateway, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}import{password as po,select as mo}from"@inquirer/prompts";var Ut="https://api.anthropic.com",go="2023-06-01",Rn=1e4;function ho(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function fo(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 ko(){return await po({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:fo})}async function wo(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Rn);try{let i=await fetch(`${Ut}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":go,Accept:"application/json"},signal:e.signal});if(i.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(i.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(i.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!i.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${i.status}).`);let s=((await i.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(i){throw i.name==="AbortError"?new Error(`Connect ${Ut} timeout sau ${Rn/1e3}s.`):i}finally{clearTimeout(n)}}async function yo(t){if(t.length===1){let n=t[0];return o.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,i)=>{let r=s=>{let a=s.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return r(n)-r(i)});return await mo({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function En(){let t=await ko();o.info(`Verify key (${ho(t)}) qua ${Ut}/v1/models...`);let e=await wo(t);o.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await yo(e);return{apiKey:t,baseUrl:Ut,model:n}}import{input as bo,password as vo,select as xo}from"@inquirer/prompts";var So="https://llm.nal.vn",_n=1e4;function N(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Co(){return await vo({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function Ao(t=So){return(await bo({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 Po(t,e){let n=new AbortController,i=setTimeout(()=>n.abort(),_n);try{let r=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(r.status===401||r.status===403)throw new Error(`API key invalid (HTTP ${r.status}).`);if(r.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let a=((await r.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(r){throw r.name==="AbortError"?new Error(`Connect ${t} timeout sau ${_n/1e3}s.`):r}finally{clearTimeout(i)}}async function $o(t){let e=t.filter(i=>i.toLowerCase().includes("claude"));if(e.length===1){let i=e[0];return o.info(`Auto-pick model: ${i} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),i}let n=e.length>0?e:t;return await xo({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(i=>({name:i,value:i}))})}async function In(){let t=await Co(),e=await Ao();o.info(`Verify key (${N(t)}) qua ${e}/v1/models...`);let n=await Po(e,t);o.success(`Endpoint OK \u2014 ${n.length} models available`);let i=await $o(n);return{apiKey:t,baseUrl:e,model:i}}import{promises as To}from"fs";import{join as Ro}from"path";var Ae=384;function Eo(t){return Ro(t,".claude","settings.json")}async function _o(t){if(!await d(t))return{};try{return await C(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 Io(t,e){let{env:n,...i}=t,r={...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&&(r.env=l)}return r}function Oo(t,e,n,i){let{ANTHROPIC_API_KEY:r,...s}=t.env||{};return{...t,env:{...s,ANTHROPIC_BASE_URL:n,ANTHROPIC_AUTH_TOKEN:e},model:i}}function No(t,e,n,i){let{ANTHROPIC_AUTH_TOKEN:r,...s}=t.env||{};return{...t,env:{...s,ANTHROPIC_BASE_URL:n,ANTHROPIC_API_KEY:e},model:i}}function Go(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 Q(t,e){let n=Eo(t),i=await _o(n),r;switch(e.provider){case"subscription":r=Io(i,e.model);break;case"llmlite":r=Oo(i,e.apiKey,e.baseUrl,e.model);break;case"anthropic":r=No(i,e.apiKey,e.baseUrl,e.model);break;case"use-global":r=Go(i,e.sourceSettings);break}await $(n,r,Ae);try{await To.chmod(n,Ae)}catch{}return{path:n,mode:Ae}}var T="sonnet";function On(t){o.warn(`\u{1F512} ${t} key \u0111\xE3 l\u01B0u PLAINTEXT v\xE0o .claude/settings.json.
|
|
12
12
|
File n\xE0y \u0111\u01B0\u1EE3c gitignore t\u1EEB Avatar v1.7.0, nh\u01B0ng workspace c\u0169 c\xF3 th\u1EC3 CH\u01AFA.
|
|
13
13
|
Ki\u1EC3m tra: grep '.claude/settings.json' .gitignore
|
|
14
|
-
N\u1EBFu ch\u01B0a c\xF3 \u2192 TH\xCAM NGAY, tr\xE1nh leak key khi commit/push.`)}async function
|
|
14
|
+
N\u1EBFu ch\u01B0a c\xF3 \u2192 TH\xCAM NGAY, tr\xE1nh leak key khi commit/push.`)}async function jt(t){try{o.info("Setup AI provider cho workspace...");let e=bt();if(e.installed)o.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(o.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),Pn(),Mt(),e=bt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=Ce();switch(await Tn(n)){case"subscription":{let r=ve();if(r.state!=="authenticated"&&(xe(),r=ve()),r.state==="authenticated"&&r.subscriptionType)return await Q(t.workspacePath,{provider:"subscription",model:T}),await f("ai_setup",`provider=subscription,result=ok,plan=${r.subscriptionType},probe=skipped`),o.success(`AI ready \xB7 Subscription (${r.subscriptionType}) \xB7 model=${T}`),{ok:!0,provider:"subscription",model:T};o.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=Se();if(!s.ok&&s.reason==="auth-expired"&&(o.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),xe(),s=Se()),!s.ok&&(s.reason==="timeout"||s.reason==="unknown"))return o.warn(`Probe verify ${s.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),s.detail?.trim()&&o.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),await Q(t.workspacePath,{provider:"subscription",model:T}),await f("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),o.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${T}`),{ok:!0,provider:"subscription",model:T};if(!s.ok){let a=s.reason??"unknown";return await f("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),o.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),s.detail?.trim()&&o.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),o.warn(` \u2192 ${xn(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await Q(t.workspacePath,{provider:"subscription",model:T}),await f("ai_setup","provider=subscription,result=ok"),o.success(`AI ready \xB7 Subscription \xB7 model=${T}`),{ok:!0,provider:"subscription",model:T}}case"llmlite":{let r=await In();return await Q(t.workspacePath,{provider:"llmlite",apiKey:r.apiKey,baseUrl:r.baseUrl,model:r.model}),await f("ai_setup",`provider=llmlite,result=ok,model=${r.model},base=${r.baseUrl}`),o.success(`AI ready \xB7 LLMLite \xB7 model=${r.model} \xB7 ${r.baseUrl}`),On("LLMLite"),{ok:!0,provider:"llmlite",model:r.model}}case"anthropic":{let r=await En();return await Q(t.workspacePath,{provider:"anthropic",apiKey:r.apiKey,baseUrl:r.baseUrl,model:r.model}),await f("ai_setup",`provider=anthropic,result=ok,model=${r.model}`),o.success(`AI ready \xB7 Anthropic Direct \xB7 model=${r.model} \xB7 ${r.baseUrl}`),On("Anthropic Direct API"),{ok:!0,provider:"anthropic",model:r.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 Q(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await f("ai_setup","provider=use-global,result=ok"),o.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 o.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),o.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await f("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}import{spawnSync as Lo}from"child_process";var Pe=1e4,Nn=3e4,Ln=5,$e="say ok",Gn="2023-06-01";async function Mo(t,e,n){o.info(`Testing LLMLite provider: ${t} (key: ${N(e)})`);let i=new AbortController,r=setTimeout(()=>i.abort(),Pe);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(v=>typeof v.id=="string"?v.id:null).filter(v=>v!==null);if(o.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let v=c.slice(0,5).join(", "),P=c.length>5?` ...+${c.length-5} more`:"";o.dim(` Models: ${v}${P}`)}o.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:$e}],max_tokens:Ln}),signal:i.signal});if(!l.ok){let v=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${v}`)}let p=await l.json(),m=typeof p.choices?.[0]?.message?.content=="string"?p.choices[0].message.content:"(empty response)",g=p.usage?.total_tokens??"?";o.success(`Response: "${String(m).trim().slice(0,100)}"`),o.dim(` Tokens used: ${g}`)}catch(s){throw s.name==="AbortError"?new Error(`Timeout ${Pe/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(r)}}function Uo(){o.info("Testing Subscription provider qua `claude --print`...");let t=Lo("claude",["--print",$e],{encoding:"utf8",timeout:Nn});if(t.signal==="SIGTERM")throw new Error(`Timeout ${Nn/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)}`)}o.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function jo(t,e,n){o.info(`Testing Anthropic Direct provider: ${t} (key: ${N(e)})`);let i=new AbortController,r=setTimeout(()=>i.abort(),Pe);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":Gn},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);o.success(`Connectivity OK \xB7 ${c.length} models available`),o.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":Gn,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:Ln,messages:[{role:"user",content:$e}]}),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);o.success(`Response: "${m}"`)}finally{clearTimeout(r)}}async function Mn(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,r=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,s=typeof t.model=="string"?t.model:"default";return r&&n?(await jo(n,r,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&i?(await Mo(n,i,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Uo(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function Ht(){let t=process.cwd(),e=Lt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
15
15
|
Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
|
|
16
16
|
B\u1EA1n \u0111ang \u1EDF: ${t}
|
|
17
|
-
Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function
|
|
18
|
-
Ch\u1EA1y 'avatar commit' trong workspace dir (c\xF3 .git v\xE0 .claude/).`);if(!
|
|
19
|
-
Ch\u1EA1y 'avatar commit' trong Avatar workspace, kh\xF4ng ph\u1EA3i project b\xECnh th\u01B0\u1EDDng.`);if(!
|
|
20
|
-
Workspace thi\u1EBFu submodule src/. Ch\u1EA1y 'avatar init' l\u1EA1i?`)}function R(t,e){let n=
|
|
21
|
-
${i}`)}return(n.stdout||"").trim()}function
|
|
17
|
+
Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function Te(t){let e=Un(t,".claude","settings.json");if(!await d(e))return{};try{return await C(e)}catch{return{}}}async function Vo(){let t=await Ht();await jt({workspacePath:t})}async function Bo(){let t=await Ht(),e=await Te(t),n=e.env||{},i=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,r=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=N(s)):i&&r?(c="LLMLite",l=N(r)):r?(c="Custom",l=N(r)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),o.info(`Project: ${t}`),o.info(`Provider: ${c}${i?` (${i})`:""}`),o.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),o.info(`Token: ${l}`)}async function Ko(){let t=await Ht(),e=await Te(t);try{let n=await Mn(e);o.success(`\u2713 ${n.message}`)}catch(n){o.error(`Test fail: ${n.message}`),process.exit(1)}}async function Fo(t){let e=await Ht(),n=Un(e,".claude","settings.json"),i=await Te(e);if(!t.yes&&!await Do({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){o.dim("\u0110\xE3 h\u1EE7y.");return}let{env:r,...s}=i,a={...s};if(r){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:p,...m}=r;Object.keys(m).length>0&&(a.env=m)}Object.keys(a).length===0?(await Ho.unlink(n).catch(()=>{}),o.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await $(n,a,384),o.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function jn(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 Vo()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await Bo()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await Ko()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await Fo(n)})}import{input as Jo}from"@inquirer/prompts";import{spawnSync as Wo}from"child_process";import{existsSync as Re}from"fs";import{join as Dt}from"path";function qo(t){let e=Dt(t,"src",".git"),n=Dt(t,".git"),i=Dt(t,".claude");if(!Re(n))throw new Error(`Kh\xF4ng ph\u1EA3i workspace root: ${t}
|
|
18
|
+
Ch\u1EA1y 'avatar commit' trong workspace dir (c\xF3 .git v\xE0 .claude/).`);if(!Re(i))throw new Error(`Kh\xF4ng th\u1EA5y .claude/ trong ${t}.
|
|
19
|
+
Ch\u1EA1y 'avatar commit' trong Avatar workspace, kh\xF4ng ph\u1EA3i project b\xECnh th\u01B0\u1EDDng.`);if(!Re(e))throw new Error(`Kh\xF4ng th\u1EA5y src/.git trong ${t}.
|
|
20
|
+
Workspace thi\u1EBFu submodule src/. Ch\u1EA1y 'avatar init' l\u1EA1i?`)}function R(t,e){let n=Wo("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:
|
|
21
|
+
${i}`)}return(n.stdout||"").trim()}function Hn(t){return R(t,["status","--porcelain"]).length>0}async function zo(t,e){let n=Dt(t,"src");if(!Hn(n))return o.dim("src/: nothing to commit (clean)"),{};o.info("Committing src/ ..."),R(n,["add","."]),R(n,["commit","-m",e.message]);let i=R(n,["rev-parse","HEAD"]);o.success(`src/ committed: ${i.slice(0,7)}`);let r=!1;return e.push&&(o.info("Pushing src/ ..."),R(n,["push"]),o.success("src/ pushed"),r=!0),{sha:i,pushed:r}}async function Yo(t,e){if(!Hn(t))return o.dim("workspace: nothing to commit (clean)"),{};o.info("Committing workspace root ..."),R(t,["add","."]),R(t,["commit","-m",e.message]);let n=R(t,["rev-parse","HEAD"]);o.success(`workspace committed: ${n.slice(0,7)}`);let i=!1;return e.push&&(o.info("Pushing workspace ..."),R(t,["push"]),o.success("workspace pushed"),i=!0),{sha:n,pushed:i}}async function Dn(t){qo(t.workspaceRoot);let e={target:t.target,skipped:[]};if(t.target==="src"||t.target==="all"){let n=await zo(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 Yo(t.workspaceRoot,t.options);e.workspaceCommitSha=n.sha,e.workspacePushed=n.pushed,n.sha||e.skipped?.push("workspace")}return e}function Vn(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 Xo(n)})}async function Xo(t){try{let e=t.message??await Jo({message:"Commit message:",validate:i=>i.trim().length>0?!0:"Message b\u1EAFt bu\u1ED9c"}),n=await Dn({workspaceRoot:process.cwd(),target:"src",options:{message:e,push:t.push}});n.srcCommitSha&&o.success(`src/: ${n.srcCommitSha.slice(0,7)}${n.srcPushed?" (pushed)":""}`),n.skipped&&n.skipped.length>0&&o.dim(`Skipped (nothing to commit): ${n.skipped.join(", ")}`)}catch(e){o.error(e instanceof Error?e.message:String(e)),process.exit(1)}}import{spawnSync as Ie}from"child_process";import{promises as Oe}from"fs";import{join as Z}from"path";import ls from"boxen";import{join as Ee}from"path";import{simpleGit as Qo}from"simple-git";function k(t=process.cwd()){return Qo({baseDir:t,binary:"git"})}async function at(t=process.cwd()){return await d(Ee(t,".git"))}async function Bn(t,e,n=process.cwd()){await k(n).subModule(["add",t,e])}async function vt(t,e,n=process.cwd()){let i=Ee(n,t);await k(i).fetch(["--tags"]),await k(i).checkout(e)}async function Vt(t,e,n=process.cwd()){let i=Ee(n,t);await k(i).fetch(["origin"]),await k(i).checkout(["-B",e,`origin/${e}`])}async function G(t=process.cwd()){return(await k(t).tags()).all}async function Bt(t=process.cwd()){try{return(await k(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function ct(t=process.cwd()){return(await k(t).revparse(["HEAD"])).trim()}import{promises as zn}from"fs";import{join as L}from"path";import{existsSync as ts}from"fs";import{dirname as Fn,join as xt}from"path";import{fileURLToPath as es}from"url";var Zo=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g;function Kn(t,e){return t.replace(Zo,(n,i)=>{let r=e[i];return r===void 0?n:String(r)})}var ns=Fn(es(import.meta.url)),Wn=os(ns),is=xt(Wn,"src","templates"),rs=xt(Wn,"src","hooks");function os(t){let e=t;for(;;){if(ts(xt(e,"package.json")))return e;let n=Fn(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function ss(t){return await I(xt(is,`${t}.tpl`))}async function Kt(t,e){let n=await ss(t);return Kn(n,e)}async function qn(t){return await I(xt(rs,`${t}.sh.tpl`))}async function as(t){if(!await d(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,i=n,r=1;for(;await d(i);)if(i=`${n}-${r}`,r++,r>5)throw new Error(`Could not find free backup name for ${t}`);return await zn.rename(t,i),i}async function Yn(t,e,n){let i=await as(t);return await st(t,e,n),i}var cs=["state","_pending","_backup"];async function Jn(t){let e=L(t,".claude");await x(e);for(let n of cs){let i=L(e,n);await x(i),await st(L(i,".gitkeep"),"")}}async function Xn(t,e){return[]}async function _e(t,e){let n=await Kt("CLAUDE.md",e);return await Yn(L(t,"CLAUDE.md"),n)}async function Qn(t,e){let n=await Kt("settings.json",e);return await Yn(L(t,".claude","settings.json"),n)}async function Zn(t){let e=L(t,".gitignore"),n=await Kt("gitignore",{}),i="# Avatar \u2014 git-ignored entries injected on `avatar init`",r="";if(await d(e)&&(r=await zn.readFile(e,"utf8"),r.includes(i)))return;let s=r.endsWith(`
|
|
22
22
|
`)||r.length===0?"":`
|
|
23
23
|
`;await st(e,`${r}${s}
|
|
24
|
-
${n}`)}async function St(t,e){let n=await
|
|
24
|
+
${n}`)}async function St(t,e){let n=await qn(e),i=L(t,"hooks");await x(i);let r=L(i,e);await st(r,n,493)}function ti(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 us(process.cwd());ps(n),e.fix&&await ms(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function us(t){let e=[],n=process.versions.node,[i,r]=n.split(".").map(K=>Number.parseInt(K,10)),s=(i??0)>18||(i??0)===18&&(r??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 q();a?z(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=Z(t,".claude","pack"),l=Z(t,"CLAUDE.md"),p=Z(t,".git","hooks","post-merge"),[m,g,v,P]=await Promise.all([at(t),d(c),d(l),d(p)]);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:v?"ok":"warn",detail:v?"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:P?"ok":"fail",detail:P?"installed":"missing \u2014 fixable",fixable:!P,fix:P?void 0:async()=>{await St(Z(t,".git"),"post-merge")}});let B=Z(t,".gitignore"),ht="";if(m){let K=!1;await d(B)&&(ht=await Oe.readFile(B,"utf8"),K=ht.includes(".claude/_pending/")),e.push({name:".gitignore Avatar entries",status:K?"ok":g?"fail":"warn",detail:K?"c\xF3 .claude/_pending/, .claude/_backup/":"thi\u1EBFu entries",fixable:!1});let F=ht.includes(".claude/settings.json")||ht.includes("/.claude/settings.json")||ht.includes(".claude/*.json");e.push({name:"\u{1F512} settings.json gitignored",status:F?"ok":"fail",detail:F?"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:!F,fix:F?void 0:async()=>{await Oe.appendFile(B,`
|
|
25
25
|
# Avatar v1.7.0 \u2014 Security: settings.json ch\u1EE9a raw API key, KH\xD4NG commit.
|
|
26
26
|
.claude/settings.json
|
|
27
27
|
.claude/settings.json.backup-*
|
|
28
|
-
`)}})}let
|
|
28
|
+
`)}})}let ln=Ie("which",["python"]),ke=Ie("which",["python3"]),un=ln.status===0,pn=ke.status===0;pn&&!un?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 ${ke.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):un?e.push({name:"Python binary",status:"ok",detail:`python: ${ln.stdout.toString().trim()}`,fixable:!1}):pn&&e.push({name:"Python binary",status:"ok",detail:`python3: ${ke.stdout.toString().trim()}`,fixable:!1});let mn=Z(t,".claude","settings.json");if(await d(mn))try{let K=await Oe.readFile(mn,"utf8"),F=JSON.parse(K);if(F.statusLine?.command){let hn=F.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(hn?.[2]){let ft=hn[2],Hr=ft.startsWith("/")?ft:Z(t,ft),fn=await d(Hr);e.push({name:"statusLine command",status:fn?"ok":"fail",detail:fn?`ref OK: ${ft}`:`BROKEN: settings.json ref '${ft}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let dn=Ie("which",["claude"]),gn=dn.status===0;return e.push({name:"Claude Code CLI",status:gn?"ok":"warn",detail:gn?dn.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),e}function ps(t){let e=[u.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,i=0,r=0;for(let s of t){let a=s.status==="ok"?u.green("\u2713"):s.status==="warn"?u.yellow("\u26A0"):u.red("\u2717");e.push(`${a} ${s.name.padEnd(28)} ${u.dim(s.detail)}`),s.status==="ok"?n+=1:(i+=1,s.fixable&&(r+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${i} issue${i===1?"":"s"}${r>0?` (${r} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${ls(e.join(`
|
|
29
29
|
`),{padding:1,borderStyle:"round"})}
|
|
30
|
-
`)}async function
|
|
30
|
+
`)}async function ms(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),o.success(`Fixed: ${n.name}`),e+=1}catch(i){o.error(`Failed to fix ${n.name}: ${i instanceof Error?i.message:String(i)}`)}e===0&&o.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}import{spawnSync as zs}from"child_process";import{promises as Ys}from"fs";import{join as yi}from"path";import{spawnSync as ni}from"child_process";import{existsSync as ds}from"fs";import{join as ei}from"path";var gs=120*1e3,hs=300*1e3,M=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,i,r=null,s){super(i),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=r,this.stderr=s}};function ii(t,e,n,i){if(n==="SIGTERM")return new M(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,i);let r=i.toLowerCase();return r.includes("eacces")||r.includes("permission denied")?new M(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,i):new M(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,i)}function Ft(t,e){return t.split(`
|
|
31
31
|
`).slice(-e).join(`
|
|
32
|
-
`)}function
|
|
33
|
-
`):i&&process.stderr.write(`${
|
|
34
|
-
`),
|
|
35
|
-
`):s&&process.stderr.write(`${
|
|
36
|
-
`),
|
|
32
|
+
`)}function ri(){let t=yt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=ni("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:gs,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(`${Ft(n,30)}
|
|
33
|
+
`):i&&process.stderr.write(`${Ft(i,30)}
|
|
34
|
+
`),ii("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function Wt(t){let e=yt(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=ni("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:hs,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let r=(n.stderr||"").trim(),s=(n.stdout||"").trim();throw r?process.stderr.write(`${Ft(r,30)}
|
|
35
|
+
`):s&&process.stderr.write(`${Ft(s,30)}
|
|
36
|
+
`),ii("analyze",n.status,n.signal,r)}let i=ei(t,".gitnexus","meta.json");if(!ds(i))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new M("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 ${ei(t,".gitnexus")})`)}import{confirm as Bs}from"@inquirer/prompts";import Ks from"boxen";import{spawnSync as oi}from"child_process";var fs=5e3,ks=/(\d+\.\d+\.\d+)/;function ws(){let e=Y()==="win32"?"where":"which",n=oi(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 ys(){let t=oi("gitnexus",["--version"],{encoding:"utf8",timeout:fs});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return ks.exec(e)?.[1]??null}var tt=null;function Ct(){if(tt!==null)return tt;let t=ws();return t?(tt={installed:!0,version:ys(),path:t},tt):(tt={installed:!1,version:null,path:null},tt)}function qt(){tt=null}import{spawnSync as bs}from"child_process";var si=300*1e3,ai="gitnexus",E=class extends Error{reason;exitCode;constructor(e,n,i=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=i}};function vs(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new E("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${ai} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new E("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new E("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function ci(){o.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=bs("npm",["install","-g",ai],{stdio:["inherit","inherit","pipe"],timeout:si,encoding:"utf8"});if(t.signal==="SIGTERM")throw new E("timeout",`npm install timeout sau ${si/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),vs(t.status,t.stderr||"");qt();let e=Ct();if(!e.installed||!e.path)throw new E("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 o.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{input as Wp,select as xs}from"@inquirer/prompts";var w=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function A(t){o.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&o.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 xs({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}import{promises as pi}from"fs";import{homedir as Ss}from"os";import{join as Cs}from"path";var li=384,ui={command:"gitnexus",args:["mcp"]};function As(){return Cs(Ss(),".claude","mcp_servers.json")}function Ps(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 $s(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await pi.copyFile(t,n),n}async function mi(){let t=As(),e={},n=!1;if(await d(t)){n=!0;try{e=await C(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&&Ps(i,ui))return o.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let r;n&&(r=await $s(t),o.dim(`Backup ${t} \u2192 ${r}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:ui}};await $(t,s,li);try{await pi.chmod(t,li)}catch{}return o.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:r}}import{spawnSync as Ns}from"child_process";import{existsSync as Gs}from"fs";import{join as ki}from"path";import{confirm as Ls}from"@inquirer/prompts";var Ts=[/^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 di(t){return t?Ts.some(e=>e.test(t)):!1}import{promises as At}from"fs";import{homedir as Rs}from"os";import{dirname as Es,join as gi}from"path";var _s=gi(Rs(),".gitnexus"),Pt=gi(_s,"config.json");async function Is(){try{let t=await At.readFile(Pt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function Os(t){await At.mkdir(Es(Pt),{recursive:!0});let e=`${Pt}.tmp-${process.pid}`;await At.writeFile(e,t,{mode:384}),await At.rename(e,Pt)}async function hi(t){let n={...await Is(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await Os(JSON.stringify(n,null,2)),await At.chmod(Pt,384).catch(()=>{})}var Ms=900*1e3,Us="nal-claude",js="claude-sonnet-4-5";function Hs(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function Ds(t){let e=ki(t,".claude","settings.json");if(!await d(e))return null;try{let n=await C(e),i=n.env||{},r=typeof i.ANTHROPIC_BASE_URL=="string"?i.ANTHROPIC_BASE_URL:null;if(!r)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=typeof i.ANTHROPIC_API_KEY=="string"?i.ANTHROPIC_API_KEY:null;if(l)return{provider:"anthropic",apiKey:l,baseUrl:Hs(r),model:c.length>0?c:js};let p=typeof i.ANTHROPIC_AUTH_TOKEN=="string"?i.ANTHROPIC_AUTH_TOKEN:null;return p?{provider:"llmlite",apiKey:p,baseUrl:r,model:c.length>0?c:Us}:null}catch{return null}}async function Vs(t,e){return await Ls({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function fi(t,e){return t.split(`
|
|
37
37
|
`).slice(-e).join(`
|
|
38
|
-
`)}async function
|
|
39
|
-
gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await
|
|
40
|
-
`):
|
|
41
|
-
`),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${l} (exit ${a.status??"null"})`}}let c=
|
|
38
|
+
`)}async function wi(t){let e=await Ds(t);if(!e)return o.warn("Subscription mode (OAuth, kh\xF4ng c\xF3 API key trong settings.json) \u2192 skip wiki gen."),o.dim(`\u0110\u1EC3 gen wiki sau, ch\u1EA1y manual:
|
|
39
|
+
gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await Vs(e.baseUrl,e.model))return o.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 i=di(e.model);await hi({apiKey:e.apiKey,baseUrl:e.baseUrl,model:e.model,isReasoningModel:i});let r=["wiki",".","--base-url",e.baseUrl,"--model",e.model];i&&r.push("--reasoning-model");let s=yt(`Generating wiki via ${e.baseUrl} (${e.provider}) model=${e.model}${i?" [reasoning]":""}`),a=Ns("gitnexus",r,{cwd:t,stdio:["ignore","pipe","pipe"],timeout:Ms,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 p=(a.stderr||"").trim(),m=(a.stdout||"").trim();return p?process.stderr.write(`${fi(p,30)}
|
|
40
|
+
`):m&&process.stderr.write(`${fi(m,30)}
|
|
41
|
+
`),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${l} (exit ${a.status??"null"})`}}let c=ki(t,".gitnexus","wiki","index.html");return Gs(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}`})}async function Fs(){let t=[u.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: ${u.cyan("npm install -g gitnexus")} (global)`];return process.stdout.write(`${Ks(t.join(`
|
|
42
42
|
`),{padding:1,borderStyle:"round",borderColor:"cyan"})}
|
|
43
|
-
`),await
|
|
43
|
+
`),await Bs({message:"C\xE0i GitNexus global?",default:!0})}async function Ws(){for(;;)try{return ci(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof E&&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.",i=await A({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(i==="abort")throw new w("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(i==="skip")return!1}}async function qs(t){for(;;)try{return Wt(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),i=e instanceof M&&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.",r=await A({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:i});if(r==="abort")throw new w("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(r==="skip")return!1}}async function zt(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{o.info("=== Phase 10: GitNexus Setup ===");let n=Ct();if(!n.installed){if(!await Fs())return await f("gitnexus_setup","result=skipped,reason=user-declined"),o.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await Ws())return await f("gitnexus_setup","result=skipped,reason=install-skipped"),o.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(qt(),n=Ct(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,o.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{ri()}catch(s){o.warn(`gitnexus setup fail: ${s.message}`),o.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(!await qs(t.workspacePath))return await f("gitnexus_setup","result=skipped,reason=analyze-skipped"),o.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let r=await wi(t.workspacePath);e.wikiGenerated=r.ran,r.skipped&&r.reason==="fail"&&o.warn(`Wiki gen fail (workspace v\u1EABn OK): ${r.detail??"unknown"}`);try{let s=await mi();e.mcpRegistered=!0,s.wasUpdated||o.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(s){o.warn(`MCP server register fail: ${s.message}`),o.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 f("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),o.success("GitNexus ready"),e}catch(n){if(n instanceof w)throw n;let i=n instanceof Error?n.message:String(n);return o.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${i}`),o.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await f("gitnexus_setup",`result=failed,error=${i.slice(0,200)}`),e.reason=i,e}}function Ne(){let t=process.cwd(),e=Lt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
44
44
|
Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
|
|
45
45
|
B\u1EA1n \u0111ang \u1EDF: ${t}
|
|
46
|
-
Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function
|
|
47
|
-
`);let i=
|
|
46
|
+
Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function Js(){let t=Ne(),e=await zt({workspacePath:t});e.ok?(o.success("GitNexus setup complete"),o.dim("Update CLAUDE.md \u0111\u1EC3 re-render section GitNexus: re-run avatar init ho\u1EB7c ch\u1EC9nh tay.")):o.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function Xs(){let t=Ne(),e=yi(t,".gitnexus","meta.json");if(!await d(e)){o.warn(`Ch\u01B0a c\xF3 ${e}. Ch\u1EA1y: avatar gitnexus install`);return}try{let n=await C(e);if(o.info(`Project: ${t}`),o.info(`Last commit: ${n.lastCommit?.slice(0,7)??"(unknown)"}`),o.info(`Indexed at: ${n.indexedAt??"(unknown)"}`),n.stats&&o.info(`Stats: ${n.stats.files??"?"} files \xB7 ${n.stats.nodes??"?"} nodes \xB7 ${n.stats.edges??"?"} edges`),n.lastCommit){let r=zs("git",["rev-parse","HEAD"],{cwd:t,encoding:"utf8"});r.status===0&&(r.stdout.trim()!==n.lastCommit?o.warn("\u26A0 Index stale \u2014 HEAD \u0111\xE3 ti\u1EBFn t\u1EEB lastCommit. Ch\u1EA1y: avatar gitnexus analyze"):o.dim("Index fresh (HEAD === lastCommit)"))}let i=yi(t,".gitnexus","wiki","index.html");if(await d(i)){let r=await Ys.stat(i);o.info(`Wiki: ${r.mtime.toISOString()}`)}else o.dim("Wiki: ch\u01B0a generate (ch\u1EA1y gitnexus wiki manual)")}catch(n){o.error(`Read meta.json fail: ${n.message}`),process.exit(1)}}async function Qs(){let t=Ne();try{Wt(t),o.success("Index refreshed. Ch\u1EA1y `avatar gitnexus status` xem chi ti\u1EBFt.")}catch(e){o.error(`Analyze fail: ${e.message}`),process.exit(1)}}function bi(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 Js()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await Xs()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await Qs()})}import{select as Jc}from"@inquirer/prompts";import Ge from"chalk";var Yt=[" \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"],Jt=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function Le(t,e,n){return Math.round(t+(e-t)*n)}function Zs(t){let n=Math.max(0,Math.min(1,t))*(Jt.length-1),i=Math.floor(n),r=Math.min(Jt.length-1,i+1),s=n-i,a=Jt[i],c=Jt[r];return[Le(a[0],c[0],s),Le(a[1],c[1],s),Le(a[2],c[2],s)]}function Me(t){if(!((process.stdout.isTTY??!1)&&Ge.level>0))return[...Yt,...t?.tagline?["",t.tagline]:[]].join(`
|
|
47
|
+
`);let i=Yt.map((r,s)=>{let a=Yt.length===1?0:s/(Yt.length-1),[c,l,p]=Zs(a);return Ge.rgb(c,l,p).bold(r)});return t?.tagline&&(i.push(""),i.push(Ge.dim(t.tagline))),i.join(`
|
|
48
48
|
`)}function lt(t){process.stdout.write(`
|
|
49
|
-
${
|
|
49
|
+
${Me(t)}
|
|
50
50
|
|
|
51
|
-
`)}import{spawnSync as
|
|
51
|
+
`)}import{spawnSync as Ci}from"child_process";import{input as ma,select as da}from"@inquirer/prompts";import{spawnSync as He}from"child_process";import{promises as ia}from"fs";import{basename as ra,join as xi}from"path";import{confirm as oa,select as sa}from"@inquirer/prompts";import{spawnSync as ta}from"child_process";var Ue=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 vi(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],i=ta("gh",n,{stdio:"inherit"});if(i.status!==0)throw i.status===1?new Ue(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 ea}from"child_process";function Xt(){let t=ea("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 na=/^[a-zA-Z0-9._-]{1,100}$/,je=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 Qt(t){if(!na.test(t))throw new je(t)}function Zt(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function ut(t){Qt(t.name),Zt(t.visibility);let e=t.org??Xt();o.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=vi({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return o.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}function aa(){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 ca(t){let e=xi(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-${aa()}`,i=xi(t,n);return await ia.rename(e,i),o.success(`Backup .git \u2192 ${n}`),i}function la(t){let e=He("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}`);o.success("Git init m\u1EDBi (branch main)"),He("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=He("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0?o.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(n.stderr||"").slice(0,200)}`):o.success("Initial commit")}async function Si(t){let e=ra(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await oa({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
|
|
52
52
|
1. Backup .git \u2192 .git.backup-{timestamp}
|
|
53
53
|
2. Git init m\u1EDBi (branch main, initial commit)
|
|
54
54
|
3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
|
|
55
|
-
Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let i=t.visibility??(t.autoYes?"private":await
|
|
56
|
-
${
|
|
57
|
-
`)}import{existsSync as
|
|
55
|
+
Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let i=t.visibility??(t.autoYes?"private":await sa({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),r=await ca(t.folderPath);return la(t.folderPath),{newRemoteUrl:ut({folder:t.folderPath,name:n,visibility:i,org:t.org}).httpsUrl,backupPath:r}}import{spawnSync as ua}from"child_process";var De=5e3;function pa(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 pt(t){let e=ua("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:De,stdio:["ignore","pipe","pipe"]});if(e.error){let r=e.error;return r.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:r.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${De/1e3}s`}:{ok:!1,reason:"unknown",detail:r.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${De/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:pa(n),detail:n.slice(0,300)}}var $t=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function ga(){let t=Ci("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function ha(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Ci("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function fa(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 ka(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function te(t){let e=t.url,n=t.initialReason,i=t.initialDetail;for(;;){let r=ga();o.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),o.dim(` L\xFD do: ${n}${i?` \u2014 ${i.slice(0,150)}`:""}`),o.info(fa(n,e,r)),r&&o.dim(` gh CLI hi\u1EC7n \u0111ang login: ${r}`);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 da({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new $t(`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){o.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await Si({folderPath:t.folderPath,visibility:t.defaultVisibility});return o.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),o.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){o.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await ma({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:p=>ka(p)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&ha(),o.info(`Verify remote l\u1EA1i: ${e}...`);let c=pt(e);if(c.ok)return o.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",i=c.detail}}import{readdirSync as Oa}from"fs";import{select as Na}from"@inquirer/prompts";import{simpleGit as Ri}from"simple-git";import{existsSync as wa,statSync as ya}from"fs";import{join as ba}from"path";function ee(t){let e=ba(t,".git");if(!wa(e))return!1;let n=ya(e);return n.isDirectory()||n.isFile()}import{simpleGit as va}from"simple-git";var Ai="chore: initial commit";async function Tt(t){let e=va({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(Ai,void 0,{"--allow-empty":null}):await e.commit(Ai))}import{existsSync as xa}from"fs";import{join as Sa}from"path";var Ca={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 Pi(t){let e=[];for(let[n,i]of Object.entries(Ca))i.some(r=>xa(Sa(t,r)))&&e.push(n);return e.length>0?e:["generic"]}import{readFileSync as Aa}from"fs";import{dirname as Pa,join as Rt}from"path";import{fileURLToPath as $a}from"url";var ne=Pa($a(import.meta.url)),Ta=[Rt(ne,"templates","gitignore"),Rt(ne,"..","templates","gitignore"),Rt(ne,"..","..","src","templates","gitignore"),Rt(ne,"..","src","templates","gitignore")],Et="# === avatar ===",et="# === /avatar ===";function Ra(t){for(let e of Ta)try{return Aa(Rt(e,`${t}.txt`),"utf8")}catch{}throw new Error(`Kh\xF4ng t\xECm th\u1EA5y template gitignore cho stack "${t}"`)}function $i(t){let n=["generic",...t.filter(i=>i!=="generic")].map(i=>`# --- ${i} ---
|
|
56
|
+
${Ra(i).trim()}`);return[Et,...n,et,""].join(`
|
|
57
|
+
`)}import{existsSync as Ea,readFileSync as _a,writeFileSync as Ve}from"fs";import{join as Ia}from"path";function Ti(t,e){let n=Ia(t,".gitignore");if(!Ea(n)){Ve(n,e,"utf8");return}let i=_a(n,"utf8"),r=i.indexOf(Et),s=i.indexOf(et);if(r!==-1&&s!==-1&&s>r){let a=i.slice(0,r),c=i.slice(s+et.length);Ve(n,`${a.trimEnd()}
|
|
58
58
|
|
|
59
|
-
${e}${c.trimStart()}`,"utf8");return}
|
|
59
|
+
${e}${c.trimStart()}`,"utf8");return}Ve(n,`${i.trimEnd()}
|
|
60
60
|
|
|
61
|
-
${e}`,"utf8")}var _t=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};async function
|
|
61
|
+
${e}`,"utf8")}var _t=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};async function Ga(t){return ee(t)?(await Ri({baseDir:t}).status()).isClean()?"clean":"dirty":Oa(t).filter(s=>s!==".git").length===0?"empty":"untracked-only"}async function La(t,e){return e.presetStrategy?e.presetStrategy:e.autoYes?"stash":t==="empty"||t==="clean"?"commit-all":await Na({message:t==="dirty"?"Folder c\xF3 changes ch\u01B0a commit. C\xE1ch x\u1EED l\xFD:":"Folder c\xF3 file ch\u01B0a version. C\xE1ch x\u1EED l\xFD:",choices:[{value:"stash",name:"1. Stash changes \u2192 bootstrap \u2192 restore (KHUY\u1EBEN NGH\u1ECA)"},{value:"commit-all",name:"2. Commit to\xE0n b\u1ED9 v\xE0o initial commit (legacy v1.1.6)"},{value:"skip",name:"3. Skip \u2014 t\xF4i commit th\u1EE7 c\xF4ng r\u1ED3i ch\u1EA1y l\u1EA1i"},{value:"branch",name:"4. Commit v\xE0o branch ri\xEAng `avatar/init` (main gi\u1EEF s\u1EA1ch)"}],default:"stash"})}async function Ma(t,e){let n=await t.status();return n.isClean()&&n.not_added.length===0?!1:(await t.stash(["push","--include-untracked","-m",e]),o.info(`Stashed changes: ${e}`),!0)}async function Ua(t,e){try{await t.stash(["pop"]),o.success(`Restored stash: ${e}`)}catch(n){o.warn("Restore stash conflict \u2014 files c\xF3 Avatar t\u1EA1o v\xE0 stash c\u1EE7a user xung \u0111\u1ED9t. Stash gi\u1EEF t\u1EA1i ref stash@{0}."),o.warn("Resolve: git stash show -p stash@{0} \u2192 fix conflict \u2192 git stash drop"),o.dim(`Detail: ${n.message}`)}}async function ja(t){try{let n=(await t.revparse(["--abbrev-ref","HEAD"])).trim();return n==="HEAD"?"main":n}catch{return"main"}}async function ie(t){let e=Pi(t);o.info(`Tech stack: ${e.join(", ")}`),Ti(t,$i(e)),o.success(".gitignore \u0111\xE3 ghi (Avatar block)")}async function Ha(t,e){let n=Ri({baseDir:t});switch(e){case"skip":throw new _t("Init aborted. Commit th\u1EE7 c\xF4ng changes hi\u1EC7n t\u1EA1i r\u1ED3i ch\u1EA1y l\u1EA1i `avatar init`.");case"stash":{let i=`avatar-init-backup-${Date.now()}`;ee(t)||(await n.init(),await n.branch(["-M","main"]).catch(()=>{})),(await n.raw(["rev-list","-n","1","--all"]).catch(()=>"")).trim()||await n.commit("chore: avatar baseline (pre-stash)",void 0,{"--allow-empty":null});let a=await Ma(n,i);try{await ie(t),await Tt(t)}finally{a&&await Ua(n,i)}break}case"commit-all":{await ie(t),await Tt(t);break}case"branch":{ee(t)||(await n.init(),await n.branch(["-M","main"]));let r=await ja(n);try{await n.checkoutLocalBranch("avatar/init")}catch{await n.checkout("avatar/init")}await ie(t),await Tt(t);try{await n.checkout(r),o.info(`Avatar init committed \u1EDF branch 'avatar/init'. Switch back v\u1EC1 '${r}'. Merge khi s\u1EB5n s\xE0ng: git merge avatar/init`)}catch{o.warn(`Kh\xF4ng switch v\u1EC1 '${r}' \u0111\u01B0\u1EE3c \u2014 \u1EDF l\u1EA1i branch 'avatar/init'. Switch tay sau.`)}break}}}async function Be(t,e={}){let n=await Ga(t);if(o.info(`Folder state: ${n}`),n==="empty"||n==="clean"){await ie(t),n==="empty"&&await Tt(t),await f("bootstrap",`state=${n},strategy=auto`);return}let i=await La(n,e);await Ha(t,i),await f("bootstrap",`state=${n},strategy=${i}`)}import{join as Li}from"path";import{spawnSync as Fe}from"child_process";import{confirm as Da,select as Va}from"@inquirer/prompts";import Ba from"boxen";function _i(t){return t.match(/github\.com[/:]([^/]+\/[^/]+?)(?:\.git)?$/)?.[1]??null}function Ei(t){return Fe("gh",["api",`repos/${t}`],{stdio:"ignore"}).status===0}function Ke(){let t=Fe("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Ka(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Fe("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}async function Fa(t){if(await Da({message:"Copy th\xF4ng tin (GitHub username + email) v\xE0o clipboard \u0111\u1EC3 d\xE1n v\xE0o Slack/email?",default:!0}))try{let{default:n}=await import("clipboardy");await n.write(t),o.success("\u0110\xE3 copy v\xE0o clipboard")}catch(n){o.dim(`Copy clipboard fail: ${n.message}`)}}function Wa(t,e,n){let i=[`${u.red("\u26D4 KH\xD4NG C\xD3 QUY\u1EC0N ACCESS")}`,"",`Repo: ${u.bold(t)}`,"","B\u1EA1n c\u1EA7n \u0111\u01B0\u1EE3c admin add v\xE0o org \u0111\u1EC3 pull team-ai-pack.","",`${u.dim("Th\xF4ng tin g\u1EEDi admin:")}`,` GitHub username: ${u.cyan(e??"(ch\u01B0a gh auth \u2014 ch\u1EA1y: gh auth login)")}`,` NAL email: ${u.cyan(n??"(ch\u01B0a avatar login \u2014 ch\u1EA1y: avatar login)")}`,` Date: ${new Date().toISOString().split("T")[0]}`,"",`${u.dim("Li\xEAn h\u1EC7:")} luke@nal.vn (Slack #avatar-setup)`];process.stdout.write(`${Ba(i.join(`
|
|
62
62
|
`),{padding:1,borderColor:"red",borderStyle:"round"})}
|
|
63
|
-
`)}function
|
|
64
|
-
`)}async function
|
|
63
|
+
`)}function qa(t,e,n){return[`Request access ${t} (NAL)`,"",`GitHub username: ${e??"(ch\u01B0a gh auth \u2014 ch\u1EA1y: gh auth login)"}`,`NAL email: ${n??"(ch\u01B0a avatar login \u2014 ch\u1EA1y: avatar login)"}`,`Date: ${new Date().toISOString().split("T")[0]}`].join(`
|
|
64
|
+
`)}async function Ii(t){if(Ei(t.repoSlug))return!0;let e=Ke();for(Wa(t.repoSlug,e,t.ssoEmail??null),await Fa(qa(t.repoSlug,e,t.ssoEmail??null));;){let n=Ke(),r=await Va({message:"C\xE1ch x\u1EED l\xFD?",choices:[{name:`\u0110\xE3 \u0111\u01B0\u1EE3c grant access v\u1EDBi GitHub username '${n??"(ch\u01B0a gh auth)"}' \u2014 ki\u1EC3m tra l\u1EA1i`,value:"retry-same"},{name:"\u0110\xE3 grant v\u1EDBi GitHub account kh\xE1c \u2014 switch gh (m\u1EDF browser)",value:"switch-account"},{name:"T\u1EA1m ng\u01B0ng \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"}]});if(r==="abort")return o.dim("T\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\xE3 accept invite t\u1EEB GitHub."),!1;if(r==="switch-account"&&Ka(),o.info("Ki\u1EC3m tra access..."),Ei(t.repoSlug)){let s=Ke();return o.success(`\u0110\xE3 c\xF3 access v\u1EDBi '${s??"(unknown)"}' \u2014 ti\u1EBFp t\u1EE5c.`),!0}o.warn(`V\u1EABn ch\u01B0a c\xF3 access v\u1EDBi account '${n??"(unknown)"}'. \u0110\u1EA3m b\u1EA3o \u0111\xE3 accept email invite ho\u1EB7c switch \u0111\xFAng account.`)}}var za=/^v?(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?$/;function Ya(t){let e=t.match(za);if(!e)return null;let[,n,i,r,s]=e;return{raw:t,major:Number.parseInt(n??"0",10),minor:Number.parseInt(i??"0",10),patch:Number.parseInt(r??"0",10),prerelease:s??null}}function U(t,e=!1){let n=t.map(Ya).filter(i=>i!==null).filter(i=>e||i.prerelease===null);return n.length===0?null:(n.sort((i,r)=>i.major!==r.major?i.major-r.major:i.minor!==r.minor?i.minor-r.minor:i.patch-r.patch),n[n.length-1]?.raw??null)}var Oi="git@github.com:nalvn/team-ai-pack.git",Ja=/^(git@github\.com:|https:\/\/github\.com\/)[A-Za-z0-9._-]+\/[A-Za-z0-9._-]+\.git$/,We=class extends Error{url;constructor(e){super(`AVATAR_TEAM_PACK_REPO_URL kh\xF4ng h\u1EE3p l\u1EC7: ${e}
|
|
65
65
|
Y\xEAu c\u1EA7u: GitHub URL c\xF3 .git suffix, vd:
|
|
66
66
|
git@github.com:owner/repo.git
|
|
67
67
|
https://github.com/owner/repo.git
|
|
68
|
-
L\xFD do block: ng\u0103n file:// (filesystem exfil) + URL malicious (git hook RCE).`),this.name="InvalidTeamPackUrlError",this.url=e}};function
|
|
68
|
+
L\xFD do block: ng\u0103n file:// (filesystem exfil) + URL malicious (git hook RCE).`),this.name="InvalidTeamPackUrlError",this.url=e}};function Ni(t){if(!Ja.test(t))throw new We(t)}function qe(){let t=process.env.AVATAR_TEAM_PACK_REPO_URL;return t?(Ni(t),t):(Ni(Oi),Oi)}var Yd=qe(),S=".claude/pack",nt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}},Gi="main";async function Mi(t,e,n,i=!1){let r=qe(),s=_i(r);if(s&&!await Ii({repoSlug:s,ssoEmail:n}))throw new nt("User ch\u1ECDn t\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\u01B0\u1EE3c add v\xE0o org.");try{await Bn(r,S,t)}catch(p){let m=p instanceof Error?p.message:String(p);throw(m.includes("Repository not found")||m.includes("not found"))&&o.error(`Repo team-ai-pack kh\xF4ng t\u1ED3n t\u1EA1i: ${r}
|
|
69
69
|
C\xE1ch fix:
|
|
70
70
|
1. T\u1EA1o repo: gh repo create <owner>/team-ai-pack --private --add-readme
|
|
71
71
|
2. Ho\u1EB7c override URL: export AVATAR_TEAM_PACK_REPO_URL=<url-repo-c\u1EE7a-b\u1EA1n>
|
|
72
|
-
3. Ho\u1EB7c d\xF9ng flag --skip-team-pack`),p}if(e)return await vt(
|
|
72
|
+
3. Ho\u1EB7c d\xF9ng flag --skip-team-pack`),p}if(e)return await vt(S,e,t),{pinnedTag:e};if(i)return await Vt(S,Gi,t),{pinnedTag:`${Gi} (HEAD)`};let a=Li(t,S),c=await G(a),l=U(c);return l&&await vt(S,l,t),{pinnedTag:l}}async function re(t){let e=Li(t,S),n=await Bt(e);return n||(await ct(e)).slice(0,7)}import{basename as Lc,join as Mc,resolve as de}from"path";import{input as Ot,select as Uc}from"@inquirer/prompts";import{spawnSync as Xa}from"child_process";function oe(){let t=Xa("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as Qa}from"child_process";function Za(t){let e=Y();return Qa(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function Ui(){let t=Y(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(Za(n))return n;return null}import{spawnSync as tc}from"child_process";var ec={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 ji(t){let e=ec[t];o.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=tc(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.`);o.success("\u0110\xE3 c\xE0i gh CLI")}import{spawnSync as nc}from"child_process";function Hi(){if(nc("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){o.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}o.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}import{spawnSync as ic}from"child_process";function Di(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=ic("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.`);o.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}async function it(t){for(;oe()==="not-installed";){o.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=Ui();if(!e){if(await A({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 w("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{ji(e)}catch(n){if(await A({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 w("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;oe()==="not-authenticated";){o.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{Di()}catch(e){if(await A({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 w("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(oe()!=="authenticated"&&await A({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 w("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(o.success("gh CLI s\u1EB5n s\xE0ng"),Hi(),t){let e=pt(t);return e.ok?(o.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await te({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{spawnSync as Vi}from"child_process";import{select as rc}from"@inquirer/prompts";function oc(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 sc(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Vi("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function ac(){o.info("M\u1EDF trang GitHub Settings \u2192 SSH Keys..."),Vi("open",["https://github.com/settings/keys"],{stdio:"ignore"}).status!==0&&o.info("URL: https://github.com/settings/keys")}async function cc(){return await rc({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 se(t,e,n,i=!1){for(;;)try{return{pinnedTag:(await Mi(t,e,n,i)).pinnedTag,skipped:!1}}catch(r){if(r instanceof nt)throw r;let s=r instanceof Error?r.message:String(r);if(oc(s)){o.warn("Pull team-ai-pack th\u1EA5t b\u1EA1i: SSH permission denied (publickey)."),o.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 cc();if(c==="abort")throw new w("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 o.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"){sc();continue}if(c==="https"){process.env.AVATAR_TEAM_PACK_REPO_URL="https://github.com/nalvn/team-ai-pack.git",o.info("Override URL sang HTTPS. L\u01B0u \xFD: HTTPS c\xF3 th\u1EC3 fail 403 n\u1EBFu read-only role."),ac();continue}continue}let a=await A({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 w("User abort t\u1EA1i b\u01B0\u1EDBc pull team-ai-pack.");if(a==="skip")return o.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}}}import{readFileSync as lc}from"fs";import{dirname as uc,resolve as pc}from"path";import{fileURLToPath as mc}from"url";var mt=null;function _(){if(mt!==null)return mt;let t=uc(mc(import.meta.url));for(let e=0;e<5;e++){let n=pc(t,...Array(e).fill(".."),"package.json");try{let i=lc(n,"utf8"),r=JSON.parse(i);if(r.name==="@nalvietnam/avatar-cli"&&typeof r.version=="string")return mt=r.version,mt}catch{}}return mt="unknown",mt}function Bi(t){let i=(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"");return i?`avatar-${i.replace(/^avatar-/,"")}-workspace`:"avatar-client-workspace"}function dc(t){return t?`
|
|
73
73
|
### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
|
|
74
74
|
|
|
75
75
|
Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
|
|
@@ -97,45 +97,46 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
|
|
|
97
97
|
\`\`\`bash
|
|
98
98
|
gitnexus wiki . --api-key <key> --base-url <url>
|
|
99
99
|
\`\`\`
|
|
100
|
-
`:""}function
|
|
101
|
-
`),
|
|
102
|
-
${
|
|
100
|
+
`:""}function ze(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:_(),packVersion:t.packVersion,lastScan:new Date().toISOString(),mode:t.mode,gitnexusSection:dc(t.gitnexusReady??!1)}}function Ye(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{join as gt}from"path";import{basename as Oc}from"path";import{confirm as ir,input as rr,select as en}from"@inquirer/prompts";import{spawnSync as rt}from"child_process";var dt=class extends Error{reason;fullName;stderr;constructor(e,n,i,r){super(i),this.name="CreateWorkspaceRemoteError",this.reason=e,this.fullName=n,this.stderr=r}};function gc(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 hc(t){return rt("gh",["repo","view",t,"--json","name"],{stdio:"ignore"}).status===0}function fc(t,e){return t.toLowerCase()===e.toLowerCase()?{ok:!0}:rt("gh",["api",`orgs/${t}/members/${e}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!0}:rt("gh",["api",`orgs/${t}`,"--silent"],{stdio:"ignore"}).status!==0?rt("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 Ki(t){Qt(t.workspaceName),Zt(t.visibility),await it();let e=Xt(),n=t.org??e,i=fc(n,e);if(!i.ok)throw new Error(`Kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi '${n}/': ${i.reason}`);let r=`${n}/${t.workspaceName}`;if(hc(r))throw new dt("repo-exists",r,`Repo '${r}' \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.`);o.info(`T\u1EA1o GitHub repo cho workspace: ${r} (${t.visibility})...`);let s=rt("gh",["repo","create",r,`--${t.visibility}`,"--source",t.workspacePath,"--remote","origin","--push"],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(s.status!==0){let l=(s.stderr||"").trim(),p=(s.stdout||"").trim(),m=[l,p].filter(Boolean).join(`
|
|
101
|
+
`),g=gc(m);throw m&&process.stderr.write(`
|
|
102
|
+
${m}
|
|
103
103
|
|
|
104
|
-
`),new dt(
|
|
105
|
-
`);return
|
|
104
|
+
`),new dt(g,r,`T\u1EA1o workspace remote th\u1EA5t b\u1EA1i (exit ${s.status}): ${g}.`,m)}let a=`git@github.com:${r}.git`,c=`https://github.com/${r}.git`;return o.success(`Workspace remote: ${a}`),{sshUrl:a,httpsUrl:c}}function Fi(t){let e=`git@github.com:${t.fullName}.git`,n=`https://github.com/${t.fullName}.git`;return rt("git",["-C",t.workspacePath,"remote","add","origin",e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]}).status!==0&&rt("git",["-C",t.workspacePath,"remote","set-url","origin",e],{stdio:"ignore"}),o.success(`Linked existing remote: ${e}`),{sshUrl:e,httpsUrl:n}}import{promises as kc}from"fs";import Wi,{join as Je}from"path";async function wc(t,e){let i=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!i?.[2])return!0;let r=i[2];if(r.startsWith("/"))return o.warn(`Pack hook reject: absolute path "${r}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let s=Je(t,r),a=Wi.resolve(t),c=Wi.resolve(s);return!c.startsWith(`${a}/`)&&c!==a?(o.warn(`Pack hook reject: path "${r}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await d(s)}function yc(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 Xe(t,e){let n=new Set,i=[];for(let r of[...t,...e]){let s=typeof r=="string"?r:JSON.stringify(r);n.has(s)||(n.add(s),i.push(r))}return i}function bc(t,e){let n=[],i={...e};for(let[r,s]of Object.entries(t)){let a=e[r]||[],c=Xe(a,s);c.length!==a.length&&n.push(r),i[r]=c}return{merged:i,touchedEvents:n}}async function ae(t){let e=Je(t,".claude","pack","templates","settings.json.tpl"),n=Je(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let i;try{let p=await I(e);i=JSON.parse(p)}catch(p){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${p.message}. Path: ${e}`)}let r={},s=!1;if(await d(n)){s=!0;try{r=await C(n)}catch(p){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${p.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...r};if(i.statusLine&&!r.statusLine&&(await wc(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 r.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=i.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),i.model&&!r.model&&(c.model=i.model,a.push("model added")),i.env){let p={...r.env||{}},m=!1;for(let[g,v]of Object.entries(i.env))g in p||(p[g]=v,m=!0);m&&(c.env=p,a.push("env vars added from pack"))}if(i.permissions){let p=r.permissions?.allow||[],m=r.permissions?.deny||[],g=i.permissions.allow||[],v=i.permissions.deny||[],P=Xe(p,g),B=Xe(m,v);(P.length!==p.length||B.length!==m.length)&&(c.permissions={allow:P,deny:B},a.push(`permissions union (+${P.length-p.length} allow, +${B.length-m.length} deny)`))}if(i.hooks){let p=r.hooks||{},{merged:m,touchedEvents:g}=bc(i.hooks,p);g.length>0&&(c.hooks=m,a.push(`hooks added for events: ${g.join(", ")}`))}if(a.length===0)return{action:"no-change",changes:[]};let l;return s&&(l=yc(n),await kc.copyFile(n,l)),await $(n,c),{action:"merged",backupPath:l,changes:a}}import{promises as ce}from"fs";import{dirname as Qe,join as zi,relative as Ze}from"path";import{promises as vc}from"fs";function xc(){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 qi(t){let e=`${t}.backup-${xc()}`;return await vc.rename(t,e),e}var tn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function Sc(t){try{return(await ce.lstat(t)).isSymbolicLink()}catch{return!1}}async function Cc(t,e,n){let i=Ze(Qe(e),e)||e;if(!await d(t))return{dir:i,action:"source-missing"};if(await d(e))if(await Sc(e))await ce.unlink(e);else if(n){let s=await qi(e),a=Ze(Qe(e),t);return await ce.symlink(a,e),{dir:i,action:"backed-up-and-linked",backupPath:s}}else return{dir:i,action:"skipped-conflict"};let r=Ze(Qe(e),t);return await ce.symlink(r,e),{dir:i,action:"created"}}async function le(t,e,n){let i=[];for(let r of tn){let s=zi(t,r),a=zi(e,r);i.push(await Cc(s,a,n))}return i}import{join as pe,relative as Tc,resolve as Qi}from"path";import{input as tr,select as Rc}from"@inquirer/prompts";import Ec from"boxen";import Ac from"boxen";var Yi=[{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 feature"},{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 feature 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 Ji(){let t=Math.max(...Yi.map(a=>a.cmd.length)),e=u.bold("\u{1F3AF} Slash commands t\u1EEB team-ai-pack"),n=u.dim("G\xF5 trong Claude Code session \u0111\u1EC3 g\u1ECDi capability c\u1EE7a pack:"),i=Yi.map(a=>` ${u.cyan(a.cmd.padEnd(t))} ${u.dim(a.desc)}`),r=u.dim("Catalog \u0111\u1EA7y \u0111\u1EE7 46 commands: cat .claude/pack/scripts/commands_data.yaml"),s=[e,n,"",...i,"",r].join(`
|
|
105
|
+
`);return Ac(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}import{readdir as Pc}from"fs/promises";import{join as $c}from"path";async function ue(t){if(!await d(t))return!0;try{return(await Pc(t)).filter(i=>!i.startsWith(".")&&i!=="Thumbs.db").length===0}catch{return!1}}async function Xi(t,e,n=10){for(let i=2;i<n;i++){let r=$c(t,`${e}-${i}`);if(await ue(r))return r}return null}function Zi(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}".
|
|
106
|
+
Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=Qi(pe(t,e)),i=Qi(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 It(t,e,n){Zi(t,e);let i=pe(t,e);if(await ue(i))return i;for(o.warn(`Workspace path "${i}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let r=await Xi(t,e);if(n&&r)return o.info(`--force: d\xF9ng ${r}`),r;let s=[];r&&s.push({name:`D\xF9ng "${r}" (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 Rc({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new w("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&r)return r;let c=await tr({message:"T\xEAn workspace m\u1EDBi:",validate:p=>{let m=p.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}});Zi(t,c.trim());let l=pe(t,c.trim());if(await ue(l))return l;o.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function me(t){return await tr({message:"Team owner email:",default:t})}async function er(t,e){if(e){o.warn("Skip commit (--no-commit). Ch\u1EA1y 'git status' + commit th\u1EE7 c\xF4ng sau.");return}let n=k(t);await n.add(["CLAUDE.md",".claude/",".gitignore",".gitmodules","notes/","scripts/"]),await n.commit("chore: initialize Avatar workspace"),o.success("\u0110\xE3 commit workspace")}function _c(t){if(t===null)return` ${u.yellow("AI:")} skipped \xB7 ${u.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${u.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${u.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${u.cyan("avatar ai setup")}`}function Ic(t){if(t===null)return` ${u.yellow("GitNexus:")} skipped \xB7 ${u.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"),` ${u.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${u.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${u.cyan("avatar gitnexus install")}`}async function nr(t,e,n=null,i=null){let r=[`${u.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Tc(process.cwd(),t)||t}`,` ${u.dim(`(flow: ${e})`)}`,_c(n),Ic(i),"",` ${u.cyan(`cd ${t}`)}`,` ${u.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${u.cyan("avatar commit src")} Commit code l\xEAn client remote`,` ${u.cyan("avatar sync")} Pull team-ai-pack m\u1EDBi`,` ${u.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${Ec(r.join(`
|
|
106
107
|
`),{padding:1,borderStyle:"round"})}
|
|
107
|
-
`);let s=
|
|
108
|
-
${
|
|
109
|
-
`)}async function tr(t,e){let i=(await k(t).getRemotes(!0)).find(l=>l.name==="origin");if(i?.refs.push)return o.success(`Folder \u0111\xE3 c\xF3 remote origin: ${i.refs.push}`),i.refs.push;if(!(e.createRemote??await Qi({message:"Folder ch\u01B0a c\xF3 remote. T\u1EA1o GitHub repo ngay \u0111\u1EC3 share team?",default:!0}))){o.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 it();let s=e.repoVisibility??await Qe({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),a=await Zi({message:"T\xEAn repo:",default:Rc(t)});return ut({folder:t,name:a,visibility:s,org:e.repoOrg}).sshUrl}async function Ze(t){await v(t.workspacePath),await k(t.workspacePath).init();let e=O(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await k(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 re(t.workspacePath,t.packVersion,t.ssoEmail,t.packLatest===!0&&!t.packVersion)).pinnedTag??"HEAD",e.succeed(`Pin team-ai-pack v\xE0o ${n}`)),await tn({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 tn(t){let e=Ke({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client"});await qn(t.workspacePath),await zn(t.workspacePath,e),await $e(t.workspacePath,e),await Yn(t.workspacePath,e),await Jn(t.workspacePath),await v(gt(t.workspacePath,"notes")),await v(gt(t.workspacePath,"scripts")),await St(gt(t.workspacePath,".git"),"post-merge"),await St(gt(t.workspacePath,".git","modules","src"),"pre-push"),o.success("C\xE0i post-merge (workspace) + pre-push (src/)"),await Ec(t.workspacePath),await h("init",`flow=${t.flow},workspace=${t.workspaceName}`),await Ji(t.workspacePath,t.skipCommit),await _c(t);let n=null;t.aiSkip?o.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau qua: avatar ai setup"):n=await Ut({workspacePath:t.workspacePath});let i=null;if(t.aiSkip||t.gitnexusSkip?t.gitnexusSkip?o.dim("B\u1ECF qua GitNexus setup (--gitnexus-skip). Setup sau: avatar gitnexus install"):o.dim("B\u1ECF qua GitNexus setup (auto-skip do --ai-skip)."):i=await Wt({workspacePath:t.workspacePath}),i?.ok){let s=Ke({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client",gitnexusReady:!0});await $e(t.workspacePath,s),o.dim("Updated CLAUDE.md v\u1EDBi GitNexus section")}await Xi(t.workspacePath,t.flow,n,i)}async function Ec(t){let e=gt(t,x);if(!await m(e)){o.dim("Pack submodule kh\xF4ng t\u1ED3n t\u1EA1i (skip auto-sync). C\xF3 th\u1EC3 ch\u1EA1y `avatar sync` sau.");return}let n=gt(t,".claude");o.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings merge)...");try{let i=await ae(e,n,!1),r=i.filter(c=>c.action==="created"||c.action==="updated").length,s=i.filter(c=>c.action==="source-missing").length;o.success(` \u2713 Symlinks: ${r} created${s>0?`, ${s} source-missing (pack thi\u1EBFu dir)`:""}`);let a=await oe(t);switch(a.action){case"merged":o.success(` \u2713 settings.json merged (${a.changes.join("; ")})`);break;case"no-change":o.dim(" - settings.json \u0111\xE3 sync, kh\xF4ng c\u1EA7n thay \u0111\u1ED5i.");break;case"no-pack-template":o.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}}catch(i){o.warn(`Auto-sync pack fail: ${i instanceof Error?i.message:i}. Ch\u1EA1y \`avatar sync\` th\u1EE7 c\xF4ng \u0111\u1EC3 retry.`)}}async function _c(t){if(t.skipCommit){o.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 Qi({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 Qe({message:"Workspace visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh, an to\xE0n)",value:"private"},{name:"public",value:"public"}]}));for(;;)try{await Vi({workspacePath:t.workspacePath,workspaceName:t.workspaceName,visibility:n,org:t.repoOrg});return}catch(i){if(i instanceof dt&&i.reason==="repo-exists"){let s=i.fullName,a=await Qe({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 w("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(a==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}if(a==="reuse"){Bi({workspacePath:t.workspacePath,fullName:s});return}let c=await Zi({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 r=await A({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(r==="abort")throw new w("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(r==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}}}async function er(t,e){let n=t.clientRepo??await Ot({message:"URL git c\u1EE7a repo:",validate:d=>d.length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),{resolvedRemoteUrl:i}=await it(n),r=i??n,s=t.teamOwner??await le(e),a=Di(r),c=t.workspaceName??await Ot({message:"T\xEAn workspace:",default:a}),l=ue(t.workspaceParent??"."),p=await It(l,c,t.force);await Ze({workspacePath:p,workspaceName:c,srcRemoteUrl:r,teamOwner:s,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho ${r}`,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 nr(t,e){let n=ue(t.folderPath??await Ot({message:"\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3:",validate:p=>p.length>0?!0:"Path b\u1EAFt bu\u1ED9c"}));await je(n,{presetStrategy:Fe(t),autoYes:t.yes});let i=await tr(n,t);if(i){let p=pt(i);p.ok||(o.warn(`Remote ${i} kh\xF4ng accessible (${p.reason??"unknown"}).`),i=(await Qt({url:i,initialReason:p.reason??"unknown",initialDetail:p.detail,folderPath:n,defaultVisibility:t.repoVisibility})).resolvedUrl)}let r=t.teamOwner??await le(e),s=t.workspaceName??`${Ic(n)}-avatar-workspace`,a=t.workspaceName??await Ot({message:"T\xEAn workspace:",default:s}),c=ue(t.workspaceParent??"."),l=await It(c,a,t.force);await Ze({workspacePath:l,workspaceName:a,srcRemoteUrl:i??n,teamOwner:r,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 ir(t,e){await it();let n=t.workspaceName??await Ot({message:"T\xEAn d\u1EF1 \xE1n:",validate:d=>d.length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),i=t.repoVisibility??await Nc({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),r=t.teamOwner??await le(e),s=ue(t.workspaceParent??"."),a=await It(s,n,t.force),c=Oc(a,"src");await v(a),await v(c),await je(c,{autoYes:!0});let l=ut({folder:c,name:n,visibility:i,org:t.repoOrg});await k(a).init();let p=O(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await k(a).subModule(["add",l.sshUrl,"src"]);let d="HEAD";t.skipTeamPack?p.succeed("Skip team-ai-pack (--skip-team-pack)"):(p.stop(),d=(await re(a,t.packVersion,e,t.latest===!0&&!t.packVersion)).pinnedTag??"HEAD",p.succeed(`Pin team-ai-pack v\xE0o ${d}`)),await tn({workspacePath:a,workspaceName:n,teamOwner:r,description:t.description??`D\u1EF1 \xE1n m\u1EDBi: ${n}`,packVersion:d,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(d){throw p.fail("Init workspace th\u1EA5t b\u1EA1i"),d}}import Hc from"boxen";import Dc from"open";var rr="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Gc="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",en="nal.vn",Lc=["openid","email","profile"],Mc="https://oauth2.googleapis.com/device/code",Uc="https://oauth2.googleapis.com/token",jc="https://oauth2.googleapis.com/revoke";async function or(){let t=new URLSearchParams({client_id:rr,scope:Lc.join(" ")}),e=await fetch(Mc,{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 sr(t){let e=new URLSearchParams({client_id:rr,client_secret:Gc,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Uc,{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 ar(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,"/"),r=Buffer.from(i,"base64").toString("utf8");return JSON.parse(r)}function cr(t){if(t.hd!==en)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${en}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}function lr(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 ur(t){let e=new URLSearchParams({token:t});await fetch(jc,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function pr(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",en),e.toString()}function mr(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 nn(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function nn(t){if(lt({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await hn(),await h("login_reset");else{let f=await q();if(f&&!z(f)){o.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${f.email}`);return}}let e=O("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await or(),e.succeed("Nh\u1EADn device code")}catch(f){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),f}let i=pr(n),r=[`1. Truy c\u1EADp: ${u.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${u.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${u.green("Allow")}...`].join(`
|
|
110
|
-
`);process.stdout.write(`${
|
|
111
|
-
`),
|
|
108
|
+
`);let s=pe(t,S);await d(s)&&process.stdout.write(`
|
|
109
|
+
${Ji()}
|
|
110
|
+
`)}async function or(t,e){let i=(await k(t).getRemotes(!0)).find(l=>l.name==="origin");if(i?.refs.push)return o.success(`Folder \u0111\xE3 c\xF3 remote origin: ${i.refs.push}`),i.refs.push;if(!(e.createRemote??await ir({message:"Folder ch\u01B0a c\xF3 remote. T\u1EA1o GitHub repo ngay \u0111\u1EC3 share team?",default:!0}))){o.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 it();let s=e.repoVisibility??await en({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),a=await rr({message:"T\xEAn repo:",default:Oc(t)});return ut({folder:t,name:a,visibility:s,org:e.repoOrg}).sshUrl}async function nn(t){await x(t.workspacePath),await k(t.workspacePath).init();let e=O(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await k(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 se(t.workspacePath,t.packVersion,t.ssoEmail,t.packLatest===!0&&!t.packVersion)).pinnedTag??"HEAD",e.succeed(`Pin team-ai-pack v\xE0o ${n}`)),await rn({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 rn(t){let e=ze({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client"});await Jn(t.workspacePath),await Xn(t.workspacePath,e),await _e(t.workspacePath,e),await Qn(t.workspacePath,e),await Zn(t.workspacePath),await x(gt(t.workspacePath,"notes")),await x(gt(t.workspacePath,"scripts")),await St(gt(t.workspacePath,".git"),"post-merge"),await St(gt(t.workspacePath,".git","modules","src"),"pre-push"),o.success("C\xE0i post-merge (workspace) + pre-push (src/)"),await Nc(t.workspacePath),await f("init",`flow=${t.flow},workspace=${t.workspaceName}`),await er(t.workspacePath,t.skipCommit),await Gc(t);let n=null;t.aiSkip?o.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau qua: avatar ai setup"):n=await jt({workspacePath:t.workspacePath});let i=null;if(t.aiSkip||t.gitnexusSkip?t.gitnexusSkip?o.dim("B\u1ECF qua GitNexus setup (--gitnexus-skip). Setup sau: avatar gitnexus install"):o.dim("B\u1ECF qua GitNexus setup (auto-skip do --ai-skip)."):i=await zt({workspacePath:t.workspacePath}),i?.ok){let s=ze({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client",gitnexusReady:!0});await _e(t.workspacePath,s),o.dim("Updated CLAUDE.md v\u1EDBi GitNexus section")}await nr(t.workspacePath,t.flow,n,i)}async function Nc(t){let e=gt(t,S);if(!await d(e)){o.dim("Pack submodule kh\xF4ng t\u1ED3n t\u1EA1i (skip auto-sync). C\xF3 th\u1EC3 ch\u1EA1y `avatar sync` sau.");return}let n=gt(t,".claude");o.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings merge)...");try{let i=await le(e,n,!1),r=i.filter(c=>c.action==="created"||c.action==="updated").length,s=i.filter(c=>c.action==="source-missing").length;o.success(` \u2713 Symlinks: ${r} created${s>0?`, ${s} source-missing (pack thi\u1EBFu dir)`:""}`);let a=await ae(t);switch(a.action){case"merged":o.success(` \u2713 settings.json merged (${a.changes.join("; ")})`);break;case"no-change":o.dim(" - settings.json \u0111\xE3 sync, kh\xF4ng c\u1EA7n thay \u0111\u1ED5i.");break;case"no-pack-template":o.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}}catch(i){o.warn(`Auto-sync pack fail: ${i instanceof Error?i.message:i}. Ch\u1EA1y \`avatar sync\` th\u1EE7 c\xF4ng \u0111\u1EC3 retry.`)}}async function Gc(t){if(t.skipCommit){o.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 ir({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 en({message:"Workspace visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh, an to\xE0n)",value:"private"},{name:"public",value:"public"}]}));for(;;)try{await Ki({workspacePath:t.workspacePath,workspaceName:t.workspaceName,visibility:n,org:t.repoOrg});return}catch(i){if(i instanceof dt&&i.reason==="repo-exists"){let s=i.fullName,a=await en({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 w("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(a==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}if(a==="reuse"){Fi({workspacePath:t.workspacePath,fullName:s});return}let c=await rr({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 r=await A({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(r==="abort")throw new w("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(r==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}}}async function sr(t,e){let n=t.clientRepo??await Ot({message:"URL git c\u1EE7a repo:",validate:m=>m.length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),{resolvedRemoteUrl:i}=await it(n),r=i??n,s=t.teamOwner??await me(e),a=Bi(r),c=t.workspaceName??await Ot({message:"T\xEAn workspace:",default:a}),l=de(t.workspaceParent??"."),p=await It(l,c,t.force);await nn({workspacePath:p,workspaceName:c,srcRemoteUrl:r,teamOwner:s,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho ${r}`,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 ar(t,e){let n=de(t.folderPath??await Ot({message:"\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3:",validate:p=>p.length>0?!0:"Path b\u1EAFt bu\u1ED9c"}));await Be(n,{presetStrategy:Ye(t),autoYes:t.yes});let i=await or(n,t);if(i){let p=pt(i);p.ok||(o.warn(`Remote ${i} kh\xF4ng accessible (${p.reason??"unknown"}).`),i=(await te({url:i,initialReason:p.reason??"unknown",initialDetail:p.detail,folderPath:n,defaultVisibility:t.repoVisibility})).resolvedUrl)}let r=t.teamOwner??await me(e),s=t.workspaceName??`${Lc(n)}-avatar-workspace`,a=t.workspaceName??await Ot({message:"T\xEAn workspace:",default:s}),c=de(t.workspaceParent??"."),l=await It(c,a,t.force);await nn({workspacePath:l,workspaceName:a,srcRemoteUrl:i??n,teamOwner:r,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 cr(t,e){await it();let n=t.workspaceName??await Ot({message:"T\xEAn d\u1EF1 \xE1n:",validate:m=>m.length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),i=t.repoVisibility??await Uc({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),r=t.teamOwner??await me(e),s=de(t.workspaceParent??"."),a=await It(s,n,t.force),c=Mc(a,"src");await x(a),await x(c),await Be(c,{autoYes:!0});let l=ut({folder:c,name:n,visibility:i,org:t.repoOrg});await k(a).init();let p=O(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await k(a).subModule(["add",l.sshUrl,"src"]);let m="HEAD";t.skipTeamPack?p.succeed("Skip team-ai-pack (--skip-team-pack)"):(p.stop(),m=(await se(a,t.packVersion,e,t.latest===!0&&!t.packVersion)).pinnedTag??"HEAD",p.succeed(`Pin team-ai-pack v\xE0o ${m}`)),await rn({workspacePath:a,workspaceName:n,teamOwner:r,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 p.fail("Init workspace th\u1EA5t b\u1EA1i"),m}}import qc from"boxen";import zc from"open";var sn="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",jc="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",on="nal.vn",Hc=["openid","email","profile"],Dc="https://oauth2.googleapis.com/device/code",Vc="https://oauth2.googleapis.com/token",Bc="https://oauth2.googleapis.com/revoke";async function lr(){let t=new URLSearchParams({client_id:sn,scope:Hc.join(" ")}),e=await fetch(Dc,{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 ur(t){let e=new URLSearchParams({client_id:sn,client_secret:jc,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Vc,{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 pr(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,"/"),r=Buffer.from(i,"base64").toString("utf8");return JSON.parse(r)}var Kc=new Set(["https://accounts.google.com","accounts.google.com"]),Fc=60;function Wc(t){if(!Kc.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==sn)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+Fc<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!==on)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${on}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}var mr=Wc;function dr(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 gr(t){let e=new URLSearchParams({token:t});await fetch(Bc,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function hr(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",on),e.toString()}function fr(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 an(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function an(t){if(lt({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await yn(),await f("login_reset");else{let g=await q();if(g&&!z(g)){o.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=O("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await lr(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let i=hr(n),r=[`1. Truy c\u1EADp: ${u.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${u.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${u.green("Allow")}...`].join(`
|
|
111
|
+
`);process.stdout.write(`${qc(r,{padding:1,borderStyle:"round"})}
|
|
112
|
+
`),zc(i).catch(()=>{o.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=O("\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 Yc(a);try{if(l=await ur(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 p=pr(l.id_token);try{mr(p)}catch(g){throw await gr(l.access_token),g}let m=dr(l,p);await wn(m),await f("login",m.email),o.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${m.email}`),o.success(`Verify hosted domain: ${p.hd} \u2713`),o.success(`L\u01B0u credential v\xE0o ${W} (chmod 600)`)}function Yc(t){return new Promise(e=>setTimeout(e,t))}function kr(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 Xc(e)}catch(n){n instanceof _t&&(o.dim(n.message),process.exit(0)),n instanceof nt&&(o.dim(n.message),process.exit(0)),n instanceof $t&&(o.dim(n.message),process.exit(0)),n instanceof w&&(o.dim(n.message),process.exit(0)),o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Xc(t){t.yes||lt({tagline:"Kh\u1EDFi t\u1EA1o Avatar trong d\u1EF1 \xE1n c\u1EE7a b\u1EA1n"}),t.mode&&o.warn("Flag --mode \u0111\xE3 deprecated t\u1EEB v1.1. D\xF9ng --project-status thay th\u1EBF.");let e=await q();for(;!e||z(e);){o.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y login flow tr\u01B0\u1EDBc khi init...");try{await an({})}catch(r){o.warn(`Login fail: ${r.message}`)}if(e=await q(),e&&!z(e))break;if(await A({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 w("User abort t\u1EA1i b\u01B0\u1EDBc login. Ch\u1EA1y 'avatar login' tay r\u1ED3i 'avatar init' l\u1EA1i.")}switch(t.projectStatus??await Qc()){case"existing-remote":await sr(t,e.email);break;case"existing-folder":await ar(t,e.email);break;case"new-project":await cr(t,e.email);break}}async function Qc(){return await Jc({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 y(t,e){return()=>{process.stdout.write(`${u.yellow("\u23F3")} ${u.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
|
|
112
113
|
`),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${u.cyan(e)}
|
|
113
114
|
`),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
|
|
114
|
-
`),process.exit(0)}}function
|
|
115
|
-
`):
|
|
115
|
+
`),process.exit(0)}}function wr(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(y("mcp-run","Milestone 09"))}import{join as Zc}from"path";import yr from"boxen";var tl=".claude/pack";function br(t){t.command("pack").description("Qu\u1EA3n l\xFD team-ai-pack submodule (status, ...)").command("status").description("Hi\u1EC3n th\u1ECB tag pack \u0111ang pin (default offline; --fetch \u0111\u1EC3 check remote)").option("--json","Output JSON cho script").option("--fetch","Fetch remote tags tr\u01B0\u1EDBc khi so s\xE1nh (ch\u1EADm th\xEAm 1-5s)").action(async n=>{try{let i=await el(process.cwd(),n.fetch===!0);n.json?process.stdout.write(`${JSON.stringify(i,null,2)}
|
|
116
|
+
`):nl(i)}catch(i){o.error(i instanceof Error?i.message:String(i)),process.exit(1)}})}async function el(t,e){let n=Zc(t,tl);if(!await d(n)||!await at(n))return{installed:!1,currentTag:null,latestTag:null,upToDate:null,fetched:!1};let i=await re(t).catch(()=>null),r=!1;if(e)try{await k(n).fetch(["--tags","origin"]),r=!0}catch{r=!1}let s=await G(n).catch(()=>[]),a=U(s);return{installed:!0,currentTag:i,latestTag:a,upToDate:i&&a?i===a:null,fetched:r}}function nl(t){if(!t.installed){let s=[`${u.bold("team-ai-pack")} \xB7 ${u.yellow("ch\u01B0a c\xE0i")}`,"\u2500".repeat(48),u.dim("Ch\u1EA1y `avatar init` ho\u1EB7c `avatar sync` \u0111\u1EC3 c\xE0i pack.")];process.stdout.write(`${yr(s.join(`
|
|
116
117
|
`),{padding:1,borderStyle:"round"})}
|
|
117
|
-
`);return}let e=t.currentTag??u.yellow("(unknown)"),n=t.latestTag??u.dim(t.fetched?"(no tags)":"(kh\xF4ng fetch)"),i;t.upToDate===!0?i=u.green("\u2713 \u0110ang d\xF9ng tag m\u1EDBi nh\u1EA5t"):t.upToDate===!1?i=`${u.yellow("\u26A0 C\xF3 version m\u1EDBi")} ${u.dim("\u2192 avatar sync")}`:i=u.dim("(kh\xF4ng so s\xE1nh \u0111\u01B0\u1EE3c)");let r=[`${u.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${u.dim("Tag hi\u1EC7n t\u1EA1i:")} ${e}`,`${u.dim("Tag m\u1EDBi nh\u1EA5t:")} ${n}`,`${u.dim("Tr\u1EA1ng th\xE1i:")} ${i}`];process.stdout.write(`${
|
|
118
|
+
`);return}let e=t.currentTag??u.yellow("(unknown)"),n=t.latestTag??u.dim(t.fetched?"(no tags)":"(kh\xF4ng fetch)"),i;t.upToDate===!0?i=u.green("\u2713 \u0110ang d\xF9ng tag m\u1EDBi nh\u1EA5t"):t.upToDate===!1?i=`${u.yellow("\u26A0 C\xF3 version m\u1EDBi")} ${u.dim("\u2192 avatar sync")}`:i=u.dim("(kh\xF4ng so s\xE1nh \u0111\u01B0\u1EE3c)");let r=[`${u.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${u.dim("Tag hi\u1EC7n t\u1EA1i:")} ${e}`,`${u.dim("Tag m\u1EDBi nh\u1EA5t:")} ${n}`,`${u.dim("Tr\u1EA1ng th\xE1i:")} ${i}`];process.stdout.write(`${yr(r.join(`
|
|
118
119
|
`),{padding:1,borderStyle:"round"})}
|
|
119
|
-
`)}function
|
|
120
|
-
`):
|
|
121
|
-
`).find(r=>r.trim()&&!r.startsWith("#")&&!r.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function
|
|
120
|
+
`)}function vr(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(y("restore","Milestone 08"))}function xr(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(y("review","Milestone 08"))}function Sr(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(y("scan","Milestone 06"))}function Cr(t){let e=t.command("secrets").description("Qu\u1EA3n l\xFD secrets trong OS keychain (M09)");e.command("list").description("Li\u1EC7t k\xEA secrets \u0111\xE3 set (ch\u1EC9 t\xEAn, kh\xF4ng value)").action(y("secrets list","Milestone 09")),e.command("set <service> <name>").description("Set/update secret (prompt \u1EA9n)").action(y("secrets set","Milestone 09")),e.command("get <service> <name>").description("L\u1EA5y secret, copy clipboard, auto-x\xF3a sau 30s").action(y("secrets get","Milestone 09")),e.command("rm <service> <name>").description("X\xF3a secret kh\u1ECFi keychain").action(y("secrets rm","Milestone 09")),e.command("check").description("Verify m\u1ECDi secret required b\u1EDFi MCP \u0111\xE3 enabled").action(y("secrets check","Milestone 09"))}import{promises as sl}from"fs";import{join as ge}from"path";import al from"boxen";import{promises as il}from"fs";import{join as rl}from"path";var ol="_backup";async function Ar(t){let e=rl(t,".claude",ol);return await d(e)?(await il.readdir(e,{withFileTypes:!0})).filter(i=>i.isDirectory()).map(i=>i.name).sort().reverse():[]}function Pr(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 cl(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
|
|
121
|
+
`):ul(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function cl(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=ge(t,".claude");if(!await d(n))return{projectName:e,cliVersion:_(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1};let r=ge(n,"_pending"),[s,a,c,l]=await Promise.all([(async()=>await at(ge(n,"pack"))?re(t).catch(()=>null):null)(),(async()=>await d(r)?(await sl.readdir(r)).filter(m=>m.endsWith(".diff.md")).length:0)(),Ar(t).then(p=>p.length).catch(()=>0),ll(n).catch(()=>"(read error)")]);return{projectName:e,cliVersion:_(),packVersion:s,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0}}async function ll(t){let e=ge(t,"project","tech-stack.md");return await d(e)?(await I(e)).split(`
|
|
122
|
+
`).find(r=>r.trim()&&!r.startsWith("#")&&!r.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function ul(t){let e=[`${u.bold("Avatar Status")} \xB7 ${u.cyan(t.projectName)}`,"\u2500".repeat(48),`${u.dim("CLI version:")} ${t.cliVersion}`,`${u.dim("Pack version:")} ${t.packVersion??u.yellow("not installed")}`,`${u.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?u.dim(" (avatar review)"):""}`,`${u.dim("Backups:")} ${t.backupCount}`,`${u.dim("Tech stack:")} ${t.techStackSummary}`];process.stdout.write(`${al(e.join(`
|
|
122
123
|
`),{padding:1,borderStyle:"round"})}
|
|
123
|
-
`)}import{join as
|
|
124
|
-
`).map(r=>r.trim()).filter(r=>r.length>0)}catch{return[]}}async function
|
|
125
|
-
Ch\u1EA1y 'avatar init' \u0111\u1EC3 add submodule, ho\u1EB7c 'git submodule update --init' n\u1EBFu \u0111\xE3 clone repo.`),process.exit(1));try{await k(i).fetch(["--tags","origin"])}catch(l){o.warn(`Kh\xF4ng fetch \u0111\u01B0\u1EE3c tags t\u1EEB origin (${l instanceof Error?l.message:l}). S\u1EBD d\xF9ng tag local hi\u1EC7n c\xF3.`)}let r=t.latest===!0&&!t.version,s=await G(i),a;if(r)a=`${
|
|
124
|
+
`)}import{join as Rr}from"path";import{join as $r}from"path";async function pl(t,e,n){let i=$r(t,n),r=$r(e,n);if(!await d(i))return"source-missing";if(!await d(r))return"needs-creation";let{promises:s}=await import("fs");return(await s.lstat(r)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function ml(t,e,n){try{return(await k(t).raw(["log","--oneline",`${e}..${n}`])).split(`
|
|
125
|
+
`).map(r=>r.trim()).filter(r=>r.length>0)}catch{return[]}}async function Tr(t,e,n){let i=await Bt(t),r=await ct(t),s=i??r.slice(0,7),a=await G(t),c=n??U(a)??"HEAD",l=await ml(t,r,c),p=[];for(let m of tn)p.push({dir:m,status:await pl(t,e,m)});return{currentVersion:s,targetVersion:c,commitsBehind:l,mountDirStatuses:p}}var he="main";async function dl(t){let e=process.cwd(),n=Rr(e,".claude"),i=Rr(e,S);await d(i)||(o.error(`team-ai-pack submodule ch\u01B0a \u0111\u01B0\u1EE3c kh\u1EDFi t\u1EA1o \u1EDF ${S}/.
|
|
126
|
+
Ch\u1EA1y 'avatar init' \u0111\u1EC3 add submodule, ho\u1EB7c 'git submodule update --init' n\u1EBFu \u0111\xE3 clone repo.`),process.exit(1));try{await k(i).fetch(["--tags","origin"])}catch(l){o.warn(`Kh\xF4ng fetch \u0111\u01B0\u1EE3c tags t\u1EEB origin (${l instanceof Error?l.message:l}). S\u1EBD d\xF9ng tag local hi\u1EC7n c\xF3.`)}let r=t.latest===!0&&!t.version,s=await G(i),a;if(r)a=`${he} (HEAD)`;else{let l=t.version??U(s);l||(o.error(`Kh\xF4ng t\xECm th\u1EA5y stable SemVer tag (vMAJOR.MINOR.PATCH) trong team-ai-pack submodule.
|
|
126
127
|
Tags hi\u1EC7n c\xF3: ${s.length>0?s.join(", "):"(none)"}
|
|
127
|
-
Pass --version <tag> r\xF5 r\xE0ng, ho\u1EB7c d\xF9ng --latest \u0111\u1EC3 pull HEAD branch ${
|
|
128
|
+
Pass --version <tag> r\xF5 r\xE0ng, ho\u1EB7c d\xF9ng --latest \u0111\u1EC3 pull HEAD branch ${he}.`),process.exit(1)),a=l}if(t.dryRun){let l=await Tr(i,n,a);if(o.info(`Pack version hi\u1EC7n t\u1EA1i: ${l.currentVersion}`),o.info(`Target version: ${l.targetVersion}`),l.commitsBehind.length===0)o.info("\u0110\xE3 \u1EDF target version, kh\xF4ng c\xF3 commit m\u1EDBi.");else{o.info(`Commits c\u1EA7n pull (${l.commitsBehind.length}):`);for(let p of l.commitsBehind.slice(0,20))console.log(` ${p}`);l.commitsBehind.length>20&&console.log(` ... v\xE0 ${l.commitsBehind.length-20} commits kh\xE1c`)}o.info(`
|
|
128
129
|
Mount dir statuses:`);for(let p of l.mountDirStatuses)console.log(` ${p.dir.padEnd(12)} ${p.status}`);o.info(`
|
|
129
|
-
Dry-run done. Kh\xF4ng apply thay \u0111\u1ED5i. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}if(r){o.info(`Pulling HEAD c\u1EE7a branch ${
|
|
130
|
-
${s.trimStart()}`.trim();a.length===0?await V(t,{force:!0}):await
|
|
131
|
-
`,"utf8")}async function
|
|
130
|
+
Dry-run done. Kh\xF4ng apply thay \u0111\u1ED5i. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}if(r){o.info(`Pulling HEAD c\u1EE7a branch ${he} (bleeding-edge mode)...`),await Vt(S,he,e);let l=await ct(i);o.dim(` HEAD = ${l.slice(0,7)}`),o.warn("\u26A0 --latest mode: workspace pin v\xE0o commit floating, kh\xF4ng reproducible. Chuy\u1EC3n v\u1EC1 tag stable: avatar sync (no flag).")}else o.info(`Checking out ${a} trong submodule...`),await vt(S,a,e);o.info("Creating symlink farm...");let c=await le(i,n,t.force===!0);gl(c,t.force===!0),o.info("Merging pack settings.json template into project settings.json...");try{let l=await ae(e);switch(l.action){case"merged":o.success(` \u2713 settings.json merged (${l.changes.join("; ")}). Backup: ${l.backupPath??"n/a"}`);break;case"no-change":o.info(" - settings.json \u0111\xE3 sync v\u1EDBi pack, kh\xF4ng c\xF3 thay \u0111\u1ED5i.");break;case"no-pack-template":o.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}}catch(l){o.warn(` ! Merge settings.json fail: ${l instanceof Error?l.message:l}. Symlinks \u0111\xE3 t\u1EA1o OK, hooks c\xF3 th\u1EC3 ch\u01B0a active. Manual merge n\u1EBFu c\u1EA7n.`)}o.success(`Synced team-ai-pack to ${a}.`)}function gl(t,e){for(let i of t)switch(i.action){case"created":o.info(` \u2713 ${i.dir} \u2192 symlinked (new)`);break;case"updated":o.info(` \u2713 ${i.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":o.info(` \u2713 ${i.dir} \u2192 symlinked (backup: ${i.backupPath})`);break;case"source-missing":o.warn(` - ${i.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":o.warn(` ! ${i.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override (s\u1EBD gi\u1EEF data \u1EDF ${i.dir}.backup-<timestamp>/).`);break}let n=t.filter(i=>i.action==="skipped-conflict").length;n>0&&!e&&o.warn(`${n} mount dir(s) b\u1ECB skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force n\u1EBFu mu\u1ED1n override (backup t\u1EF1 \u0111\u1ED9ng).`)}function Er(t){t.command("sync").description("Pull team-ai-pack m\u1EDBi nh\u1EA5t + t\u1EA1o symlink farm v\xE0o .claude/").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--version <tag>","Pin v\xE0o version c\u1EE5 th\u1EC3 (vd: v0.2.0)").option("--latest","Bleeding-edge: pull HEAD c\u1EE7a main branch (b\u1ECF qua tag SemVer)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply thay \u0111\u1ED5i").action(dl)}function _r(t){let e=t.command("tools").description("Qu\u1EA3n l\xFD system tools + MCP servers (M09)");e.command("list").description("Li\u1EC7t k\xEA tool \u0111\xE3 c\xE0i / c\xF2n thi\u1EBFu").option("--installed","Ch\u1EC9 li\u1EC7t k\xEA tool \u0111\xE3 c\xE0i").option("--missing","Ch\u1EC9 li\u1EC7t k\xEA tool c\xF2n thi\u1EBFu").option("--json","Output JSON cho script").action(y("tools list","Milestone 09")),e.command("install [tool-ids...]").description("C\xE0i tool v\xE0 \u0111\u0103ng k\xFD v\xE0o ~/.claude.json").option("--all-recommended","C\xE0i m\u1ECDi MCP \u0111\u01B0\u1EE3c recommend cho project type").option("--verify","Ch\u1EA1y MCP th\u1EED \u0111\u1EC3 verify (m\u1EA5t ~30s/tool)").option("--no-secrets","Skip prompt secrets, set sau qua 'avatar secrets'").action(y("tools install","Milestone 09")),e.command("remove <tool-id>").description("G\u1EE1 tool kh\u1ECFi ~/.claude.json (optional uninstall binary)").option("--keep-secrets","Kh\xF4ng x\xF3a secrets kh\u1ECFi keychain").option("--keep-binary","Kh\xF4ng uninstall npm global binary").action(y("tools remove","Milestone 09"))}import{relative as xl}from"path";import{confirm as Sl}from"@inquirer/prompts";import Cl from"boxen";import{cp as fe,mkdir as Ir,writeFile as hl}from"fs/promises";import{homedir as fl}from"os";import{basename as kl,join as j}from"path";var wl=j(fl(),".avatar","uninstall-backups");async function Or(t,e,n){let i=kl(t),r=new Date().toISOString().replace(/[:.]/g,"-"),s=j(wl,`${i}-${r}`);if(await Ir(s,{recursive:!0,mode:448}),e.claudeDir&&await fe(e.claudeDir,j(s,".claude"),{recursive:!0}),e.claudeMd&&await fe(e.claudeMd,j(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=j(s,"hooks");await Ir(c,{recursive:!0}),e.postMergeHook&&await fe(e.postMergeHook,j(c,"post-merge")),e.prePushHook&&await fe(e.prePushHook,j(c,"pre-push"))}let a={projectName:i,projectPath:t,timestamp:r,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await hl(j(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as yl}from"fs";import{join as H}from"path";function D(t){return yl(t)?t:null}function Nr(t){let e=D(H(t,".claude")),n=D(H(t,"CLAUDE.md")),i=D(H(t,".git","hooks","post-merge")),r=D(H(t,".git","modules","src","hooks","pre-push")),s=D(H(t,".gitignore")),a=D(H(t,".gitmodules")),c=D(H(t,"notes")),l=D(H(t,"scripts"));return{hasAnyArtifact:!!(e||n||i||r),claudeDir:e,claudeMd:n,postMergeHook:i,prePushHook:r,gitignorePath:s,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as Gr,rm as V,writeFile as Lr}from"fs/promises";async function Mr(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:i}=await import("path"),r=await n(t.claudeDir);for(let s of r)s!=="pack"&&await V(i(t.claudeDir,s),{recursive:!0,force:!0})}else await V(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await V(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await V(t.postMergeHook,{force:!0}),t.prePushHook&&await V(t.prePushHook,{force:!0})),t.gitignorePath&&await bl(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await vl(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:i}=await import("fs/promises");(await i(n)).length===0&&await V(n,{recursive:!0,force:!0})}}async function bl(t){let e=await Gr(t,"utf8"),n=e.indexOf(Et),i=e.indexOf(et);if(n===-1||i===-1)return;let r=e.slice(0,n),s=e.slice(i+et.length),a=`${r.trimEnd()}
|
|
131
|
+
${s.trimStart()}`.trim();a.length===0?await V(t,{force:!0}):await Lr(t,`${a}
|
|
132
|
+
`,"utf8")}async function vl(t,e){let i=(await Gr(t,"utf8")).split(`
|
|
132
133
|
`),r=[],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||r.push(c)}let a=r.join(`
|
|
133
|
-
`).trim();a.length===0?await V(t,{force:!0}):await
|
|
134
|
-
`,"utf8")}function
|
|
134
|
+
`).trim();a.length===0?await V(t,{force:!0}):await Lr(t,`${a}
|
|
135
|
+
`,"utf8")}function Ur(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 Al(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Al(t){let e=process.cwd(),n=Nr(e);if(!n.hasAnyArtifact){o.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Pl(e,n,t),t.dryRun){o.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Sl({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){o.info("\u0110\xE3 h\u1EE7y.");return}let i=null;t.noBackup||(i=await Or(e,n,_()),o.success(`Backup t\u1EA1o t\u1EA1i: ${i}`)),await Mr(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await f("uninstall",`project=${e},backup=${i??"skipped"}`),$l(i)}function Pl(t,e,n){o.info(`Project: ${t}`),o.plain(""),o.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&o.plain(` ${u.red("\u2717")} ${xl(t,e.claudeDir)||".claude/"}`),e.claudeMd&&o.plain(` ${u.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&o.plain(` ${u.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&o.plain(` ${u.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&o.plain(` ${u.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&o.plain(` ${u.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),o.plain(""),o.plain("Kh\xF4ng \u0111\u1EE5ng:"),o.plain(` ${u.green("\u2713")} src/ (code kh\xE1ch)`),o.plain(` ${u.green("\u2713")} Git history`),o.plain(` ${u.green("\u2713")} ~/.avatar/config.json (token SSO)`),o.plain(` ${u.green("\u2713")} Secrets trong keychain`),o.plain("")}function $l(t){let e=[`${u.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${u.dim("Backup:")} ${t}`),e.push(` ${u.dim("Restore:")} ${u.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${Cl(e.join(`
|
|
135
136
|
`),{padding:1,borderStyle:"round"})}
|
|
136
|
-
`)}var
|
|
137
|
-
${
|
|
137
|
+
`)}var cn=_(),b=new Tl;b.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(cn,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
|
|
138
|
+
${Me({tagline:`v${cn} \xB7 AI harness CLI for NAL Vietnam`})}
|
|
138
139
|
|
|
139
|
-
`);var
|
|
140
|
+
`);var Rl=process.argv.includes("-v")||process.argv.includes("--version");Rl&&(lt({tagline:`v${cn} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));fr(b);kr(b);Er(b);Sr(b);xr(b);Pr(b);ti(b);vr(b);Vn(b);_r(b);Cr(b);wr(b);jn(b);bi(b);br(b);Ur(b);b.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}
|
|
140
141
|
`),process.exit(1)});
|
|
141
142
|
//# sourceMappingURL=index.js.map
|