cli-z-develop 0.0.56 → 0.0.58

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.
Files changed (2) hide show
  1. package/dist/index.js +1 -1
  2. package/package.json +9 -7
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- import{program as e}from"commander";import t from"node:fs";import n from"node:path";import a,{AxiosError as i}from"axios";import s from"chalk";import r from"dayjs";import o from"shelljs";import c from"node:os";import l from"child_process";import{input as m,confirm as u,select as p,password as d,checkbox as g}from"@inquirer/prompts";import f from"ora";import h from"fs-extra";import{select as w}from"inquirer-select-pro";import v from"lint-staged";import y from"minimist";const b=".z",j="develop-config.json",$=".z",k=".commit-msg-tpl",x="project.json";function S(e=""){return n.join($,e)}function O(){return S(x)}function z(){return S(k)}function N(){l.spawnSync("git",["config","core.hooksPath",S()]),l.spawnSync("git",["config","commit.template",z()])}function T(e=""){return n.resolve(c.homedir(),b,e)}function E(){return T(j)}function _(){return t.existsSync(T())&&t.existsSync(E())}const{red:D,green:I,blue:A,magenta:B}=s;function U(...e){console.log(I(...e))}function C(e,t=!1){let n=e;e instanceof Error?(n=e.message,a.isAxiosError(e)&&(n=`请求失败:${e.message}`),console.log(D(n)),console.log(B(e.stack))):console.log(D(e)),t||process.exit(1)}function W(e){return!!t.existsSync(e)&&t.lstatSync(e).isDirectory()}function J(e=process.cwd()){if(W(e)){return t.readdirSync(e).filter((t=>W(n.resolve(e,t))))}return[]}function P(e=process.cwd()){if(W(e)){return t.readdirSync(e).filter((t=>!W(n.resolve(e,t))))}return[]}async function q(e,t={removeTailLinkBreak:!0,silent:!0}){let n=await new Promise(((n,a)=>{try{n(o.exec(e,{silent:t.silent}))}catch(e){a(e)}}));return n=n.toString(),t.removeTailLinkBreak&&(n=n.replace(/\n$/,"")),n}function L(){return r(Date.now()).format("YYMMDD")}function R(){t.existsSync(S())||C("当前不在项目根目录。请切换到项目根目录")}function F(){t.existsSync(n.resolve(".git"))||C("当前不是git项目根目录,请先执行git init,或切换到根目录")}async function M(){""!==await q("git status -s")&&C("请先提交代码变动,再进行操作")}function G(e){return["fe-biz"].includes(e)}const Q=[{name:"开发环境 - dev",value:"dev"},{name:"测试环境 - test",value:"test"},{name:"预发环境 - release",value:"release"},{name:"正式环境 - production",value:"production"}],V=[{name:"网页 - h5",value:"h5"},{name:"依赖包 - npm",value:"npm"}],H=["master","dev","test","release"],K={1:"研发",2:"测试",3:"产品",4:"设计",5:"运营",6:"销售",7:"行政",8:"财务",9:"其他"},Z="http://git.cxlqd.com",Y={PROJECT_NAME:"fe-pages-tpl",GIT_URL:`${Z}/fe-component/fe-pages-tpl.git`},X=["fe-biz","fe-base","fe-tpl","fe-component","fe-demo"],ee="fe-biz7tvsd";var te={name:"cli-z-develop",version:"0.0.56",description:"前端本地开发命令行工具",main:"dist/index.js",bin:{z:"bin/z.js","z-develop":"bin/z.js"},scripts:{prepare:"[ -n '$z' ] && z init prepare || echo 'Warning: z not exist at global'",dev:"rollup -c -w --environment DEBUG:true",build:"rollup -c",eslint:"eslint '**/*.{ts,js}' --fix",prettier:"prettier -wu .",upload:"npm run build && npm publish --access public --registry https://registry.npmjs.org/","upload:patch":"npm version patch --no-commit-hooks && npm run upload","upload:minor":"npm version minor --no-commit-hooks && npm run upload","upload:major":"npm version major --no-commit-hooks && npm run upload"},type:"module",repository:{type:"git",url:"http://git.cxlqd.com/fe-base/cli-z-develop.git"},author:"z",devDependencies:{"@lonely9/eslint-config-team":"^1.2.1","@rollup/plugin-json":"^6.1.0","@rollup/plugin-node-resolve":"^16.0.1","@rollup/plugin-replace":"v6.0.2","@rollup/plugin-terser":"^0.4.4","@types/fs-extra":"^11.0.4","@types/inquirer":"^9.0.7","@types/minimist":"^1.2.5","@types/node":" ^22.14.1","@types/shelljs":"^0.8.15","@typescript-eslint/eslint-plugin":"^8.30.1","@typescript-eslint/parser":"^8.30.1",eslint:"^9.24.0",jiti:"^2.4.2",prettier:"^3.5.3",rollup:" 4.40.0","rollup-plugin-typescript2":"^0.36.0",typescript:" ^5.8.3"},dependencies:{"@inquirer/prompts":"^7.4.1",axios:"^1.8.4",chalk:"^5.4.1",commander:"^13.1.0",dayjs:"^1.11.13",figures:"^6.1.0","fs-extra":"^11.3.0","inquirer-select-pro":"^1.0.0-alpha.9","lint-staged":"^15.5.1",minimist:"^1.2.8",ora:"^8.2.0",rxjs:"^7.8.2",shelljs:"^0.9.2"}};const ne={profile:{ldapAccount:"",ldapPassword:"",gitToken:"",gitUserId:0,gitName:"",gitEnglishName:"",gitEmail:"",weWorkName:"",weWorkUserId:"",jobType:"fe",zenTaoToken:"",k8sToken:""},main:{version:te.version,latestCheckVersionTimestamp:0,versionCheckDuring:3,weWorkListCache:[]},constants:{ZenTaoDomain:"",K8sDomain:"",FEServerDomain:""}};function ae(e){return e?ne.profile[e]:ne.profile}function ie(e){return e?ne.main[e]:ne.main}function se(e){return e?ne.constants[e]:ne.constants}let re=null,oe=null;function ce(){if(re)return re;const e=n.join("package.json");return t.existsSync(e)||C(`当前目录(${o.pwd()})不存在${e}文件,请在项目根目录执行该命令。`),re=h.readJsonSync(e),re}function le(){if(oe)return oe;const e=O();return t.existsSync(e)||C(`当前目录(${o.pwd()})不存在${e}文件,请在项目根目录执行该命令,或者初始化项目(z init .)。`),oe=h.readJsonSync(e),oe}function me(e){oe=oe?Object.assign(Object.assign({},oe),e):e,h.writeJSONSync(O(),oe,{spaces:2})}const ue={id:0,name:"",group:"",sourceBranch:"",mergeRequestUrl:""};async function pe(){if(!ue.name){const e=ce();ue.name=e.name}if(!ue.group){const e=le();e.repository.group&&(ue.group=e.repository.group)}if(!ue.id)try{const e=ue.group?`${ue.group}/${ue.name}`:ue.name,[t]=await _e(e);ue.mergeRequestUrl=`${t.web_url}/merge_requests`,ue.id=t.id,ue.group=t.namespace.path;const n=le();n.repository.group||me({repository:Object.assign(Object.assign({},n.repository),{group:t.namespace.path})})}catch(e){C(e)}return ue.sourceBranch||(ue.sourceBranch=await q("git branch --show-current")),ue}function de(){h.writeJSONSync(E(),{main:ie(),profile:ae(),constants:se()},{spaces:2})}async function ge(e){const a=T(),i=process.cwd(),s=n.join(a,Y.PROJECT_NAME);t.existsSync(s)?(o.cd(s),await q("git pull"),o.cd(i)):(o.cd(a),await q(`git clone ${Y.GIT_URL} --depth=1`),o.cd(i));const r=n.join(s,"src",e);return t.existsSync(r)||C("模板库错误!"),r}function fe(e){return e.split("-").map((e=>e.charAt(0).toUpperCase()+e.slice(1))).join("")}function he(e,n){const a=t.readFileSync(e,"utf-8"),i=n.replace(/\//g,"-"),s=a.replace(/name: "-_-NAME-_-"/g,`name: "${fe(i)}"`).replace(/class="-_-NAME-_-"/g,`class="${i}"`).replace(/.-_-NAME-_-/g,`.${i}`);t.writeFileSync(e,s,"utf-8")}function we(e,t){void 0!==t?ne.profile[e]=t:ne.profile=Object.assign(Object.assign({},ne.profile),e)}function ve(e,t){ne.constants=Object.assign(Object.assign({},ne.constants),e)}function ye(e,t){void 0!==t?ne.main[e]=t:ne.main=Object.assign(Object.assign({},ne.main),e)}async function be(e){var t;const n=e.method||"get",r=e.headers||{},o=e.data||{},c=e.dataKey||["GET","get"].includes(n)?"params":"data";try{const t={url:e.url,method:n,[c]:o,headers:r};s.magenta(t.method.toUpperCase()),s.yellow(t.url),JSON.stringify(t.headers),s.gray(JSON.stringify(t[c],null,2));const i=await a(t);return s.green("Response"),s.grey(JSON.stringify(i.data,null,2)),Promise.resolve(i.data)}catch(e){return e instanceof i&&(s.red("Error"),s.grey(JSON.stringify(null===(t=null==e?void 0:e.response)||void 0===t?void 0:t.data))),Promise.reject(e)}}async function je(){const e=ae("gitToken");if(e)return e;{const{access_token:e}=await be({url:`${Z}/oauth/token`,method:"post",data:{grant_type:"password",username:ae("ldapAccount"),password:ae("ldapPassword")}}),t=`Bearer ${e}`;return we("gitToken",t),de(),t}}async function $e(e){return be(Object.assign(Object.assign({},e),{headers:{Authorization:await je()}}))}async function ke(){const e=ae("zenTaoToken");if(e)return e;{const{token:e}=await be({url:`${se("ZenTaoDomain")}/api.php/v1/tokens`,method:"post",data:{account:ae("ldapAccount"),password:ae("ldapPassword")}});return we("zenTaoToken",e),de(),e}}async function xe(e){const t=await be(Object.assign(Object.assign({},e),{headers:{Token:await ke()}}));if("string"==typeof t){const e=JSON.parse(t.slice(0,t.indexOf("<script>")));return JSON.parse(e.data)}if("object"==typeof t)return"Unauthorized"===t.error||t.data.indexOf("<script>")<0?(we("zenTaoToken",""),xe(Object.assign({},e))):t}async function Se(){const e=ae("k8sToken");if(e)return e;{const{access_token:e}=await be({url:`${se("K8sDomain")}/oauth/login/LDAP`,method:"post",headers:{"Content-Type":"application/x-www-form-urlencoded"},data:{username:ae("ldapAccount"),password:ae("ldapPassword")}}),t=`Bearer ${e}`;return we("k8sToken",t),de(),t}}async function Oe(e){const t=await be(Object.assign(Object.assign({},e),{headers:{Authorization:await Se()}}));return 401===t.code?(we("k8sToken",""),Oe(Object.assign({},e))):t}const ze=()=>`${Z}/api/v4`,Ne=e=>`${ze()}/projects/100/repository/files/${encodeURIComponent(e)}/raw?ref=master`;function Te(){return $e({url:Ne("src/data/template-projects.json")})}function Ee(){return $e({url:Ne("src/data/z-develop-config.json")})}function _e(e){return $e({url:`${ze()}/projects`,data:{search:e,search_namespaces:!0}})}async function De(){const e=await $e({url:`${ze()}/groups`}),t=X.map((t=>{const n=e.find((e=>e.name===t));return!!n&&{name:n.name,id:n.id,description:n.description}})).filter((e=>Boolean(e)));h.writeJSONSync(T("fe-groups.json"),{groups:t},{spaces:2})}async function Ie(e){const t=`${se("FEServerDomain")}/api`,{data:n}=await be({url:`${t}/auth/z-develop/login`,method:"post"}),a=await be({url:t+e.url,headers:{Authorization:`Bearer ${n}`},data:e.data,method:e.method||"post"});return 0!==a.code&&C(`${a.code}: ${a.message}`),a.data}function Ae(e){return Ie({url:"/wework-approval/approval/detail",data:e})}async function Be(){const e=(await Ie({url:"/user/list2"})).filter((e=>[1,2,3].includes(e.title))).map((e=>({name:`${e.nick} - ${K[e.title]}`,value:e.weWorkUserId}))),t=ie("weWorkListCache"),n=t.map((e=>e.value)),a=[];return e.forEach((e=>{const i=n.indexOf(e.value);a.push(Object.assign(Object.assign({},e),{usageCount:i>-1?t[i].usageCount:0}))})),a.sort(((e,t)=>t.usageCount-e.usageCount))}async function Ue(e,t,n){const a=await Be();let i=await w({message:e,loop:!0,clearInputWhenSelected:!0,multiple:n,options:e=>e?a.filter((t=>t.name.includes(e))):a,validate:t});return i=Array.isArray(i)?i:[i],function(e,t){const n=[];t.forEach((t=>{e.includes(t.value)&&n.push(t)}));const a=ie("weWorkListCache"),i=a.map((e=>e.value));n.forEach((e=>{const t=i.indexOf(e.value);t>-1?a[t].usageCount+=1:a.push(Object.assign(Object.assign({},e),{usageCount:1}))})),ye({weWorkListCache:a.sort(((e,t)=>t.usageCount-e.usageCount))}),de()}(i,a),i}async function Ce(){return await m({message:"请输入的企微审批编号"})}async function We(e,t){const{status:n,applierId:a,form:i}=await(s={id:e},Ie({url:"/wework-approval/approval/detail",data:s}));var s;2!==n&&C("审批单的状态不是已通过"),a!==ae("weWorkUserId")&&C("提交审批者与当前执行命令的用户不是同一人");const r={projectName:"",branch:"",rollback:""};i.forEach((({label:e,value:t})=>{"项目"===e&&(r.projectName=t),"分支"===e&&(r.branch=t),"回滚"===e&&(r.rollback=t)})),t.name!==r.projectName&&C("审批单中的项目与当前项目不是同一个项目"),t.branch&&t.branch!==r.branch&&C("审批单中的分支与当前分支不是同一个项目"),t.rollback&&"否"===r.rollback&&C("审批单中没有选择回滚")}async function Je(){try{if(_()){await u({message:"系统中已存在z的配置文件,确认重新配置?"})?o.rm("-rf",E()):process.exit(0)}t.mkdirSync(T(),{recursive:!0});const e=await p({message:"请选择岗位类型",choices:[{name:"前端",value:"fe"},{name:"后端",value:"be"}]}),n=await m({message:"请输入LDAP账号:"}),a=await d({message:"请输入LDAP密码:",mask:!0});we("jobType",e),we("ldapAccount",n),we("ldapPassword",a);ve(await Ee());const i=await Be(),s=await Ue("请选择你自己(用于企微通知):",(e=>e.length>1?"只能选一个":!(e.length<1)||"请选一个"),!1),{name:r,value:c}=i.find((e=>e.value===s[0]));we({weWorkName:r,weWorkUserId:c})}catch(e){C(e)}const e=f("配置信息初始化中").start();try{const t=await $e({url:`${ze()}/user`});we({gitUserId:t.id,gitName:t.name,gitEnglishName:t.username,gitEmail:t.email}),ye("latestCheckVersionTimestamp",Date.now()),de(),await De(),e.succeed("配置信息初始化完成"),process.exit(0)}catch(t){e.fail("配置信息初始化失败"),a.isAxiosError(t)&&C("请检查你的域名及令牌配置"),C(t)}}const Pe='{\n "printWidth": 120,\n "tabWidth": 2,\n "useTabs": false,\n "semi": true,\n "singleQuote": false,\n "quoteProps": "as-needed",\n "jsxSingleQuote": false,\n "trailingComma": "all",\n "bracketSpacing": true,\n "bracketSameLine": false,\n "arrowParens": "always",\n "requirePragma": false,\n "insertPragma": false,\n "proseWrap": "preserve",\n "htmlWhitespaceSensitivity": "css",\n "vueIndentScriptAndStyle": false,\n "endOfLine": "lf",\n "embeddedLanguageFormatting": "auto",\n "singleAttributePerLine": false\n}\n',qe="# 系统 信息文件\n.DS_Store/\nThumbs.db\n\n# 编辑器配置文件\n.idea/\n.vscode/\n\n# 输出目录\ndist/\n\n# Log files\n*.log\n\n# 本地配置文件\n*.local\n\n# 缓存文件\n*.cache\n\n# 垃圾文件\n.Trashes\n",Le={Java:{"**/*.{java}":"./run.sh pmd -d ../../../src/main/java/ -f text -R rulesets/java/quickstart.xml"},JavaScript:{"**/*.{vue,js,jsx,cjs,mjs,ts,tsx,cts,mts}":"eslint --fix","*":"prettier -wu"}};let Re;async function Fe(e,n){if(e)o.rm("-rf",$);else if(t.existsSync(S())){await u({message:`当前项目中已存在配置文件夹${$},确认重新配置?`})?o.rm("-rf",$):process.exit(0)}t.mkdirSync(S(),{recursive:!0}),await async function(e,t){const n=await q("git remote get-url --push origin");n||C("获取项目远程地址失败,请配置后重试");let a=e;a||(a=await p({message:"请选择项目语言",choices:[{name:"JavaScript",value:"JavaScript"},{name:"Java",value:"Java"}]}));let i=t;if(!i){const e=ce(),t=await _e(e.name);1===t.length?i=t[0].namespace.path:t.length>1&&(i=await p({message:"请选择当前项目所属分组",choices:t.map((e=>({name:e.namespace.path,value:e.namespace.path})))}))}Re=f("项目初始化中").start(),me({language:a,"lint-staged":Le[a],repository:{url:n,group:i}})}(e,n),await(t.writeFileSync(z(),"",{mode:493}),t.writeFileSync(S("commit-msg"),"#!/usr/bin/env sh\nz lint commit-msg",{mode:493}),t.writeFileSync(S("pre-commit"),"#!/usr/bin/env sh\nz lint commit-files",{mode:493}),t.writeFileSync(S(".prettierrc.json"),Pe,{mode:493}),t.writeFileSync(S(".prettierignore"),qe,{mode:493}),t.writeFileSync(S(".gitignore"),".commit-msg-tpl\n",{mode:493}),void N()),null==Re||Re.succeed("初始化完成。")}async function Me(e){if(!e)return"不能为空";if(!/^[a-z0-9-]+$/.test(e))return"格式为小写字母、中横线(可选)、数字(不推荐)。如apple, apple-tree";if(J().includes(e))return"当前目录下已存在同名文件夹,请先处理。";return!(await _e(e)).find((t=>t.name===e))||"远程仓库中已存在同名项目"}function Ge(e){return!!e||"不能为空"}function Qe(e){return e?!!/^[a-zA-Z0-9.]+$/.test(e)||"格式为大小写字母、数字、小数点,小驼峰命名。如userInfo、systemRouter3":"不能为空"}function Ve(e){if(!e)return"目录名称不能为空";if(!/^[a-z][a-z0-9\-/]*$/.test(e))return"支持小写字母、数字,连字符(-)命名,小写字母开头,多层目录使用/隔开。如user、modules-3、setting/profile";if(e.includes("/")){const t=e.split("/");for(const e of t){if(!e)return"子级目录/页面名称不能为空";if(!/^[a-z][a-z0-9-]*$/.test(e))return"支持小写字母、数字,连字符(-)命名,小写字母开头。如user、modules-3"}}let a,i;a=e.includes("/")?n.resolve("src","pages",...e.split("/")):n.resolve("src","pages",e);const s=n.resolve(a,"..");return i=!!t.existsSync(s)&&t.readdirSync(s).map((e=>{const t=e.lastIndexOf(".");return t>-1?e.slice(0,t):e})).includes(e),!i||"本地目录已存在该文件"}e.command("init").alias("i").argument("[type]","非必填。不传为初始化工具配置;传 . 为初始化当前项目。").description("初始化工具配置、项目配置").action((async function(e){e?"."===e?(F(),await Fe(void 0,void 0),process.exit(0)):"prepare"===e?(F(),await async function(){t.existsSync(S())&&t.existsSync(z())?N():await Fe(),process.exit(0)}()):C("参数错误。执行 z init -h 查看帮助。"):await Je()}));const He=e=>`${se("ZenTaoDomain")}${e}`;function Ke(e,n){let a="";if(["feat","fix","refactor"].includes(e)){a=`${e}(${n.map((e=>e.value)).join(",")}): ${n.map((e=>e.name)).join(";")}`}"chore"===e&&(a=`${e}: ${n}`),t.writeFileSync(z(),a,"utf-8"),console.log(s.yellow(a)),U("commit msg模板写入成功,可以进行提交了")}async function Ze(){const e=await p({message:"请选择你要创建的模板类型",choices:[{name:"feat - 业务需求开发",value:"feat"},{name:"fix - bug修复",value:"fix"},{name:"refactor - 技术内部需求",value:"refactor"},{name:"chore - 其他",value:"chore"}]});if(["feat","refactor"].includes(e)){const e=await async function(){const e=await xe({url:He("/my-work-task.json?tid=mrrferp8"),method:"get"});return null==e?void 0:e.tasks.filter((e=>"done"!==e.status)).map((({id:e,name:t})=>({value:{name:t,value:e},name:t})))}();Ke("feat",await g({message:"请关联开发任务(可多选):",validate:e=>0!==e.length||"请选择项",choices:e.map((e=>Object.assign(Object.assign({},e),{name:`[${e.value.value}]: ${e.name}`})))}))}if("fix"===e){const e=await async function(){const e=await xe({url:He("/my-work-bug.json?tid=mrrferp8"),method:"get"});return null==e?void 0:e.bugs.map((({id:e,title:t})=>({name:t,value:{name:t,value:e}})))}();Ke("fix",await g({message:"请关联Bug(可多选):",validate:e=>0!==e.length||"请选择项",choices:e.map((e=>Object.assign(Object.assign({},e),{name:`[${e.value.value}]: ${e.name}`})))}))}if("chore"===e){Ke("chore",await m({message:"请输入commit msg:",validate:e=>0!==e.length||"请输入commit msg"}))}}async function Ye(){var e,a,i,r,c,l;try{const u={},d=async()=>{const e=J(),{projects:t}=await Te(),n=t.map((t=>{const n={name:`${t.name} [${t.desc}]`,value:t.name,disabled:!1};return e.includes(t.name)&&(n.disabled="目录下已存在同名文件夹"),n})),a=await p({message:"请选择一个项目模板",choices:n}),i=t.find((e=>e.name===a));u.tplName=i.name,u.tplUrl=i.url,u.tplLanguage=i.language},g=async()=>{var e;const{groups:n}=function(){const e=T("fe-groups.json");return t.existsSync(e)?h.readJSONSync(e):{groups:[]}}(),a=await p({message:"请选择一个分组",choices:n.map((e=>({name:`${e.name} [${e.description}]`,value:e.id,short:e.name})))});u.group={id:a,name:null===(e=n.find((e=>e.id===a)))||void 0===e?void 0:e.name}},w=async()=>{u.projectName=await m({message:"请输入项目名称",validate:Me}),u.projectDesc=await m({message:"请输入项目描述",validate:Ge})};await d(),await g(),await w();const v=`${Z}/${null===(e=u.group)||void 0===e?void 0:e.name}/${u.projectName}.git`,y=f("模版初始化中").start();await q(`git clone --depth=1 ${u.tplUrl}`),t.renameSync(u.tplName,u.projectName),o.cd(u.projectName),o.rm("-rf",".git"),await q("git init --initial-branch=master"),await q(`git remote add origin ${v}`),await q(`git config user.name ${ae("gitName")}`),await q(`git config user.email ${ae("gitEmail")}`);const b=ce();l=Object.assign(Object.assign({},b),{name:u.projectName,description:u.projectDesc,scripts:Object.assign(Object.assign({},b.scripts),{prepare:"[ -n '$z' ] && z init prepare || echo 'Warning: z not exist at global'"})}),re=re?Object.assign(Object.assign({},re),l):l,h.writeJSONSync(n.join("package.json"),re,{spaces:2}),await h.writeFile(n.resolve("README.md"),function(e="项目中文名",t="项目描述"){return`\n# ${e}\n${t}\n\n## 快速开始\n### 安装依赖\n\`\`\`bash\n npm i\n\`\`\`\n\n### 本地开发\n\`\`\`bash\n npm run dev:{environment}:{platform}\n\`\`\`\n\n### 打包代码\n\`\`\`bash\n npm run build:{environment}:{platform}\n\`\`\`\n\n\n### 查找、修复代码问题\n\`\`\`bash\n z lint eslint\n # 或者\n z l e\n\`\`\`\n\n### 格式化代码\n\`\`\`bash\n z lint prettier\n # 或者\n z l p\n\`\`\`\n\n## 常见问题\n列出开发中常见的问题和解决方案\n\n## 了解更多\n在此处放入飞书文档链接。请在[前端团队-项目手册](https://hxhtbr8t8uy.feishu.cn/drive/folder/QfQ7favVWljQk7d63Prc8mUGnJf)中按照分类新建文档。\n`}(u.projectName,u.projectDesc)),y.succeed("模版初始化完成"),await Fe(u.tplLanguage,null===(a=u.group)||void 0===a?void 0:a.name);const j=f("依赖安装中").start();await q("npm install --registry https://registry.npmmirror.com/"),j.succeed("依赖安装完成");const $=f("项目推送中").start();await q('git add . && git commit -m "chore: init"');const k=await(c={name:u.projectName,description:u.projectDesc,path:u.projectName,visibility:"private",namespace_id:null===(i=u.group)||void 0===i?void 0:i.id},$e({url:`${ze()}/projects`,method:"post",data:c}));await q("git push -u origin master"),$.succeed(`项目已推送到远程,地址: ${s.blue(v)}`);const x=f("初始化分支中").start(),S=`feat_init_${L()}`,O=G(null===(r=u.group)||void 0===r?void 0:r.name)?["dev","test","release",S]:[S],z=await Promise.allSettled(O.map((e=>function(e){return $e({url:`${ze()}/projects/${e.id}/repository/branches`,method:"post",data:e})}({id:k.id,branch:e,ref:"master"})))),N=[];z.forEach((({status:e},t)=>{"fulfilled"===e&&N.push(O[t])})),await q("git pull"),N.includes(S)?(await q(`git checkout -b ${S} origin/${S}`),x.succeed(`项目已切换到初始分支: ${s.blue(S)}`)):x.warn("开发分支检出失败!项目当前在主分支,请自行检出开发分支。");const E=`cd ${u.projectName} && z start`;console.log(`输入 ${s.green(E)} 开始开发吧~`),process.exit(0)}catch(e){C(e)}}function Xe(e){return Oe({url:`${se("K8sDomain")}/kapis/clusters/youshou-local/devops.kubesphere.io/v1alpha3/namespaces/${ee}/pipelines/${e.projectName}/pipelineruns?branch=${e.branchName}`,method:"post",data:e.params})}async function et(e){const{targetBranch:t,platform:n,notify:a,processId:i,projectName:s}=e,r=a?[...new Set([ae("weWorkUserId"),...a])]:[ae("weWorkUserId")];await Xe({projectName:s,branchName:t,params:{parameters:[{name:"platform",value:n},{name:"notify",value:r.join(",")},{name:"processId",value:i||""},{name:"promoterName",value:ae("weWorkName")},{name:"promoterId",value:ae("weWorkUserId")},{name:"rollback",value:String(e.rollback)}]}})}async function tt(e,t){R(),F(),await M();let n=e;if(n&&!H.includes(n)&&C("仅支持发布指定环境分支"),!n){const e=await p({message:"请选择部署环境",choices:Q});n="production"===(a=e)?"master":a}var a;const i=await p({message:"请选择平台",choices:V}),s=await Ue("请选择部署成功要通知的人员:",(e=>!(e.length<1)||"至少选一个"),!0);var r;!function(e,t){var n,a,i,s,r;const o=ce(),c=le();(null===(n=null==c?void 0:c.repository)||void 0===n?void 0:n.url)||C(".z/project.json中缺少repository.url");const l=`build:${e}`;(null==o?void 0:o.scripts[l])||C(`项目package.json文件scripts不存在命令${l}。`),["alipay","weapp"].includes(t)&&(o.mini||C("请在package.json配置mini字段"),"alipay"===t&&((null===(i=null===(a=o.mini)||void 0===a?void 0:a.alipay)||void 0===i?void 0:i.appid)||C("请在package.json配置mini.alipay.appid字段")),"weapp"===t&&((null===(r=null===(s=o.mini)||void 0===s?void 0:s.weapp)||void 0===r?void 0:r.appid)||C("请在package.json配置mini.weapp.appid字段")))}("master"===(r=n)?"production":r,i);const o=ce();let c=t||"";"master"===n&&(c||(c=await Ce(),await We(c,{name:o.name}),await Ae({id:c,state:2}))),await et({projectName:o.name,targetBranch:n,platform:i,notify:s,processId:c,rollback:!1})}function nt(){const e=O();if(t.existsSync(e)){const t=h.readJsonSync(e);if(t["lint-staged"])return t["lint-staged"];throw new Error("未找到lint配置")}throw new Error("请先初始化项目(z i .)。")}async function at(e="latest"){await q(`npm i -g ${te.name}@${e} --registry https://registry.npmmirror.com/`),ye("latestCheckVersionTimestamp",Date.now()),ye("version",e),de(),process.exit(0)}async function it(){const e=y(process.argv.slice(2));if(!(e.h||e.help||e.v||e.version)&&(!["init","i"].includes(e._[0])||[".","prepare"].includes(e._[1])))if(_()){const e=h.readJSONSync(E());we(e.profile),ye(e.main),ve(e.constants),await async function(){const e=ie("latestCheckVersionTimestamp"),t=ie("versionCheckDuring");if(Date.now()-Number(e)>24*Number(t)*3600*1e3){let e;ve(await Ee());try{const t="TIME_OUT",n=new Promise((e=>{setTimeout((()=>e(t)),3e3)}));e=await Promise.race([n,q(`npm view ${te.name} version --registry https://registry.npmmirror.com/`)]),e!==t&&e!==te.version&&(console.log(`${s.blue(te.name)}本地版本为${te.version},低于线上版本${e},开始升级`),await at(e),console.log(s.green(`升级完成,当前版本${e}`)),process.exit(0)),ye("latestCheckVersionTimestamp",Date.now()),de()}catch(e){console.log("升级出错!请重试,或者手动升级"),C(e)}}}()}else C("请先初始化z-develop(执行 z i)。更多见https://hxhtbr8t8uy.feishu.cn/docx/RnOhdtqJWo5UooxHTaUcQFwtnKg")}e.command("create").alias("c").argument("[type]","可选值为project|pr, branch|b, commit-msg|cm, page|p, component|co").description("创建项目/分支/提交信息/页面/组件").action((async function(e){let a;e?["project","pr"].includes(e)?a="project":["branch","b"].includes(e)?a="branch":["page","pa"].includes(e)?a="page":["component","co"].includes(e)?a="component":["commit-msg","cm"].includes(e)?a="commit-msg":C("参数输入错误"):a=await p({message:"请选择你要创建的类型",choices:[{name:"提交信息(commit message)",value:"commit-msg"},{name:"分支(branch)",value:"branch"},{name:"页面(page)",value:"page"},{name:"组件(component)",value:"component"},{name:"项目(project)",value:"project"}]}),"project"===a?await Ye():"branch"===a?(R(),F(),await M(),await async function(){R(),F(),await M();const e=`${await p({message:"请选择创建分支的类型",choices:[{name:"开发新功能(feat)",value:"feat"},{name:"修复BUG(fix)",value:"fix"},{name:"重构/优化代码(refactor)",value:"refactor"}]})}_${await m({message:"请输入创建分支的目的(大小写字母、数字,小驼峰式命名。如userInfo)",validate:Qe})}_${L()}`;await q("git fetch origin master"),await q(`git checkout -b ${e} origin/master`),await q(`git push -u origin ${e}`)}()):"commit-msg"===a?(R(),F(),await Ze()):"page"===a?(R(),F(),await async function(){const e="pages",a=await ge(e);t.mkdirSync(n.resolve("src",e),{recursive:!0});const i=await m({message:"请输入目录及页面名称(如user/list,不带文件后缀)",validate:Ve});let s;if(i.includes("/")){const a=i.split("/");a.pop(),s=n.resolve("src",e,...i.split("/")),t.mkdirSync(n.resolve("src",e,...a),{recursive:!0})}else s=n.resolve("src",e,i);const{pages:r}=await Te(),c=t.readdirSync(a),l=await p({message:"请选择模版",choices:r.filter((e=>c.includes(e.name))).map((e=>({name:`${e.name}: ${e.desc}(${e.author})`,value:e.name})))}),u=f("页面生成中...").start(),d=s+n.extname(n.join(a,l));o.cp("-r",n.join(a,l),d),W(d)?P(d).forEach((e=>{he(n.join(d,e),i)})):he(d,i),u.succeed("页面生成成功")}()):"component"===a&&(R(),F(),await async function(){const e="components",a=await ge("components");t.mkdirSync(n.resolve("src",e),{recursive:!0});const i=await m({message:"请输入目录及组件名称(如user/list,不带文件后缀)",validate:Ve});let s;if(i.includes("/")){const a=i.split("/"),r=a.pop();s=n.resolve("src",e,...a,fe(r)),t.mkdirSync(n.resolve("src",e,...a),{recursive:!0})}else s=n.resolve("src",e,fe(i));const{components:r}=await Te(),c=t.readdirSync(a),l=await p({message:"请选择模版",choices:r.filter((e=>c.includes(e.name))).map((e=>({name:`${e.name}: ${e.desc}(${e.author})`,value:e.name})))}),u=f("组件生成中...").start(),d=s+n.extname(n.join(a,l));o.cp("-r",n.join(a,l),d),W(d)?P(d).forEach((e=>{he(n.join(d,e),i)})):he(d,i),u.succeed("组件生成成功")}()),process.exit(0)})),e.command("merge").alias("m").description("合并当前分支到指定远程分支,并部署。").action((async function(){var e;R(),F(),await M();const t=await pe();H.includes(t.sourceBranch)&&C(`当前分支${t.sourceBranch}不可作为源分支合并到目标分支。`);const n=await(i=t.id,$e({url:`${ze()}/projects/${i}/repository/branches`}));var i;const r=G(t.group),o=n.filter((e=>![t.sourceBranch].includes(e.name))).map((e=>{let t=e.name;return r&&H.includes(t)&&(t=s.bold.blue(t)),{name:t,value:e.name}})),c=await p({message:"请选择要合并到的目标分支:",choices:o});let l;"master"===c&&(l=await Ce(),await We(l,{name:t.name,branch:t.sourceBranch}),await Ae({id:l,state:1}));const m=f(`${t.sourceBranch}分支推送中`).start();await q(`git ls-remote --heads origin ${t.sourceBranch}`)&&await q("git pull"),await q(`git push -u origin ${t.sourceBranch}`),m.succeed(`${t.sourceBranch}已推送到远程`),await async function(e,t,n){const a=`origin/${e}`,i=await q("git log -b origin/master -1 --format=%H"),s=await q(`git log -b origin/${e} -1 --format=%H`),r=await q(`git log ${i}...${s} -b ${a}`);if(r||C(`分支${e}上不存在新的commit,合并终止。`),"master"===t){if(n){const e=r.split("\n").filter((e=>e)),t=await q("git log -b origin/test -1 --format=%H");(function(e,t){for(const n of t)if(!e.includes(n))return!1;return!0})((await q(`git log ${i}...${t} -b origin/test`)).split("\n").filter((e=>e)),e)||C("请先在测试环境发布要部署的代码")}}else await q(`git branch --contains ${i} -r ${a}`)||C(`请先将最新的主分支(master)代码合并到${a},再进行部署。推荐命令:git merge origin/master`)}(t.sourceBranch,c,r);const u=f("分支合并中").start();let d=0;try{const{iid:e}=await(g={title:"MR by z-develop",id:t.id,source_branch:t.sourceBranch,target_branch:c,remove_source_branch:"master"===c},$e({url:`${ze()}/projects/${g.id}/merge_requests`,method:"post",data:g}));d=e}catch(n){if(a.isAxiosError(n)){409===(null===(e=null==n?void 0:n.response)||void 0===e?void 0:e.status)&&C(`存在重复的合并请求,前往查看${t.mergeRequestUrl}`)}C(n)}var g;const h=async()=>{var e;try{await new Promise((e=>{setTimeout(e,5e3)})),await function(e){return $e({url:`${ze()}/projects/${e.id}/merge_requests/${e.iid}/merge`,method:"put"})}({id:t.id,iid:d})}catch(n){if(a.isAxiosError(n)){406===(null===(e=null==n?void 0:n.response)||void 0===e?void 0:e.status)?await h():(await function(e){return $e({url:`${ze()}/projects/${e.id}/merge_requests/${e.iid}`,method:"put",data:e})}({id:t.id,iid:d,state_event:"close"}),u.fail("合并过程中出现冲突,MR已关闭。建议在本地完成合并。"),process.exit(1))}else C(n)}};await h(),u.succeed(`当前分支${t.sourceBranch}已合并到分支${c}`),"master"===c&&(await q("git checkout master"),await q("git pull"),await q(`git branch -d ${t.sourceBranch}`),U(`\n${t.sourceBranch}分支已移除,当前已切换到最新的master。\n如需继续开发,请检出新分支(z c b)。\n`)),r&&H.includes(c)&&await tt(c,l)})),e.command("deploy").alias("d").description("发布指定远程分支").argument("[branchName]","可选值为dev/test/release/master").argument("[processId]","审批编号,branchName为master时需要").action(tt),e.command("start").alias("s").description("启动本地开发环境").argument("[env]","可选值 dev|d, test|t, release|r, prod|p").action((async function(e){let t=e;t?["dev","test","release","production","d","t","r","p"].includes(e)?"d"===t?t="dev":"t"===t?t="test":"r"===t?t="release":"p"===t&&(t="production"):C("参数输入有误"):t=await p({message:"请选择环境",choices:Q});const n=`dev:${t}`;if(ce().scripts[n])try{o.exec(`npm run ${n}`,{silent:!1})}catch(e){C(e)}else C(`项目中(package.josn > scripts)不存在命令${n},请先添加!`)})),e.command("lint").alias("l").argument("[type]","非必填。可选值为commit-msg|cm, commit-files|cf, prettier|p, eslint|e。").option("--quiet","eslint只打印error").option("--max-warnings [VALUE]","eslint结果warn超过多少个会异常退出,默认为100",parseInt).description("执行 lint 脚本。包含 lint staged 、 lint commit message、 prettier、 eslint。").action((async function(e,a){let i;if(R(),F(),e?["commit-msg","cm"].includes(e)?i="commit-msg":["commit-files","cf"].includes(e)?i="commit-files":["prettier","p"].includes(e)?i="prettier":["eslint","e"].includes(e)?i="eslint":C("参数输入错误"):i=await p({message:"请选择你要执行的操作",choices:[{name:"全局执行 prettier",value:"prettier"},{name:"全局执行 eslint",value:"eslint"}]}),"commit-msg"===i)await function(){const e=function(e){if(e.includes("Merge")&&e.includes("# Conflicts:"))return!0;const t=e.split(": ");if(1===t.length)return"body前缺少「: 」";{const e=t[0];return/^(feat|fix|refactor)/.test(e)?!!/\(\d+(?:,\d+)*\)/.test(e)||"ID缺少或者格式不正确。":/^(chore|ci)/.test(e)?!!["chore","ci"].includes(e)||"chore/ci类型无需填写ID。":"不存在的提交类型。"}}(t.readFileSync(n.resolve(".git","COMMIT_EDITMSG"),"utf-8"));!0===e?(U("commit msg validate success."),process.exit(0)):C(`commit msg格式错误。${e}`)}();else if("commit-files"===i)await async function(){await v({concurrent:!1,debug:!1,config:nt()})?(U("代码风格检测通过!"),process.exit(0)):C("代码风格检测未通过!")}();else if("prettier"===i){t.existsSync(n.resolve("node_modules",".bin","prettier"))||C("该项目未安装prettier,请安装后重试");try{await q("npx prettier --write . --config .z/.prettierrc.json --ignore-path .z/.prettierignore --ignore-unknown --no-error-on-unmatched-pattern",{silent:!1}),U("prettier执行成功"),process.exit(0)}catch(e){C("prettier校验出错")}}else if("eslint"===i){t.existsSync(n.resolve("node_modules",".bin","eslint"))||C("该项目未安装eslint,请安装后重试");try{let e="npx eslint '**/*.{vue,js,jsx,cjs,mjs,ts,tsx,cts,mts}' --fix";"win32"===process.platform&&(e='npx eslint "**/*.{vue,js,jsx,cjs,mjs,ts,tsx,cts,mts}" --fix'),a.quiet&&(e+=" --quiet"),a.maxWarnings&&!isNaN(Number(a.maxWarnings))?e+=` --max-warnings ${Number(a.maxWarnings)}`:e+=" --max-warnings 100",await q(e,{silent:!1}),U("eslint执行成功"),process.exit(0)}catch(e){C("eslint校验出错")}}})),e.command("rollback").alias("r").description("正式环境回滚").argument("[processId]","非必填。审批单ID。").action((async function(e){R(),F();let t=e||"";const n=ce();t||(t=await Ce(),await We(t,{name:n.name,rollback:!0}),await Ae({id:t,state:2}));const a=await p({message:"请选择平台",choices:V}),i=await Ue("请选择回滚成功要通知的人员:",(e=>!(e.length<1)||"至少选一个"),!0);await et({projectName:n.name,targetBranch:"master",platform:a,notify:i,processId:t,rollback:!0})})),e.command("update").alias("u").argument("[version]","版本号","latest").description("升级").action(at),e.name("z-develop").alias("z").description(`z-develop, 开发流程管理工具。\n了解更多: ${s.blue("https://hxhtbr8t8uy.feishu.cn/docx/RnOhdtqJWo5UooxHTaUcQFwtnKg")}`).usage("<command> [options]").hook("preAction",(async function(){process.on("unhandledRejection",(e=>{})),process.on("uncaughtException",(e=>{})),o.config.fatal=!0,o.config.silent=!0,o.config.verbose=Boolean(void 0)||!1,await it()})).version(te.version,"-v, --version","当前版本号").helpOption("-h, --help","帮助").showHelpAfterError("可以使用z -h查看帮助。"),e.parse();
1
+ import{program as e}from"commander";import t from"node:fs";import n from"node:path";import a,{AxiosError as i}from"axios";import s from"chalk";import r from"dayjs";import o from"shelljs";import c from"node:os";import l from"child_process";import{confirm as u,select as m,input as p,password as d,checkbox as g}from"@inquirer/prompts";import f from"ora";import h from"fs-extra";import{select as w}from"inquirer-select-pro";import v from"lint-staged";import y from"semver";import $ from"minimist";const j=".z",b="develop-config.json",x=".z",S=".commit-msg-tpl",k="project.json";function z(e=""){return n.join(x,e)}function O(){return z(k)}function N(){return z(S)}function T(){l.spawnSync("git",["config","core.hooksPath",z()]),l.spawnSync("git",["config","commit.template",N()])}function E(e=""){return n.resolve(c.homedir(),j,e)}function _(){return E(b)}function A(){return t.existsSync(E())&&t.existsSync(_())}const{red:D,green:B,blue:C,magenta:U}=s;function J(...e){console.log(B(...e))}function W(e,t=!1){let n=e;e instanceof Error?(n=e.message,a.isAxiosError(e)&&(n=`请求失败:${e.message}`),console.log(D(n)),console.log(U(e.stack))):console.log(D(e)),t||process.exit(1)}function P(e){return!!t.existsSync(e)&&t.lstatSync(e).isDirectory()}function q(e=process.cwd()){if(P(e)){return t.readdirSync(e).filter((t=>P(n.resolve(e,t))))}return[]}function I(e=process.cwd()){if(P(e)){return t.readdirSync(e).filter((t=>!P(n.resolve(e,t))))}return[]}async function L(e,t={removeTailLinkBreak:!0,silent:!0}){let n=await new Promise(((n,a)=>{try{n(o.exec(e,{silent:t.silent}))}catch(e){a(e)}}));return n=n.toString(),t.removeTailLinkBreak&&(n=n.replace(/\n$/,"")),n}function R(){return r(Date.now()).format("YYMMDD")}function F(){t.existsSync(z())||W("当前不在项目根目录。请切换到项目根目录")}function M(){t.existsSync(n.resolve(".git"))||W("当前不是git项目根目录,请先执行git init,或切换到根目录")}async function G(){""!==await L("git status -s")&&W("请先提交代码变动,再进行操作")}function H(e){return["fe-biz"].includes(e)}const Q=[{name:"开发环境 - dev",value:"dev"},{name:"测试环境 - test",value:"test"},{name:"预发环境 - release",value:"release"},{name:"正式环境 - production",value:"production"}],V=[{name:"网页 - h5",value:"h5"},{name:"依赖包 - npm",value:"npm"}],K=["master","dev","test","release"],Z={1:"研发",2:"测试",3:"产品",4:"设计",5:"运营",6:"销售",7:"行政",8:"财务",9:"其他"},Y="http://git.cxlqd.com",X={PROJECT_NAME:"fe-pages-tpl",GIT_URL:`${Y}/fe-component/fe-pages-tpl.git`},ee=["fe-biz","fe-base","fe-tpl","fe-component","fe-demo"];var te={name:"cli-z-develop",version:"0.0.58",description:"前端本地开发命令行工具",main:"dist/index.js",bin:{z:"bin/z.js","z-develop":"bin/z.js"},scripts:{prepare:"[ -n '$z' ] && z init prepare || echo 'Warning: z not exist at global'",dev:"rollup -c -w --environment DEBUG:true",build:"rollup -c",eslint:"eslint '**/*.{ts,js}' --fix",prettier:"prettier -wu .",upload:"npm run build && npm publish --access public --registry https://registry.npmjs.org/","upload:patch":"npm version patch && npm run upload","upload:minor":"npm version minor && npm run upload","upload:major":"npm version major && npm run upload"},type:"module",repository:{type:"git",url:"http://git.cxlqd.com/fe-base/cli-z-develop.git"},author:"z",devDependencies:{"@lonely9/eslint-config-team":"^1.2.1","@rollup/plugin-json":"^6.1.0","@rollup/plugin-node-resolve":"^16.0.1","@rollup/plugin-replace":"v6.0.2","@rollup/plugin-terser":"^0.4.4","@types/fs-extra":"^11.0.4","@types/inquirer":"^9.0.7","@types/minimist":"^1.2.5","@types/node":" ^22.14.1","@types/semver":"^7.7.0","@types/shelljs":"^0.8.15","@typescript-eslint/eslint-plugin":"^8.31.0","@typescript-eslint/parser":"^8.31.0",eslint:"^9.25.1",jiti:"^2.4.2",prettier:"^3.5.3",rollup:" 4.40.0","rollup-plugin-typescript2":"^0.36.0",typescript:" ^5.8.3"},dependencies:{"@inquirer/prompts":"^7.4.1",axios:"^1.8.4",chalk:"^5.4.1",commander:"^13.1.0",dayjs:"^1.11.13",figures:"^6.1.0","fs-extra":"^11.3.0","inquirer-select-pro":"^1.0.0-alpha.9","lint-staged":"^15.5.1",minimist:"^1.2.8",ora:"^8.2.0",rxjs:"^7.8.2",semver:"^7.7.1",shelljs:"^0.9.2"}};const ne={profile:{ldapAccount:"",ldapPassword:"",gitToken:"",gitUserId:0,gitName:"",gitEnglishName:"",gitEmail:"",weWorkName:"",weWorkUserId:"",jobType:"fe",zenTaoToken:"",k8sToken:""},main:{version:te.version,latestCheckVersionTimestamp:0,versionCheckDuring:3,weWorkListCache:[]},constants:{ZenTaoDomain:"",K8sDomain:"",FEServerDomain:""}};function ae(e){return e?ne.profile[e]:ne.profile}function ie(e){return e?ne.main[e]:ne.main}function se(e){return e?ne.constants[e]:ne.constants}let re=null,oe=null;function ce(){if(re)return re;const e=n.join("package.json");return t.existsSync(e)||W(`当前目录(${o.pwd()})不存在${e}文件,请在项目根目录执行该命令。`),re=h.readJsonSync(e),re}function le(){if(oe)return oe;const e=O();return t.existsSync(e)||W(`当前目录(${o.pwd()})不存在${e}文件,请在项目根目录执行该命令,或者初始化项目(z init .)。`),oe=h.readJsonSync(e),oe}function ue(e){oe=oe?Object.assign(Object.assign({},oe),e):e,h.writeJSONSync(O(),oe,{spaces:2})}const me={id:0,name:"",group:"",sourceBranch:"",mergeRequestUrl:""};async function pe(){if(!me.name){const e=ce();me.name=e.name}if(!me.group){const e=le();e.repository.group&&(me.group=e.repository.group)}if(!me.id)try{const e=me.group?`${me.group}/${me.name}`:me.name,[t]=await Te(e);me.mergeRequestUrl=`${t.web_url}/merge_requests`,me.id=t.id,me.group=t.namespace.path;const n=le();n.repository.group||ue({repository:Object.assign(Object.assign({},n.repository),{group:t.namespace.path})})}catch(e){W(e)}return me.sourceBranch||(me.sourceBranch=await L("git branch --show-current")),me}function de(){h.writeJSONSync(_(),{main:ie(),profile:ae(),constants:se()},{spaces:2})}async function ge(e){const a=E(),i=process.cwd(),s=n.join(a,X.PROJECT_NAME);t.existsSync(s)?(o.cd(s),await L("git pull"),o.cd(i)):(o.cd(a),await L(`git clone ${X.GIT_URL} --depth=1`),o.cd(i));const r=n.join(s,"src",e);return t.existsSync(r)||W("模板库错误!"),r}function fe(e){return e.split("-").map((e=>e.charAt(0).toUpperCase()+e.slice(1))).join("")}function he(e,n){const a=t.readFileSync(e,"utf-8"),i=n.replace(/\//g,"-"),s=a.replace(/name: "-_-NAME-_-"/g,`name: "${fe(i)}"`).replace(/class="-_-NAME-_-"/g,`class="${i}"`).replace(/.-_-NAME-_-/g,`.${i}`);t.writeFileSync(e,s,"utf-8")}function we(e,t){void 0!==t?ne.profile[e]=t:ne.profile=Object.assign(Object.assign({},ne.profile),e)}function ve(e,t){ne.constants=Object.assign(Object.assign({},ne.constants),e)}function ye(e,t){void 0!==t?ne.main[e]=t:ne.main=Object.assign(Object.assign({},ne.main),e)}async function $e(e){var t;const n=e.method||"get",r=e.headers||{},o=e.data||{},c=e.dataKey||["GET","get"].includes(n)?"params":"data";try{const t={url:e.url,method:n,[c]:o,headers:r};s.magenta(t.method.toUpperCase()),s.yellow(t.url),JSON.stringify(t.headers),s.gray(JSON.stringify(t[c],null,2));const i=await a(t);return s.green("Response"),s.grey(JSON.stringify(i.data,null,2)),Promise.resolve(i.data)}catch(e){return e instanceof i&&(s.red("Error"),s.grey(JSON.stringify(null===(t=null==e?void 0:e.response)||void 0===t?void 0:t.data))),Promise.reject(e)}}async function je(){const e=ae("gitToken");if(e)return e;{const{access_token:e}=await $e({url:`${Y}/oauth/token`,method:"post",data:{grant_type:"password",username:ae("ldapAccount"),password:ae("ldapPassword")}}),t=`Bearer ${e}`;return we("gitToken",t),de(),t}}async function be(e){return $e(Object.assign(Object.assign({},e),{headers:{Authorization:await je()}}))}async function xe(){const e=ae("zenTaoToken");if(e)return e;{const{token:e}=await $e({url:`${se("ZenTaoDomain")}/api.php/v1/tokens`,method:"post",data:{account:ae("ldapAccount"),password:ae("ldapPassword")}});return we("zenTaoToken",e),de(),e}}async function Se(e){const t=await $e(Object.assign(Object.assign({},e),{headers:{Token:await xe()}}));if("string"==typeof t){const e=JSON.parse(t.slice(0,t.indexOf("<script>")));return JSON.parse(e.data)}if("object"==typeof t)return"Unauthorized"===t.error||t.data.indexOf("<script>")<0?(we("zenTaoToken",""),Se(Object.assign({},e))):t}const ke=()=>`${Y}/api/v4`,ze=e=>`${ke()}/projects/100/repository/files/${encodeURIComponent(e)}/raw?ref=master`;function Oe(){return be({url:ze("src/data/template-projects.json")})}function Ne(){return be({url:ze("src/data/z-develop-config.json")})}function Te(e){return be({url:`${ke()}/projects`,data:{search:e,search_namespaces:!0}})}async function Ee(){const e=await be({url:`${ke()}/groups`}),t=ee.map((t=>{const n=e.find((e=>e.name===t));return!!n&&{name:n.name,id:n.id,description:n.description}})).filter((e=>Boolean(e)));h.writeJSONSync(E("fe-groups.json"),{groups:t},{spaces:2})}async function _e(e){const t=`${se("FEServerDomain")}/api`,{data:n}=await $e({url:`${t}/auth/z-develop/login`,method:"post"}),a=await $e({url:t+e.url,headers:{Authorization:`Bearer ${n}`},data:e.data,method:e.method||"post"});return 0!==a.code&&W(`${a.code}: ${a.message}`),a.data}async function Ae(){const e=(await _e({url:"/user/list2"})).filter((e=>[1,2,3].includes(e.title))).map((e=>({name:`${e.nick} - ${Z[e.title]}`,value:e.weWorkUserId}))),t=ie("weWorkListCache"),n=t.map((e=>e.value)),a=[];return e.forEach((e=>{const i=n.indexOf(e.value);a.push(Object.assign(Object.assign({},e),{usageCount:i>-1?t[i].usageCount:0}))})),a.sort(((e,t)=>t.usageCount-e.usageCount))}async function De(e,t,n){const a=await Ae();let i=await w({message:e,loop:!0,clearInputWhenSelected:!0,multiple:n,options:e=>e?a.filter((t=>t.name.includes(e))):a,validate:t});return i=Array.isArray(i)?i:[i],function(e,t){const n=[];t.forEach((t=>{e.includes(t.value)&&n.push(t)}));const a=ie("weWorkListCache"),i=a.map((e=>e.value));n.forEach((e=>{const t=i.indexOf(e.value);t>-1?a[t].usageCount+=1:a.push(Object.assign(Object.assign({},e),{usageCount:1}))})),ye({weWorkListCache:a.sort(((e,t)=>t.usageCount-e.usageCount))}),de()}(i,a),i}async function Be(){try{if(A()){await u({message:"系统中已存在z的配置文件,确认重新配置?"})?o.rm("-rf",_()):process.exit(0)}t.mkdirSync(E(),{recursive:!0});const e=await m({message:"请选择岗位类型",choices:[{name:"前端",value:"fe"},{name:"后端",value:"be"}]}),n=await p({message:"请输入LDAP账号:"}),a=await d({message:"请输入LDAP密码:",mask:!0});we("jobType",e),we("ldapAccount",n),we("ldapPassword",a);ve(await Ne());const i=await Ae(),s=await De("请选择你自己(用于企微通知):",(e=>e.length>1?"只能选一个":!(e.length<1)||"请选一个"),!1),{name:r,value:c}=i.find((e=>e.value===s[0]));we({weWorkName:r,weWorkUserId:c})}catch(e){W(e)}const e=f("配置信息初始化中").start();try{const t=await be({url:`${ke()}/user`});we({gitUserId:t.id,gitName:t.name,gitEnglishName:t.username,gitEmail:t.email}),ye("latestCheckVersionTimestamp",Date.now()),de(),await Ee(),e.succeed("配置信息初始化完成"),process.exit(0)}catch(t){e.fail("配置信息初始化失败"),a.isAxiosError(t)&&W("请检查你的域名及令牌配置"),W(t)}}const Ce='{\n "printWidth": 120,\n "tabWidth": 2,\n "useTabs": false,\n "semi": true,\n "singleQuote": false,\n "quoteProps": "as-needed",\n "jsxSingleQuote": false,\n "trailingComma": "all",\n "bracketSpacing": true,\n "bracketSameLine": false,\n "arrowParens": "always",\n "requirePragma": false,\n "insertPragma": false,\n "proseWrap": "preserve",\n "htmlWhitespaceSensitivity": "css",\n "vueIndentScriptAndStyle": false,\n "endOfLine": "lf",\n "embeddedLanguageFormatting": "auto",\n "singleAttributePerLine": false\n}\n',Ue="# 系统 信息文件\n.DS_Store/\nThumbs.db\n\n# 编辑器配置文件\n.idea/\n.vscode/\n\n# 输出目录\ndist/\n\n# Log files\n*.log\n\n# 本地配置文件\n*.local\n\n# 缓存文件\n*.cache\n\n# 垃圾文件\n.Trashes\n",Je={Java:{"**/*.{java}":"./run.sh pmd -d ../../../src/main/java/ -f text -R rulesets/java/quickstart.xml"},JavaScript:{"**/*.{vue,js,jsx,cjs,mjs,ts,tsx,cts,mts}":"eslint --fix","*":"prettier -wu"}};let We;async function Pe(e,n){if(e)o.rm("-rf",x);else if(t.existsSync(z())){await u({message:`当前项目中已存在配置文件夹${x},确认重新配置?`})?o.rm("-rf",x):process.exit(0)}t.mkdirSync(z(),{recursive:!0}),await async function(e,t){const n=await L("git remote get-url --push origin");n||W("获取项目远程地址失败,请配置后重试");let a=e;a||(a=await m({message:"请选择项目语言",choices:[{name:"JavaScript",value:"JavaScript"},{name:"Java",value:"Java"}]}));let i=t;const s=ce();if(!i){const e=await Te(s.name);1===e.length?i=e[0].namespace.path:e.length>1&&(i=await m({message:"请选择当前项目所属分组",choices:e.map((e=>({name:e.namespace.path,value:e.namespace.path})))}))}We=f("项目初始化中").start(),ue({language:a,"lint-staged":Je[a],repository:{url:n,group:i}})}(e,n),await(t.writeFileSync(N(),"",{mode:493}),t.writeFileSync(z("commit-msg"),"#!/usr/bin/env sh\nz lint commit-msg",{mode:493}),t.writeFileSync(z("pre-commit"),"#!/usr/bin/env sh\nz lint commit-files",{mode:493}),t.writeFileSync(z(".prettierrc.json"),Ce,{mode:493}),t.writeFileSync(z(".prettierignore"),Ue,{mode:493}),t.writeFileSync(z(".gitignore"),".commit-msg-tpl\n",{mode:493}),void T()),null==We||We.succeed("初始化完成。")}async function qe(e){if(!e)return"不能为空";if(!/^[a-z0-9-]+$/.test(e))return"格式为小写字母、中横线(可选)、数字(不推荐)。如apple, apple-tree";if(q().includes(e))return"当前目录下已存在同名文件夹,请先处理。";return!(await Te(e)).find((t=>t.name===e))||"远程仓库中已存在同名项目"}function Ie(e){return!!e||"不能为空"}function Le(e){return e?!!/^[a-zA-Z0-9.]+$/.test(e)||"格式为大小写字母、数字、小数点,小驼峰命名。如userInfo、systemRouter3":"不能为空"}function Re(e){if(!e)return"目录名称不能为空";if(!/^[a-z][a-z0-9\-/]*$/.test(e))return"支持小写字母、数字,连字符(-)命名,小写字母开头,多层目录使用/隔开。如user、modules-3、setting/profile";if(e.includes("/")){const t=e.split("/");for(const e of t){if(!e)return"子级目录/页面名称不能为空";if(!/^[a-z][a-z0-9-]*$/.test(e))return"支持小写字母、数字,连字符(-)命名,小写字母开头。如user、modules-3"}}let a,i;a=e.includes("/")?n.resolve("src","pages",...e.split("/")):n.resolve("src","pages",e);const s=n.resolve(a,"..");return i=!!t.existsSync(s)&&t.readdirSync(s).map((e=>{const t=e.lastIndexOf(".");return t>-1?e.slice(0,t):e})).includes(e),!i||"本地目录已存在该文件"}e.command("init").alias("i").argument("[type]","非必填。不传为初始化工具配置;传 . 为初始化当前项目。").description("初始化工具配置、项目配置").action((async function(e){e?"."===e?(M(),await Pe(void 0,void 0),process.exit(0)):"prepare"===e?(M(),await async function(){t.existsSync(z())&&t.existsSync(N())?T():await Pe(),process.exit(0)}()):W("参数错误。执行 z init -h 查看帮助。"):await Be()}));const Fe=e=>`${se("ZenTaoDomain")}${e}`;function Me(e,n){let a="";if(["feat","fix","refactor"].includes(e)){a=`${e}(${n.map((e=>e.value)).join(",")}): ${n.map((e=>e.name)).join(";")}`}"chore"===e&&(a=`${e}: ${n}`),t.writeFileSync(N(),a,"utf-8"),console.log(s.yellow(a)),J("commit msg模板写入成功,可以进行提交了")}async function Ge(){const e=await m({message:"请选择你要创建的模板类型",choices:[{name:"feat - 业务需求开发",value:"feat"},{name:"fix - bug修复",value:"fix"},{name:"refactor - 技术内部需求",value:"refactor"},{name:"chore - 其他",value:"chore"}]});if(["feat","refactor"].includes(e)){const e=await async function(){const e=await Se({url:Fe("/my-work-task.json?tid=mrrferp8"),method:"get"});return null==e?void 0:e.tasks.filter((e=>"done"!==e.status)).map((({id:e,name:t})=>({value:{name:t,value:e},name:t})))}();Me("feat",await g({message:"请关联开发任务(可多选):",validate:e=>0!==e.length||"请选择项",choices:e.map((e=>Object.assign(Object.assign({},e),{name:`[${e.value.value}]: ${e.name}`})))}))}if("fix"===e){const e=await async function(){const e=await Se({url:Fe("/my-work-bug.json?tid=mrrferp8"),method:"get"});return null==e?void 0:e.bugs.map((({id:e,title:t})=>({name:t,value:{name:t,value:e}})))}();Me("fix",await g({message:"请关联Bug(可多选):",validate:e=>0!==e.length||"请选择项",choices:e.map((e=>Object.assign(Object.assign({},e),{name:`[${e.value.value}]: ${e.name}`})))}))}if("chore"===e){Me("chore",await p({message:"请输入commit msg:",validate:e=>0!==e.length||"请输入commit msg"}))}}async function He(){var e,a,i,r,c,l;try{const u={},d=async()=>{const e=q(),{projects:t}=await Oe(),n=t.map((t=>{const n={name:`${t.name} [${t.desc}]`,value:t.name,disabled:!1};return e.includes(t.name)&&(n.disabled="目录下已存在同名文件夹"),n})),a=await m({message:"请选择一个项目模板",choices:n}),i=t.find((e=>e.name===a));u.tplName=i.name,u.tplUrl=i.url,u.tplLanguage=i.language},g=async()=>{var e;const{groups:n}=function(){const e=E("fe-groups.json");return t.existsSync(e)?h.readJSONSync(e):{groups:[]}}(),a=await m({message:"请选择一个分组",choices:n.map((e=>({name:`${e.name} [${e.description}]`,value:e.id,short:e.name})))});u.group={id:a,name:null===(e=n.find((e=>e.id===a)))||void 0===e?void 0:e.name}},w=async()=>{u.projectName=await p({message:"请输入项目名称",validate:qe}),u.projectDesc=await p({message:"请输入项目描述",validate:Ie})};await d(),await g(),await w();const v=`${Y}/${null===(e=u.group)||void 0===e?void 0:e.name}/${u.projectName}.git`,y=f("模版初始化中").start();await L(`git clone --depth=1 ${u.tplUrl}`),t.renameSync(u.tplName,u.projectName),o.cd(u.projectName),o.rm("-rf",".git"),await L("git init --initial-branch=master"),await L(`git remote add origin ${v}`),await L(`git config user.name ${ae("gitName")}`),await L(`git config user.email ${ae("gitEmail")}`);const $=ce();l=Object.assign(Object.assign({},$),{name:u.projectName,description:u.projectDesc,scripts:Object.assign(Object.assign({},$.scripts),{prepare:"[ -n '$z' ] && z init prepare || echo 'Warning: z not exist at global'"})}),re=re?Object.assign(Object.assign({},re),l):l,h.writeJSONSync(n.join("package.json"),re,{spaces:2}),await h.writeFile(n.resolve("README.md"),function(e="项目中文名",t="项目描述"){return`\n# ${e}\n${t}\n\n## 快速开始\n### 安装依赖\n\`\`\`bash\n npm i\n\`\`\`\n\n### 本地开发\n\`\`\`bash\n npm run dev:{environment}:{platform}\n\`\`\`\n\n### 打包代码\n\`\`\`bash\n npm run build:{environment}:{platform}\n\`\`\`\n\n\n### 查找、修复代码问题\n\`\`\`bash\n z lint eslint\n # 或者\n z l e\n\`\`\`\n\n### 格式化代码\n\`\`\`bash\n z lint prettier\n # 或者\n z l p\n\`\`\`\n\n## 常见问题\n列出开发中常见的问题和解决方案\n\n## 了解更多\n在此处放入飞书文档链接。请在[前端团队-项目手册](https://hxhtbr8t8uy.feishu.cn/drive/folder/QfQ7favVWljQk7d63Prc8mUGnJf)中按照分类新建文档。\n`}(u.projectName,u.projectDesc)),y.succeed("模版初始化完成"),await Pe(u.tplLanguage,null===(a=u.group)||void 0===a?void 0:a.name);const j=f("依赖安装中").start();await L("npm install --registry https://registry.npmmirror.com/"),j.succeed("依赖安装完成");const b=f("项目推送中").start();await L('git add . && git commit -m "chore: 项目初始化"'),await L("git tag v0.0.1");const x=await(c={name:u.projectName,description:u.projectDesc,path:u.projectName,visibility:"private",namespace_id:null===(i=u.group)||void 0===i?void 0:i.id},be({url:`${ke()}/projects`,method:"post",data:c}));await L("git push -u origin master"),await L(`git push ${origin} HEAD:master --tags`),b.succeed(`项目已推送到远程,地址: ${s.blue(v)}`);const S=f("初始化分支中").start(),k=`feat_init_${R()}`,z=H(null===(r=u.group)||void 0===r?void 0:r.name)?["dev","test","release",k]:[k],O=await Promise.allSettled(z.map((e=>function(e){return be({url:`${ke()}/projects/${e.id}/repository/branches`,method:"post",data:e})}({id:x.id,branch:e,ref:"master"})))),N=[];O.forEach((({status:e},t)=>{"fulfilled"===e&&N.push(z[t])})),await L("git pull"),N.includes(k)?(await L(`git checkout -b ${k} origin/${k}`),S.succeed(`项目已切换到初始分支: ${s.blue(k)}`)):S.warn("开发分支检出失败!项目当前在主分支,请自行检出开发分支。");const T=`cd ${u.projectName} && z start`;console.log(`输入 ${s.green(T)} 开始开发吧~`),process.exit(0)}catch(e){W(e)}}async function Qe(e){const{targetBranch:t,platform:n,notify:a,projectName:i}=e,s=a?[...new Set([ae("weWorkUserId"),...a])]:[ae("weWorkUserId")];var r;await(r={username:ae("ldapAccount"),password:ae("ldapPassword"),projectName:i,branchName:t,params:{parameters:[{name:"platform",value:n},{name:"notify",value:s.join(",")}]}},_e({url:"/common/k8s/create/task",data:r}))}async function Ve(e){F(),M(),await G();let t=e;if(t&&!K.includes(t)&&W("仅支持发布指定环境分支"),!t){const e=await m({message:"请选择部署环境",choices:Q});t="production"===(n=e)?"master":n}var n;const a=await m({message:"请选择平台",choices:V}),i=await De("请选择部署成功要通知的人员:",(e=>!(e.length<1)||"至少选一个"),!0);var s;!function(e){var t;const n=ce(),a=le();(null===(t=null==a?void 0:a.repository)||void 0===t?void 0:t.url)||W(".z/project.json中缺少repository.url");const i=`build:${e}`;(null==n?void 0:n.scripts[i])||W(`项目package.json文件scripts不存在命令${i}。`)}("master"===(s=t)?"production":s);const r=ce();await Qe({projectName:r.name,targetBranch:t,platform:a,notify:i})}async function Ke(e,t,n){var i;const s=f(`分支合并中,${e} -> ${t}`).start();let r=0;try{const{iid:a}=await(o={title:`${e} -> ${t} by z-develop`,id:n.id,source_branch:e,target_branch:t,remove_source_branch:"master"===t},be({url:`${ke()}/projects/${o.id}/merge_requests`,method:"post",data:o}));r=a}catch(e){if(a.isAxiosError(e)){409===(null===(i=null==e?void 0:e.response)||void 0===i?void 0:i.status)&&W(`存在重复的合并请求,前往查看${n.mergeRequestUrl}`)}W(e)}var o;const c=async()=>{var e;try{await new Promise((e=>{setTimeout(e,5e3)})),await function(e){return be({url:`${ke()}/projects/${e.id}/merge_requests/${e.iid}/merge`,method:"put"})}({id:n.id,iid:r})}catch(t){if(a.isAxiosError(t)){406===(null===(e=null==t?void 0:t.response)||void 0===e?void 0:e.status)?await c():(await function(e){return be({url:`${ke()}/projects/${e.id}/merge_requests/${e.iid}`,method:"put",data:e})}({id:n.id,iid:r,state_event:"close"}),s.fail("合并过程中出现冲突,MR已关闭。建议在本地完成合并。"),process.exit(1))}else W(t)}};await c(),s.succeed(`分支${e}已合并到分支${t}`)}function Ze(){const e=O();if(t.existsSync(e)){const t=h.readJsonSync(e);if(t["lint-staged"])return t["lint-staged"];throw new Error("未找到lint配置")}throw new Error("请先初始化项目(z i .)。")}function Ye(){const e=function(e){if(e.includes("Merge")&&e.includes("# Conflicts:"))return!0;if(y.valid(e))return!0;const t=e.split(": ");if(1===t.length)return"body前缺少「: 」";const n=t[0];return/^(feat|fix|refactor)/.test(n)?!!/\(\d+(?:,\d+)*\)/.test(n)||"ID缺少或者格式不正确。":/^(chore|ci)/.test(n)?!!["chore","ci"].includes(n)||"chore/ci类型无需填写ID。":"BREAKING CHANGE"===n||"不存在的提交类型。"}(t.readFileSync(n.resolve(".git","COMMIT_EDITMSG"),"utf-8"));!0===e?(J("commit msg validate success."),process.exit(0)):W(`commit msg格式错误。${e}`)}async function Xe(e="latest"){await L(`npm i -g ${te.name}@${e} --registry https://registry.npmmirror.com/`),ye("latestCheckVersionTimestamp",Date.now()),ye("version",e),de(),process.exit(0)}async function et(){const e=$(process.argv.slice(2));if(!(e.h||e.help||e.v||e.version)&&(!["init","i"].includes(e._[0])||[".","prepare"].includes(e._[1])))if(A()){const e=h.readJSONSync(_());we(e.profile),ye(e.main),ve(e.constants),await async function(){const e=ie("latestCheckVersionTimestamp"),t=ie("versionCheckDuring");if(Date.now()-Number(e)>24*Number(t)*3600*1e3){let e;ve(await Ne());try{const t="TIME_OUT",n=new Promise((e=>{setTimeout((()=>e(t)),3e3)}));e=await Promise.race([n,L(`npm view ${te.name} version --registry https://registry.npmmirror.com/`)]),e!==t&&e!==te.version&&(console.log(`${s.blue(te.name)}本地版本为${te.version},低于线上版本${e},开始升级`),await Xe(e),console.log(s.green(`升级完成,当前版本${e}`)),process.exit(0)),ye("latestCheckVersionTimestamp",Date.now()),de()}catch(e){console.log("升级出错!请重试,或者手动升级"),W(e)}}}()}else W("请先初始化z-develop(执行 z i)。更多见https://hxhtbr8t8uy.feishu.cn/docx/RnOhdtqJWo5UooxHTaUcQFwtnKg")}e.command("create").alias("c").argument("[type]","可选值为project|pr, branch|b, commit-msg|cm, page|p, component|co").description("创建项目/分支/提交信息/页面/组件").action((async function(e){let a;e?["project","pr"].includes(e)?a="project":["branch","b"].includes(e)?a="branch":["page","pa"].includes(e)?a="page":["component","co"].includes(e)?a="component":["commit-msg","cm"].includes(e)?a="commit-msg":W("参数输入错误"):a=await m({message:"请选择你要创建的类型",choices:[{name:"提交信息(commit message)",value:"commit-msg"},{name:"分支(branch)",value:"branch"},{name:"页面(page)",value:"page"},{name:"组件(component)",value:"component"},{name:"项目(project)",value:"project"}]}),"project"===a?await He():"branch"===a?(F(),M(),await G(),await async function(){F(),M(),await G();const e=`${await m({message:"请选择创建分支的类型",choices:[{name:"开发新功能(feat)",value:"feat"},{name:"修复BUG(fix)",value:"fix"},{name:"重构/优化代码(refactor)",value:"refactor"}]})}_${await p({message:"请输入创建分支的目的(大小写字母、数字,小驼峰式命名。如userInfo)",validate:Le})}_${R()}`;await L("git fetch origin master"),await L(`git checkout -b ${e} origin/master`),await L(`git push -u origin ${e}`)}()):"commit-msg"===a?(F(),M(),await Ge()):"page"===a?(F(),M(),await async function(){const e="pages",a=await ge(e);t.mkdirSync(n.resolve("src",e),{recursive:!0});const i=await p({message:"请输入目录及页面名称(如user/list,不带文件后缀)",validate:Re});let s;if(i.includes("/")){const a=i.split("/");a.pop(),s=n.resolve("src",e,...i.split("/")),t.mkdirSync(n.resolve("src",e,...a),{recursive:!0})}else s=n.resolve("src",e,i);const{pages:r}=await Oe(),c=t.readdirSync(a),l=await m({message:"请选择模版",choices:r.filter((e=>c.includes(e.name))).map((e=>({name:`${e.name}: ${e.desc}(${e.author})`,value:e.name})))}),u=f("页面生成中...").start(),d=s+n.extname(n.join(a,l));o.cp("-r",n.join(a,l),d),P(d)?I(d).forEach((e=>{he(n.join(d,e),i)})):he(d,i),u.succeed("页面生成成功")}()):"component"===a&&(F(),M(),await async function(){const e="components",a=await ge("components");t.mkdirSync(n.resolve("src",e),{recursive:!0});const i=await p({message:"请输入目录及组件名称(如user/list,不带文件后缀)",validate:Re});let s;if(i.includes("/")){const a=i.split("/"),r=a.pop();s=n.resolve("src",e,...a,fe(r)),t.mkdirSync(n.resolve("src",e,...a),{recursive:!0})}else s=n.resolve("src",e,fe(i));const{components:r}=await Oe(),c=t.readdirSync(a),l=await m({message:"请选择模版",choices:r.filter((e=>c.includes(e.name))).map((e=>({name:`${e.name}: ${e.desc}(${e.author})`,value:e.name})))}),u=f("组件生成中...").start(),d=s+n.extname(n.join(a,l));o.cp("-r",n.join(a,l),d),P(d)?I(d).forEach((e=>{he(n.join(d,e),i)})):he(d,i),u.succeed("组件生成成功")}()),process.exit(0)})),e.command("merge").alias("m").description("合并当前分支到指定远程分支,并部署。").action((async function(){F(),M(),await G();const e=await pe();K.includes(e.sourceBranch)&&W(`当前分支${e.sourceBranch}不可作为源分支合并到目标分支。`);const t=await(n=e.id,be({url:`${ke()}/projects/${n}/repository/branches`}));var n;const a=H(e.group),i=t.filter((t=>![e.sourceBranch].includes(t.name))).map((e=>{let t=e.name;return a&&K.includes(t)&&(t=s.bold.blue(t)),{name:t,value:e.name}})),r=await m({message:"请选择要合并到的目标分支:",choices:i}),o=f(`${e.sourceBranch}分支推送中`).start();await L(`git ls-remote --heads origin ${e.sourceBranch}`)&&await L("git pull"),await L(`git push -u origin ${e.sourceBranch}`),o.succeed(`${e.sourceBranch}已推送到远程`),await async function(e,t,n){const a=`origin/${e.sourceBranch}`,i=await L("git log -b origin/master -1 --format=%H"),r=await L(`git log -b origin/${e.sourceBranch} -1 --format=%H`),o=await L(`git log ${i}...${r} -b ${a}`);if(o||W(`分支${e.sourceBranch}上不存在新的commit,不需要合并。`),await L(`git branch --contains ${i} -r ${a}`)||(console.log(s.yellow("注意:当前分支上不存在远程主分支最新代码,将在自动合入后继续。")),await Ke("master",e.sourceBranch,e)),"master"===t&&n){const e=o.split("\n").filter((e=>e)),t=await L("git log -b origin/test -1 --format=%H");(function(e,t){for(const n of t)if(!e.includes(n))return!1;return!0})((await L(`git log ${i}...${t} -b origin/test`)).split("\n").filter((e=>e)),e)||W("请先在测试环境发布要部署的代码")}}(e,r,a),await Ke(e.sourceBranch,r,e),"master"===r&&(await L("git checkout master"),await L("git pull"),await L(`git branch -d ${e.sourceBranch}`),J(`${e.sourceBranch}分支已移除,当前已切换到最新的master。如需继续开发,请检出新分支(z c b)。`)),a&&K.includes(r)&&await Ve(r)})),e.command("deploy").alias("d").description("发布指定远程分支").argument("[branchName]","可选值为dev/test/release/master").action(Ve),e.command("start").alias("s").description("启动本地开发环境").argument("[env]","可选值 dev|d, test|t, release|r, prod|p").action((async function(e){let t=e;t?["dev","test","release","production","d","t","r","p"].includes(e)?"d"===t?t="dev":"t"===t?t="test":"r"===t?t="release":"p"===t&&(t="production"):W("参数输入有误"):t=await m({message:"请选择环境",choices:Q});const n=`dev:${t}`;if(ce().scripts[n])try{o.exec(`npm run ${n}`,{silent:!1})}catch(e){W(e)}else W(`项目中(package.josn > scripts)不存在命令${n},请先添加!`)})),e.command("lint").alias("l").argument("[type]","非必填。可选值为commit-msg|cm, commit-files|cf, prettier|p, eslint|e。").option("--quiet","eslint只打印error").option("--max-warnings [VALUE]","eslint结果warn超过多少个会异常退出,默认为100",parseInt).description("执行 lint 脚本。包含 lint staged 、 lint commit message、 prettier、 eslint。").action((async function(e,a){let i;if(F(),M(),e?["commit-msg","cm"].includes(e)?i="commit-msg":["commit-files","cf"].includes(e)?i="commit-files":["prettier","p"].includes(e)?i="prettier":["eslint","e"].includes(e)?i="eslint":W("参数输入错误"):i=await m({message:"请选择你要执行的操作",choices:[{name:"全局执行 prettier",value:"prettier"},{name:"全局执行 eslint",value:"eslint"}]}),"commit-msg"===i)await Ye();else if("commit-files"===i)await async function(){await v({concurrent:!1,debug:!1,config:Ze()})?(J("代码风格检测通过!"),process.exit(0)):W("代码风格检测未通过!")}();else if("prettier"===i){t.existsSync(n.resolve("node_modules",".bin","prettier"))||W("该项目未安装prettier,请安装后重试");try{await L("npx prettier --write . --config .z/.prettierrc.json --ignore-path .z/.prettierignore --ignore-unknown --no-error-on-unmatched-pattern",{silent:!1}),J("prettier执行成功"),process.exit(0)}catch(e){W("prettier校验出错")}}else if("eslint"===i){t.existsSync(n.resolve("node_modules",".bin","eslint"))||W("该项目未安装eslint,请安装后重试");try{let e="npx eslint '**/*.{vue,js,jsx,cjs,mjs,ts,tsx,cts,mts}' --fix";"win32"===process.platform&&(e='npx eslint "**/*.{vue,js,jsx,cjs,mjs,ts,tsx,cts,mts}" --fix'),a.quiet&&(e+=" --quiet"),a.maxWarnings&&!isNaN(Number(a.maxWarnings))?e+=` --max-warnings ${Number(a.maxWarnings)}`:e+=" --max-warnings 100",await L(e,{silent:!1}),J("eslint执行成功"),process.exit(0)}catch(e){W("eslint校验出错")}}})),e.command("update").alias("u").argument("[version]","版本号","latest").description("升级").action(Xe),e.name("z-develop").alias("z").description(`z-develop, 开发流程管理工具。\n了解更多: ${s.blue("https://hxhtbr8t8uy.feishu.cn/docx/RnOhdtqJWo5UooxHTaUcQFwtnKg")}`).usage("<command> [options]").hook("preAction",(async function(){process.on("unhandledRejection",(e=>{})),process.on("uncaughtException",(e=>{})),o.config.fatal=!0,o.config.silent=!0,o.config.verbose=Boolean(void 0)||!1,await et()})).version(te.version,"-v, --version","当前版本号").helpOption("-h, --help","帮助").showHelpAfterError("可以使用z -h查看帮助。"),e.parse();
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "cli-z-develop",
3
- "version": "0.0.56",
3
+ "version": "0.0.58",
4
4
  "description": "前端本地开发命令行工具",
5
5
  "main": "dist/index.js",
6
6
  "bin": {
@@ -14,9 +14,9 @@
14
14
  "eslint": "eslint '**/*.{ts,js}' --fix",
15
15
  "prettier": "prettier -wu .",
16
16
  "upload": "npm run build && npm publish --access public --registry https://registry.npmjs.org/",
17
- "upload:patch": "npm version patch --no-commit-hooks && npm run upload",
18
- "upload:minor": "npm version minor --no-commit-hooks && npm run upload",
19
- "upload:major": "npm version major --no-commit-hooks && npm run upload"
17
+ "upload:patch": "npm version patch && npm run upload",
18
+ "upload:minor": "npm version minor && npm run upload",
19
+ "upload:major": "npm version major && npm run upload"
20
20
  },
21
21
  "type": "module",
22
22
  "repository": {
@@ -34,10 +34,11 @@
34
34
  "@types/inquirer": "^9.0.7",
35
35
  "@types/minimist": "^1.2.5",
36
36
  "@types/node": " ^22.14.1",
37
+ "@types/semver": "^7.7.0",
37
38
  "@types/shelljs": "^0.8.15",
38
- "@typescript-eslint/eslint-plugin": "^8.30.1",
39
- "@typescript-eslint/parser": "^8.30.1",
40
- "eslint": "^9.24.0",
39
+ "@typescript-eslint/eslint-plugin": "^8.31.0",
40
+ "@typescript-eslint/parser": "^8.31.0",
41
+ "eslint": "^9.25.1",
41
42
  "jiti": "^2.4.2",
42
43
  "prettier": "^3.5.3",
43
44
  "rollup": " 4.40.0",
@@ -57,6 +58,7 @@
57
58
  "minimist": "^1.2.8",
58
59
  "ora": "^8.2.0",
59
60
  "rxjs": "^7.8.2",
61
+ "semver": "^7.7.1",
60
62
  "shelljs": "^0.9.2"
61
63
  }
62
64
  }