jja 2.3.13 → 2.3.15

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/bin.mjs +1 -1
  2. package/package.json +2 -2
package/bin.mjs CHANGED
@@ -1,2 +1,2 @@
1
1
  #!/usr/bin/env node
2
- import{Dog as e}from"@qqi/log";import{isFalse as n,isUndefined as t,isArray as o,isString as i,isEmptyObject as a,isNull as r,isBusinessEmptyString as s,isPlainObject as c}from"a-type-of-js";import{Command as l}from"a-command";import{redPen as p,hexPen as d,randomPen as u,greenPen as g,yellowPen as f,pen as m,hidePen as w,cyanPen as $,boldPen as h,strInOneLineOnTerminal as y,italicPen as v,magentaPen as b}from"color-pen";import{_p as k,isWindows as j,runOtherCode as S,fileExist as R,cursorMoveUp as O,cursorAfterClear as D,getDirectoryBy as V,readFileToJsonSync as A,pathJoin as E,getNpmPkgInfo as N,cursorLineClear as T,colorLine as P,isTTY as q,__p as C,typewrite as L}from"a-node-tools";import{sleep as M,intersection as x}from"a-js-tools";import{copyTextToClipboard as I}from"@qqi/copy-text";import{terminalRegExp as F}from"@color-pen/static";import{Table as _}from"colored-table";import G,{Resolver as U}from"node:dns";import W from"node:net";const Y=new e({name:"jja",type:!1});n(Y.type);const B=new l("jja");B.bind({"package <pkg> (包管理)":["--diff <-d> (分析当下包的差异)","--upDependencies <-u> (更新依赖,强制更新到 'latest',使用需谨慎)"],"remove <rm> (做一个简单的兼容的移除文件或文件夹的命令)":["--ignore <-i> (不打印日志(默认打印的))","--subdirectories <-s> (这是一个危险的命令,用于 windows 下递归删除执行命令下所有的给定文件名称)"],"update <up> (做一个简单的 npm 升级程序,对,简单的)":["--ignore <-i> (不建议你这么用,你会发现你像个傻子一样在那等结果)",`--dependencies <-d> (更新依赖,跟 ${u("npm update")} 一样)`],"clearScreen <cls> (清理终端显示屏幕,同 clearTerminal )":"","clearTerminal <clear> (清理终端显示屏幕,同 clearScreen )":"","git (一些关于 git 的操作)":[`commit (git 提交代码,是 ${p("commit")} 提交啊,不是 ${d("#666")("push")} 推送)`,"merge (合并两个分支)","tag (给提交打上 tag)"],"dns (使用 dns 获取 ip 地址)":["--domain <-d> (获取给定域名获取 ip 地址,默认 github.com )","--port <-p> (配置要校验的 port 端口号,默认为 443)","--not-covered <-n> (不覆盖解析过程,默认不展示)"]}).run(),Y("参数已绑定,且已解析用户参数",B.args),Y.warn("当前的执行状态",B.state),B.isEnd(!0);const H={log:!1,subdirectories:!1};async function X(e,n,t=5){return!await Reflect.apply(e,void 0,n)&&t--?(k(g("执行失败,现在重试中")),Y.warn(n,"执行失败,第",5-t,"次执行"),await X(e,n,t)):await Reflect.apply(e,void 0,n)}function J(e,n){k(`清理 ${e} 文件${n?"完成":"失败"}`)}async function z(e,n){H.log||k(d("#aaa")(` ${e} 文件/夹存在,准备删除`));let t={error:null};return j?n.isDirectory()?(t=await S(`rd /q /s ${e.replace(/\//,"\\")}`),J(e,t.success)):n.isFile()&&(t=await S(`del ${H.subdirectories?"/s":""} /q ${e.replace(/\//,"\\")}`),J(e,t.success)):(t=await S(`rm -rf ${e.replace(/\\/,"/")}`),J(e,t.success)),!!t.success||(console.log(t.error),!1)}async function K(e){H.log||k(d("#336")(`当前清理文件为 ${e}`));const n=R(e);H.log||k(d("#666")(`正在检测 ${e} 文件/夹是否存在`)),Y(`文件${e} 数据为 ${n}`),t(n)?H.log||k(f(`${e} 文件不存在`)):await X(z,[e,n])}async function Q(e=!0){const n=V("package.json","file");if(null==n)return k("当前目录下不存在 package.json 文件");const t=A(E(n,"package.json"))?.dependencies||{};if(e&&k(g("初始化完成,等待下一步命令")),Object.keys(t).length>0){let e="npm install --save";for(let n=Object.keys(t),o=n.length,i=0;i<o;i++){e+=` ${n[i]}@latest `}await S(e)}var o;e&&k($("正在更新 dev 依赖")),await S("npm update --save"),e&&(o="开发依赖更新完毕",O(),D(),k(o))}const Z={log:!0};async function ee(e){Boolean(e["--dependencies"])&&await Q(Z.log)}const ne={local:null,online:null,dependenceList:{},updateDependence:[],latestDependence:[],binning:function(e,n=!1){if(t(e)||a(e)||r(this.local))return;const{overrides:o}=this.local;for(const t in e){if(i(o?.[t])){B.WARN(`${t} 被锁定在 ${o[t]}`);continue}const a=e[t];this.dependenceList[t]={type:n?"devDependencies":"dependencies",version:a,localVersion:"",latestVersion:"",onlineVersion:"",tag:"",time:""}}},preReleaseDependence:[],timeoutDependence:[],registry:void 0};const te=h.green,oe=d("0x993311"),ie=d("0x339999"),ae=d("0x666666");function re(e){const n=A(E("node_modules/",e,"package.json"));let t="";return t=r(n)?"":n.version||"",ne.dependenceList[e].localVersion=t,Y(e,"本地安装的版本为:",t),t}async function se(e){const n=await N(e,ne.registry,4567);if("timeout"===n.status)return void Y.warn("请求超时");if(r(n.data))return void Y.warn("获取包信息失败");const t=Date.now();Y("获取数据",e,"包信息");const o=ne.dependenceList[e],{"dist-tags":i,time:a,version:s}=n.data;let c=s,l="latest";o.onlineVersion=c,o.tag=l;const p=Object.keys(i);Y("共需比对",p.length);for(let e=0,n=p.length;e<n;e++){const n=p[e],t=i[n];a[t]>a[c]&&(c=t,l=n)}Y("比对",e,"完成"),Y("耗时",Date.now()-t),"latest"!==l&&(Y(e,"最新版本为:",c,"tag为:",l),o.tag=l,o.latestVersion=c);const d=new Date(a[c]).toLocaleString();o.time=d,Y(`${e} 的最后发布时间为:${d}`)}async function ce(e){const{msg:n,list:t,type:a,copy:r}=e,s=m[a],c=function(){const e=e=>Boolean(V(e,"file"));if(e("pnpm-lock.yaml"))return"pnpm";if(e("yarn.lock"))return"yarn";if(e("package-lock.json"))return"npm";const n=process.env.npm_config_user_agent||"";return n.includes("pnpm")?"pnpm":n.includes("yarn")?"yarn":n.includes("npm")?"npm":process.env.PNPM_HOME?"pnpm":process.env.YARN_IGNORE_PATH?"yarn":"npm"}(),l="npm"===c?"npm install --save":"yarn"===c?"yarn add":"pnpm add";k(),k(s.reversed(n),!1),k(r?ae.reversed`已复制到剪切板 📋`:""),k(),k(`${s(l)} ${w`\\`}\n${function(e){if(!o(e)||e.some(e=>!i(e)))throw new TypeError("参数必须为字符串数组");return e.map((n,t)=>t%3==2&&t!==e.length-1?`${n} ${w`\\`}\n`:n.concat(" ")).join("")}(t)}`),k(),r&&await I(`npm install --save ${t.join(" ")}`.replace(F(),""))}function le(e){return`${h(e)}@${te`latest`}`}function pe(e,n){return`${v(e)}@${v.random(n)}`}async function de(){const{local:e,preReleaseDependence:n,latestDependence:t,dependenceList:o,timeoutDependence:i}=ne;if(r(e))return;const{dependencies:a,devDependencies:s}=e;if(ne.binning(a),ne.binning(s,!0),await async function(){const{dependenceList:e,timeoutDependence:n}=ne,t=Object.keys(e);let o=!1;const i=()=>{O(),T(!0),D(!0)};for(let a=0,r=t.length;a<r;a++){const r=t[a],s=e[r],c=e[r].version;o&&i(),B.CURRENT(y(`获取 ${r} 的本地安装信息`)),await M(36),i();const l=re(r);if(B.SUCCESS(y(`${r} 的本地安装版本:${s.localVersion}`)),await M(36),await se(r),i(),""===s.onlineVersion&&""===s.latestVersion){n.push(r),B.ERROR(`${r} 本地 ${s.localVersion} 请求错误`),o=!1;continue}if(l===s.onlineVersion){let e=`${r} 的本地${c} 安装版本为 ${l} 最新是 ${s.onlineVersion} `;if(""===s.latestVersion){B.INFO(y(e)),o=!0,await M(36);continue}o=!1,ne.preReleaseDependence.push(r),e+=`;最新预发布版本为 ${g(s.latestVersion)}`,B.INFO(e),await M(36);continue}if(o=!1,l===s.latestVersion){B.INFO(`${r} 的本地版本${c} 安装版本为 ${l}`),await M(36);continue}let p=`${r} 的本地 ${g(c)} 安装版本为 ${g(l)} 最新是 ${ie(s.onlineVersion)} `;ne.latestDependence.push(r),""!==s.latestVersion?(ne.preReleaseDependence.push(r),p+=`;最新预发布版本为 ${g(s.latestVersion)}`,B.INFO(p),await M(36)):(B.INFO(p),await M(36))}!0===o&&(i(),o=!1)}(),0===n.length&&0===t.length&&0===i.length)return k(ae`看起来似乎没有依赖版本差异`);if(0===n.length&&0===t.length)return k(p`看起来网络不太好讷,所有的包线上版本的请求都出错了。或者是还没有执行 npm install 呐`);k(m.brightGreen`\n版本差异的依赖为:\n`);const c=[...t,...n];var l;new _({header:[$("包名"),{content:"发布时间",color:"#f26"},g`最新版本`],body:[...c.map(e=>{const{type:n,latestVersion:t,time:i,onlineVersion:a}=o[e];return["dependencies"===n?h(e):v(e),u(i),t||v(a)]})]})(),k(m.brightRed(`\n目前仅关注版本号是否为最新 ${m.brightMagenta("latest")}`)),k("使用 npm install --save 命令安装更新\n"),i.length>0&&(l=p`有一些包没有返回结果,请注意:`,k(y(l)),new _([...i.map(e=>[e])])());const d=[...t.filter(e=>!1===n.includes(e)).map(e=>m.bold(e).concat("@latest")),...n.map(e=>pe(e,o[e].tag))],f=t.map(e=>le(e)),w=[...t.map(e=>le(e)),...n.filter(e=>!1===t.includes(e)).map(e=>pe(e,o[e].tag))],b=d.length<n.length+t.length,j=n.length>0,S=t.length>0;j&&(b&&await ce({msg:"‼️ 预发布版本优先:",list:d,type:"brightRed"}),await ce({msg:"⚠️ latest 版本优先:",list:w,type:"brightMagenta",copy:!S})),S&&await ce({msg:"🎉 最佳安装:",list:f,type:"brightGreen",copy:!0})}async function ue(e){ne.registry=e[0]?.toString(),await async function(){const e=V("package.json","file");if(t(e))return void Y.warn("未找到当前包的 package.json 文件的位置");const n=A(E(e,"package.json"));if(r(n))return void Y.error("未找到当前包的 package.json 文件,该事件发生的概率极低");const o=n.name||"",i=await N(o,ne.registry);ne.local=n,ne.online=i.data}();const{local:n,online:o}=ne,i=n?.version||"",a=o?.version||"";if(r(n))return Y("未找到包 package.json 文件退出"),k(oe("未找到当前包 package.json"));const s=oe(`当前包本地版本为: ${i}`);if(o){k(s);const e=" ".repeat(6);k(ie(`${e}线上版本为:${a}@latest`));const n=new Date(o&&o.time.modified||"").toLocaleString();B.INFO(`最后的发布时间为: ${n.toString()}`),P()}await de()}async function ge(e){e["--upDependencies"]?await Q(!0):e["--diff"]?await ue(e["--diff"]):B.help("package")}async function fe(e){const n=V("package.json","file",process.cwd());if(null==n)return!0;const t=A(E(n,"package.json"));if(r(t)||!t.version)return!0;const o=t.version;await S({code:`git tag -a v${o} -m '${e}'`,cwd:n}),await S({code:"git push origin --tag",cwd:n})}async function me(e){e.commit?await async function(e="版本维护",n=!1){const t=V(".git","directory");if(null==t)return k(d("#ff0")("not a git repository(当前目录非 git 储存库)"));const o=await S({code:"git add .",cwd:t});if(!o.success)return null!=console.log(o.error);const i=await S({code:"git status",cwd:t});if(!/nothing to commit, working tree clean/.test(i.data||"")){const o=await S({code:"git add .",cwd:t});if(!o.success)return null!=console.log(o.error);""==e.trim()&&(e=await B.question({text:"请输入提交信息"})??"");const i=await S({code:`git commit -m "${e}"`,cwd:t});return i.success?(i.error&&console.log(i.error),await S({code:"git push",cwd:t}),n&&await fe(e),!0):null!=console.log(i.error)}return!1}(e.commit.join(" ")):e.merge?await async function(e){if(s(e)){const n=await S("git branch -a");console.log(n.data),e=await B.question({text:"请输入要合并分支的名称"})??""}if(s(e))return B.ERROR("没有获取到要合并的分支");const n=await B.selection({data:[{label:"正常快进合并",value:""},{label:"非快进合并 (--no-ff)",value:"--no-ff"},{label:"多提交记录合并为一条 (--squash)",value:"--squash",tip:"适用于将众多小的提交合并成一个提交"}]});t(n)?B.ERROR("您选择退出合并"):await S(`git merge ${e} ${n}`)}(e.merge.join("")):e.tag?await async function(){const e=await B.question({text:"请输入待标记的信息"});t(e)||await fe(e)}():B.help("git")}const we=["A","AAAA","ANY","CAA","CNAME","MX","NAPTR","NS","PTR","SOA","SRV","TLSA","TXT"],$e={domain:"github.com",port:443,notCovered:!1,dnsServers:["1.1.1.1","8.8.8.8","8.8.4.4","9.9.9.9","208.67.222.222","114.114.115.115","223.5.5.5","180.76.76.76"],ips:{},rrtype:["A"],reset(e){this.ips={},this.domain="github.com",this.port=443,this.notCovered=!1,this.rrtype=[],t(e)||e.forEach(e=>{const n=e["--domain"];!t(n)&&n.length>0&&(this.domain=n[0].toString());const o=e["--port"];!t(o)&&o.length>0&&(this.port=Number(o[0]));const i=e["--not-covered"];t(i)||!1===i[0]||(this.notCovered=!0);const a=e["--rrtype"];if(!t(a)&&a.length>0){const e=x(we,a);e.length>0&&(this.rrtype=e)}})}};async function he(e,n=3e3){const{ips:t,port:o}=$e;if(void 0===t[e])return Y(`校验 ${e} 的联通性`),new Promise(t=>{const i=new W.Socket;i.setTimeout(n),i.on("connect",()=>{i.destroy(),t(!0)}),i.on("timeout",()=>{i.destroy(),t(!1)}),i.on("error",()=>{i.destroy(),t(!1)}),Y("测试的接口为:",o),i.connect(o,e)});Y(`${e} 联通性已校验过 ${t[e]?"🔥":"⛓️‍💥"}`)}async function ye(e="1.1.1.1"){const n=new U;n.setServers([e]);const{domain:o,ips:i}=$e;try{const a=await new Promise(t=>{n.resolve4(o,(n,i)=>{if(n)return Y.error(`使用 ${e} 获取 ${o} 的 ip 出错`,n),void t(void 0);Y(`使用 ${e} 获取 ${o} 的 ip 值为 ${i.join(" --- ")}`),t(i)})});if(t(a))return{dnsServer:e,error:"获取 ip 错误"};const r=await Promise.all(a.map(async e=>{const n=await he(e);return t(n)||(i[e]=n),{ip:e.padEnd(16," "),isAlive:t(n)?i[e]:n}}));return{dnsServer:e,results:r}}catch(n){return Y.error("获取 ip 错误",n),{dnsServer:e,error:n}}}async function ve(e){$e.reset(e.options),await async function(){const{domain:e,ips:n}=$e;try{const o=await new Promise(n=>{G.lookup(e,{all:!0,family:4},(t,o)=>{if(t)return Y.error(`获取本地 ${e} 的 ip 出错`,t),void n(void 0);Y(`获取本地 ${e} 的 ip 值为 ${o.join(" --- ")}`),n(o.map(e=>e.address))})});if(t(o))return{error:"获取 ip 错误"};const i=await Promise.all(o.map(async e=>{const o=await he(e);return t(o)||(n[e]=o),{ip:e.padEnd(16," "),isAlive:t(o)?n[e]:o}}));return i.length>0&&(k(oe`本地配置 ${e} 的 ip 地址及联通性为:\n`),i.forEach(e=>{k($`- ${e.ip}`,!1),k(e.isAlive?oe` -> ✅`:ae` -> ❌`)}),k()),{results:i}}catch(e){return Y.error(e),{error:e}}}(),await async function(){const{domain:e,dnsServers:n,notCovered:o}=$e;let i=0;k(`${e} ip 列表:`);for(const e of n){const{dnsServer:n,results:a}=await ye(e);t(a)?Y.warn(`${n} 获取 ip 失败`):(o||0===i||(O(i),D(!0)),k(),k(m.reversed`DNS: ${n}`),k(),a.forEach(({ip:e,isAlive:n})=>{const t=n?`${g(e)} ${oe`->`} ✅`:`${ae(e)} ${p`⛓️‍💥`} ❌`;k(y(t))}),i=3+a.length)}o||(O(i+1),D(!0))}();const{ips:n}=$e;Object.keys(n).length>0?await async function(){const{domain:e,ips:n,port:t}=$e;await L(`${m.brightMagenta`${e}`} 域名解析结果:`),k();for(const e in n)if(Object.prototype.hasOwnProperty.call(n,e)){const t=n[e];k(t?`${g`- ${e.padEnd(16)}`} ${oe`->`} ✅`:`${ae`- ${e.padEnd(16)}`} ${p`⛓️‍💥`} ❌`)}k(),await L(p.italic.dim`${e} 联通性接口判断为 ${t.toString()}`),k()}():await async function(){const{domain:e,port:n}=$e;await L(`\n找不到 ${p(e)} 的服务器 IP 地址 ${ae`测试使用 ${n.toString()} 端口`}\n`)}()}const be=B.args.$arrMap;0===B.args.$only.length&&(Y("没有匹配到子命令,打印帮助信息并退出"),B.help(),B.end());try{await async function e(){if(0===be.length)return;const n=be.shift();if(t(n))return await e();"remove"in n?(Y("执行文件移除"),await async function(e){const n=e.value||[],t=e.options;for(t?.forEach(e=>{const t=e["--ignore"];t&&(H.log=!0,n.concat(t));const o=e["--subdirectories"];o&&(H.subdirectories=!0,n.concat(o))}),H.log||k(m.brightCyan`当前系统为: ${j?"windows":"linux/mac"}`),0==n.length&&(Y.warn("待删除列表",n),k(m.random`没有待清理的文件/文件夹`));n.length;){let e=n.pop();e=void 0===e?"undefined":e.toString(),await K(e)}}(n.remove)):"clearScreen"in n||"clear"in n?(Y("执行清屏"),await async function(){q()?(Y("执行第一遍清理"),await S({code:j?"cls":"clear",printLog:!1}),Y("执行第二遍清理"),C("3J"),C("c",!1),O(1/0),D(!0)):(Y.warn("当前系统不支持 TTY"),k("当前环境不支持 "))}()):"git"in n?(Y("执行 git 相关命令"),await async function(e){if(c(e)&&(e={value:[]}),t(e.options)||0===e.options.length)return Y.warn("没有配置项,直接返回的帮助文档"),B.help("git");const n=e.options;for(let e=0,t=n.length;e<t;e++)Y("本次执行",n[e]),await me(n[e])}(n.git)):"package"in n?(Y("执行 package 相关命令"),await async function(e){if(t(e.options)||0==e.options.length)return Y.warn("没有参数输入"),B.help("package");const n=e.options;for(let e=0,t=n.length;e<t;e++)await ge(n[e])}(n.package)):"update"in n?(Y("执行 update 相关命令"),await async function(e){if(t(e.options)||0==e.options.length)return Y.warn("没有匹配到选项,自动抛出帮助文档"),B.help("update");const n=e.options;Z.log=!n.some(e=>e["--ignore"]);for(let e=0,t=n.length;e<t;e++)await ee(n[e])}(n.update)):"dns"in n?(Y("执行 dns 相关的命令"),await ve(n.dns)):"runOtherCode"in n?Y("执行运行其他命令"):"run"in n&&(Y("执行执行"),await async function(){const e=V("package.json","file");if(null==e)return k(b`当前目录下不存在 package.json 文件`);const n=A(E(e,"package.json"))?.scripts||{};if(a(n))return k($`没有可执行的命令`);k(n);const o=(()=>{const e=Object.keys(n);return e.filter(n=>{if(!n.startsWith("pre")&&!n.startsWith("post"))return!0;const t=n.replace(/^(pre|post)/,"");return!e.includes(t)&&!["build","install","publish","pack","version","test","start","restart","stop","publishOnly","deploy","pare"].includes(t)}).map(e=>({value:n[e],label:e,tip:n[e]}))})(),i=await B.selection(o);Y(i),t(i)||await S({code:i})}());try{await e()}catch(e){Y.error("执行 run 报错",e)}}(),P(" 终结分割线 ",!0)}catch(e){Y.error("执行 run 报错",e)}
2
+ import{Dog as e}from"@qqi/log";import{isFalse as t,isUndefined as n,isArray as o,isString as r,isEmptyObject as i,isNull as s,isBusinessEmptyString as a,isPlainObject as c,isEmptyArray as l}from"a-type-of-js";import{Command as p,enPrefixList as d}from"a-command";import{redPen as u,hexPen as g,randomPen as m,greenPen as f,yellowPen as w,pen as $,strInOneLineOnTerminal as y,hidePen as h,cyanPen as v,boldPen as b,italicPen as j}from"color-pen";import{_p as k,isWindows as D,runOtherCode as S,fileExist as V,detectPackageManager as A,cursorMoveUp as R,cursorAfterClear as E,getDirectoryBy as O,readFileToJsonSync as P,pathJoin as q,getNpmPkgInfo as T,waitingTips as C,colorLine as L,isTTY as x,__p as N,typewrite as M}from"a-node-tools";import F from"node:https";import{copyTextToClipboard as G}from"@qqi/copy-text";import{terminalRegExp as I}from"@color-pen/static";import{Table as X}from"colored-table";import{intersection as Y}from"a-js-tools";import B,{Resolver as H}from"node:dns";import J from"node:net";const U=new e({name:"jja",type:!1});t(U.type);const W=new p("jja");W.bind({"package <pkg> (包管理)":["--diff <-d> (分析当下包的差异)","--upDependencies <-u> (更新依赖,强制更新到 'latest',使用需谨慎)"],"remove <rm> (做一个简单的兼容的移除文件或文件夹的命令)":["--ignore <-i> (不打印日志(默认打印的))","--subdirectories <-s> (这是一个危险的命令,用于 windows 下递归删除执行命令下所有的给定文件名称)"],"update <up> (做一个简单的 npm 升级程序,对,简单的)":["--ignore <-i> (不建议你这么用,你会发现你像个傻子一样在那等结果)",`--dependencies <-d> (更新依赖,跟 ${m("npm update")} 一样)`],"clearScreen <cls> (清理终端显示屏幕,同 clearTerminal )":"","clearTerminal <clear> (清理终端显示屏幕,同 clearScreen )":"","git (一些关于 git 的操作)":[`commit (git 提交代码,是 ${u("commit")} 提交啊,不是 ${g("#666")("push")} 推送)`,"merge (合并两个分支)","tag (给提交打上 tag)"],"dns (使用 dns 获取 ip 地址)":["--domain <-d> (获取给定域名获取 ip 地址,默认 github.com )","--port <-p> (配置要校验的 port 端口号,默认为 443)","--not-covered <-n> (不覆盖解析过程,默认不展示)"]}).run(),U("参数已绑定,且已解析用户参数",W.args),U.warn("当前的执行状态",W.state),W.isEnd(!0);const _={log:!1,subdirectories:!1};async function z(e,t,n=5){return!await Reflect.apply(e,void 0,t)&&n--?(k(f("执行失败,现在重试中")),U.warn(t,"执行失败,第",5-n,"次执行"),await z(e,t,n)):await Reflect.apply(e,void 0,t)}function K(e,t){k(`清理 ${e} 文件${t?"完成":"失败"}`)}async function Q(e,t){_.log||k(g("#aaa")(` ${e} 文件/夹存在,准备删除`));let n={error:null};return D?t.isDirectory()?(n=await S(`rd /q /s ${e.replace(/\//,"\\")}`),K(e,n.success)):t.isFile()&&(n=await S(`del ${_.subdirectories?"/s":""} /q ${e.replace(/\//,"\\")}`),K(e,n.success)):(n=await S(`rm -rf ${e.replace(/\\/,"/")}`),K(e,n.success)),!!n.success||(console.log(n.error),!1)}async function Z(e){_.log||k(g("#336")(`当前清理文件为 ${e}`));const t=V(e);_.log||k(g("#666")(`正在检测 ${e} 文件/夹是否存在`)),U(`文件${e} 数据为 ${t}`),n(t)?_.log||k(w(`${e} 文件不存在`)):await z(Q,[e,t])}const ee=(()=>{const e=A();return"npm"===e?"npm install --save":"yarn"===e?"yarn add":"pnpm add"})();async function te(e=!0){const t=O("package.json","file");if(null==t)return k("当前目录下不存在 package.json 文件");const n=P(q(t,"package.json"))?.dependencies||{};if(e&&k(f("初始化完成,等待下一步命令")),Object.keys(n).length>0){let e="npm install --save";for(let t=Object.keys(n),o=t.length,r=0;r<o;r++){e+=` ${t[r]}@latest `}await S(e)}var o;e&&k(v("正在更新 dev 依赖")),await S("npm update --save"),e&&(o="开发依赖更新完毕",R(),E(),k(o))}const ne={log:!0};async function oe(e){Boolean(e["--dependencies"])&&await te(ne.log)}const re={local:null,online:null,dependenceList:{},updateDependence:[],latestDependence:[],binning:function(e,t=!1){if(n(e)||i(e)||s(this.local))return;const{overrides:o}=this.local;for(const n in e){if(r(o?.[n])){W.WARN(`${n} 被锁定在 ${o[n]}`);continue}const i=e[n];this.dependenceList[n]={type:t?"devDependencies":"dependencies",version:i,localVersion:"",latestVersion:"",onlineVersion:"",tag:"",time:""}}},preReleaseDependence:[],timeoutDependence:[],registry:void 0};const ie=b.green,se=g("0x993311"),ae=g("0x339999"),ce=g("0x666666");const le=C({show:!1,info:"请稍等",beforeDestroyed:e=>e&&W.end()});async function pe(){const e={"淘宝":"registry.npmmirror.com","官方":"registry.npmjs.org","腾讯":"mirrors.tencent.com/npm","中科大":"npmreg.proxy.ustclug.org",yarn:"registry.yarnpkg.com"},n=()=>Object.keys(e).map(t=>function(e){const t=Date.now();return new Promise(n=>{const o=`https://${e}/nry`,r=F.request(o,{method:"HEAD",timeout:3500},o=>{n((o?.statusCode??0)>=200&&(o?.statusCode??0)<300?{url:e,time:Date.now()-t,success:!0}:{url:e,time:Date.now()-t,success:!1})});r.on("error",()=>n({url:e,time:Date.now()-t,success:!1})),r.on("timeout",()=>{r.destroy(),n({url:e,time:Date.now()-t,success:!1})}),r.end()})}(e[t])),o={};for(let e=0;e<3;e++){le.run(`请稍等,正在进行第${["一","二","三","四","五"][e]}次安装源延迟测试`);(await Promise.all(n())).forEach(e=>{t(e.success)||(o[e.url]?o[e.url]=(o[e.url]+e.time)/2:o[e.url]=e.time)})}return le.destroyed(),Object.keys(e).reduce((t,n)=>o[e[n]]<(o[e[t]]??1/0)?n:t,"")}async function de(){const{dependenceList:e,timeoutDependence:t}=re,o=Object.keys(e);o.length>18&&(n(re.registry)||!["官方","淘宝","腾讯","中科大","yarn"].some(e=>e===re.registry))&&(re.registry=await pe(),k(`由于您未设置或者设置有误,现使用 ${v(re.registry)} npm 源`));const r=o.map(n=>new Promise(o=>{(async()=>{const r=e[n],i=e[n].version;le.run(`${d.current()}获取 ${n} 的本地安装信息`);const a=function(e){const t=P(q("node_modules/",e,"package.json"));let n="";return n=s(t)?"":t.version||"",re.dependenceList[e].localVersion=n,U(e,"本地安装的版本为:",n),n}(n);if(le.run(`${d.success()} ${n} 的本地安装版本:${r.localVersion}`),await async function(e){const t=await T(e,re.registry,9867);if("timeout"===t.status)return void U.warn("请求超时");if(s(t.data))return void U.warn("获取包信息失败");const n=Date.now();U("获取数据",e,"包信息");const o=re.dependenceList[e],{"dist-tags":r,time:i,version:a}=t.data;let c=a,l="latest";o.onlineVersion=c,o.tag=l;const p=Object.keys(r);U("共需比对",p.length);for(let e=0,t=p.length;e<t;e++){const t=p[e],n=r[t];i[n]>i[c]&&(c=n,l=t)}U("比对",e,"完成"),U("耗时",Date.now()-n),"latest"!==l&&(U(e,"最新版本为:",c,"tag为:",l),o.tag=l,o.latestVersion=c);const d=new Date(i[c]).toLocaleString();o.time=d,U(`${e} 的最后发布时间为:${d}`)}(n),""===r.onlineVersion&&""===r.latestVersion)return t.push(n),le.log(`${d.error()} ${n} 本地 ${r.localVersion} 请求错误`),o(!0);if(a===r.onlineVersion){const e=`${n} 的本地${i} 安装版本为 ${a} 最新是 ${r.onlineVersion} `;return""===r.latestVersion?(le.run(d.info()+e),o(!0)):(re.preReleaseDependence.push(n),le.log(`${d.info()} ${e};最新预发布版本为 ${f(r.latestVersion)}`),o(!0))}if(a===r.latestVersion)return le.log(`${d.info()} {key} 的本地版本${i} 安装版本为 ${a}`),o(!0);const c=`${n} 的本地 ${f(i)} 安装版本为 ${f(a)} 最新是 ${ae(r.onlineVersion)} `;re.latestDependence.push(n),""===r.latestVersion?(le.log(d.info()+c),o(!0)):(re.preReleaseDependence.push(n),le.log(`${d.info()} ${c};最新预发布版本为 ${f(r.latestVersion)}`),o(!0))})()}));return Promise.all(r).then(()=>{le.destroyed()})}async function ue(e){const{msg:t,list:n,type:i,copy:s}=e,a=$[i];k(),k(a.reversed(t),!1),k(s?ce.reversed`已复制到剪切板 📋`:""),k(),k(`${a(ee)} ${h`\\`}\n${function(e){if(!o(e)||e.some(e=>!r(e)))throw new TypeError("参数必须为字符串数组");return e.map((t,n)=>n%3==2&&n!==e.length-1?`${t} ${h`\\`}\n`:t.concat(" ")).join("")}(n)}`),k(),s&&G(`${ee} ${n.join(" ")}`.replace(I(),""))}function ge(e){return`${b(e)}@${ie`latest`}`}function me(e,t){return`${j(e)}@${j.random(t)}`}async function fe(){const{local:e,preReleaseDependence:t,latestDependence:n,dependenceList:o,timeoutDependence:r}=re;if(s(e))return;const{dependencies:i,devDependencies:a}=e;if(re.binning(i),re.binning(a,!0),await de(),0===t.length&&0===n.length&&0===r.length)return k(ce`看起来似乎没有依赖版本差异`);if(0===t.length&&0===n.length)return k(u`看起来网络不太好讷,所有的包线上版本的请求都出错了。或者是还没有执行 npm install 呐`);k($.brightGreen`\n版本差异的依赖为:\n`);const c=[...n,...t];var l;new X({header:[v("包名"),{content:"发布时间",color:"#f26"},f`最新版本`],body:[...c.map(e=>{const{type:t,latestVersion:n,time:r,onlineVersion:i}=o[e];return["dependencies"===t?b(e):j(e),m(r),n||j(i)]})]})(),k($.brightRed(`\n目前仅关注版本号是否为最新 ${$.brightMagenta("latest")}`)),k(`使用 ${ee} 命令安装更新\n`),r.length>0&&(l=u`有一些包没有返回结果,请注意:`,k(y(l)),new X([...r.map(e=>[e])])());const p=[...n.filter(e=>!1===t.includes(e)).map(e=>$.bold(e).concat("@latest")),...t.map(e=>me(e,o[e].tag))],d=n.map(e=>ge(e)),g=[...n.map(e=>ge(e)),...t.filter(e=>!1===n.includes(e)).map(e=>me(e,o[e].tag))],w=p.length<t.length+n.length,h=t.length>0,D=n.length>0;h&&(w&&await ue({msg:"‼️ 预发布版本优先:",list:p,type:"brightRed"}),await ue({msg:"⚠️ latest 版本优先:",list:g,type:"brightMagenta",copy:!D})),D&&await ue({msg:"🎉 最佳安装:",list:d,type:"brightGreen",copy:!0})}async function we(e){re.registry=e[0]?.toString()||void 0,await async function(){const e=O("package.json","file");if(n(e))return void U.warn("未找到当前包的 package.json 文件的位置");const t=P(q(e,"package.json"));if(s(t))return void U.error("未找到当前包的 package.json 文件,该事件发生的概率极低");const o=t.name||"",r=await T(o,re.registry,9800);re.local=t,re.online=r.data}();const{local:t,online:o}=re,r=t?.version||"",i=o?.version||"";if(s(t))return U("未找到包 package.json 文件退出"),k(se("未找到当前包 package.json"));const a=se(`当前包本地版本为: ${r}`);if(o){k(a);const e=" ".repeat(6);k(ae(`${e}线上版本为:${i}@latest`));const t=new Date(o&&o.time.modified||"").toLocaleString();W.INFO(`最后的发布时间为: ${t.toString()}`),L()}await fe()}async function $e(e){e["--upDependencies"]?await te(!0):e["--diff"]?await we(e["--diff"]):W.help("package")}async function ye(e){const t=O("package.json","file",process.cwd());if(null==t)return!0;const n=P(q(t,"package.json"));if(s(n)||!n.version)return!0;const o=n.version;await S({code:`git tag -a v${o} -m '${e}'`,cwd:t}),await S({code:"git push origin --tag",cwd:t})}async function he(e){e.commit?await async function(e="版本维护",t=!1){const n=O(".git","directory");if(null==n)return k(g("#ff0")("not a git repository(当前目录非 git 储存库)"));const o=await S({code:"git add .",cwd:n});if(!o.success)return null!=console.log(o.error);const r=await S({code:"git status",cwd:n});if(!/nothing to commit, working tree clean/.test(r.data||"")){const o=await S({code:"git add .",cwd:n});if(!o.success)return null!=console.log(o.error);""==e.trim()&&(e=await W.question({text:"请输入提交信息"})??"");const r=await S({code:`git commit -m "${e}"`,cwd:n});return r.success?(r.error&&console.log(r.error),await S({code:"git push",cwd:n}),t&&await ye(e),!0):null!=console.log(r.error)}return!1}(e.commit.join(" ")):e.merge?await async function(e){if(a(e)){const t=await S("git branch -a");console.log(t.data),e=await W.question({text:"请输入要合并分支的名称"})??""}if(a(e))return W.ERROR("没有获取到要合并的分支");const t=await W.selection({data:[{label:"正常快进合并",value:""},{label:"非快进合并 (--no-ff)",value:"--no-ff"},{label:"多提交记录合并为一条 (--squash)",value:"--squash",tip:"适用于将众多小的提交合并成一个提交"}]});n(t)?W.ERROR("您选择退出合并"):await S(`git merge ${e} ${t}`)}(e.merge.join("")):e.tag?await async function(){const e=await W.question({text:"请输入待标记的信息"});n(e)||await ye(e)}():W.help("git")}const ve=["A","AAAA","ANY","CAA","CNAME","MX","NAPTR","NS","PTR","SOA","SRV","TLSA","TXT"],be={domain:"github.com",port:443,notCovered:!1,dnsServers:["1.1.1.1","8.8.8.8","8.8.4.4","9.9.9.9","208.67.222.222","114.114.115.115","223.5.5.5","180.76.76.76"],ips:{},rrtype:["A"],reset(e){this.ips={},this.domain="github.com",this.port=443,this.notCovered=!1,this.rrtype=[],n(e)||e.forEach(e=>{const t=e["--domain"];!n(t)&&t.length>0&&(this.domain=t[0].toString());const o=e["--port"];!n(o)&&o.length>0&&(this.port=Number(o[0]));const r=e["--not-covered"];n(r)||!1===r[0]||(this.notCovered=!0);const i=e["--rrtype"];if(!n(i)&&i.length>0){const e=Y(ve,i);e.length>0&&(this.rrtype=e)}})}};async function je(e,t=3e3){const{ips:n,port:o}=be;if(void 0===n[e])return U(`校验 ${e} 的联通性`),new Promise(n=>{const r=new J.Socket;r.setTimeout(t),r.on("connect",()=>{r.destroy(),n(!0)}),r.on("timeout",()=>{r.destroy(),n(!1)}),r.on("error",()=>{r.destroy(),n(!1)}),U("测试的接口为:",o),r.connect(o,e)});U(`${e} 联通性已校验过 ${n[e]?"🔥":"⛓️‍💥"}`)}async function ke(e="1.1.1.1"){const t=new H;t.setServers([e]);const{domain:o,ips:r}=be;try{const i=await new Promise(n=>{t.resolve4(o,(t,r)=>{if(t)return U.error(`使用 ${e} 获取 ${o} 的 ip 出错`,t),void n(void 0);U(`使用 ${e} 获取 ${o} 的 ip 值为 ${r.join(" --- ")}`),n(r)})});if(n(i))return{dnsServer:e,error:"获取 ip 错误"};const s=await Promise.all(i.map(async e=>{const t=await je(e);return n(t)||(r[e]=t),{ip:e.padEnd(16," "),isAlive:n(t)?r[e]:t}}));return{dnsServer:e,results:s}}catch(t){return U.error("获取 ip 错误",t),{dnsServer:e,error:t}}}async function De(e){be.reset(e.options),await async function(){const{domain:e,ips:t}=be;try{const o=await new Promise(t=>{B.lookup(e,{all:!0,family:4},(n,o)=>{if(n)return U.error(`获取本地 ${e} 的 ip 出错`,n),void t(void 0);U(`获取本地 ${e} 的 ip 值为 ${o.join(" --- ")}`),t(o.map(e=>e.address))})});if(n(o))return{error:"获取 ip 错误"};const r=await Promise.all(o.map(async e=>{const o=await je(e);return n(o)||(t[e]=o),{ip:e.padEnd(16," "),isAlive:n(o)?t[e]:o}}));return r.length>0&&(k(se`本地配置 ${e} 的 ip 地址及联通性为:\n`),r.forEach(e=>{k(v`- ${e.ip}`,!1),k(e.isAlive?se` -> ✅`:ce` -> ❌`)}),k()),{results:r}}catch(e){return U.error(e),{error:e}}}(),await async function(){const{domain:e,dnsServers:t,notCovered:o}=be;let r=0;k(`${e} ip 列表:`);for(const e of t){const{dnsServer:t,results:i}=await ke(e);n(i)?U.warn(`${t} 获取 ip 失败`):(o||0===r||(R(r),E(!0)),k(),k($.reversed`DNS: ${t}`),k(),i.forEach(({ip:e,isAlive:t})=>{const n=t?`${f(e)} ${se`->`} ✅`:`${ce(e)} ${u`⛓️‍💥`} ❌`;k(y(n))}),r=3+i.length)}o||(R(r+1),E(!0))}();const{ips:t}=be;Object.keys(t).length>0?await async function(){const{domain:e,ips:t,port:n}=be;await M(`${$.brightMagenta`${e}`} 域名解析结果:`),k();for(const e in t)if(Object.prototype.hasOwnProperty.call(t,e)){const n=t[e];k(n?`${f`- ${e.padEnd(16)}`} ${se`->`} ✅`:`${ce`- ${e.padEnd(16)}`} ${u`⛓️‍💥`} ❌`)}k(),await M(u.italic.dim`${e} 联通性接口判断为 ${n.toString()}`),k()}():await async function(){const{domain:e,port:t}=be;await M(`\n找不到 ${u(e)} 的服务器 IP 地址 ${ce`测试使用 ${t.toString()} 端口`}\n`)}()}const Se=W.args.$arrMap;W.args.$isVoid?(U("没有匹配到子命令,打印帮助信息并退出"),W.help(),W.end()):l(W.args.$only)&&(console.log(`${W.args.$nomatch.map(e=>m(e)).join("、")} 不是有效的参数`),W.end());try{await async function e(){if(0===Se.length)return;const t=Se.shift();if(n(t))return await e();t.remove?(U("执行文件移除"),await async function(e){const t=e.value||[],n=e.options;for(n?.forEach(e=>{const n=e["--ignore"];n&&(_.log=!0,t.concat(n));const o=e["--subdirectories"];o&&(_.subdirectories=!0,t.concat(o))}),_.log||k($.brightCyan`当前系统为: ${D?"windows":"linux/mac"}`),0==t.length&&(U.warn("待删除列表",t),k($.random`没有待清理的文件/文件夹`));t.length;){let e=t.pop();e=void 0===e?"undefined":e.toString(),await Z(e)}}(t.remove)):t.clearScreen||t.clearTerminal?(U("执行清屏"),await async function(){x()?(U("执行第一遍清理"),await S({code:D?"cls":"clear",printLog:!1}),U("执行第二遍清理"),N("3J"),N("c",!1),R(1/0),E(!0)):(U.warn("当前系统不支持 TTY"),k("当前环境不支持 "))}()):t.git?(U("执行 git 相关命令"),await async function(e){if(c(e)&&(e={value:[]}),n(e.options)||0===e.options.length)return U.warn("没有配置项,直接返回的帮助文档"),W.help("git");const t=e.options;for(let e=0,n=t.length;e<n;e++)U("本次执行",t[e]),await he(t[e])}(t.git)):t.package?(U("执行 package 相关命令"),await async function(e){if(n(e.options)||0==e.options.length)return U.warn("没有参数输入"),W.help("package");const t=e.options;for(let e=0,n=t.length;e<n;e++)await $e(t[e])}(t.package)):t.update?(U("执行 update 相关命令"),await async function(e){if(n(e.options)||0==e.options.length)return U.warn("没有匹配到选项,自动抛出帮助文档"),W.help("update");const t=e.options;ne.log=!t.some(e=>e["--ignore"]);for(let e=0,n=t.length;e<n;e++)await oe(t[e])}(t.update)):t.dns?(U("执行 dns 相关的命令"),await De(t.dns)):t.runOtherCode&&U("执行运行其他命令");try{await e()}catch(e){U.error("执行 run 报错",e)}}(),L(" 终结分割线 ",!0)}catch(e){U.error("执行 run 报错",e)}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "jja",
3
3
  "type": "module",
4
- "version": "2.3.13",
4
+ "version": "2.3.15",
5
5
  "description": "一些在终端的执行动作 🥜",
6
6
  "license": "MIT",
7
7
  "dependencies": {
@@ -13,7 +13,7 @@
13
13
  "a-node-tools": "^4.2.16",
14
14
  "a-type-of-js": "^1.0.5",
15
15
  "color-pen": "^2.0.12",
16
- "colored-table": "^0.0.2"
16
+ "colored-table": "^0.1.0"
17
17
  },
18
18
  "author": {
19
19
  "name": "🥜",