ttmg-pack 0.0.18-beta.3 → 0.0.18-beta.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -1,10 +1,10 @@
1
1
  /**
2
2
  * ==========================================
3
3
  * @Description: ttmg pack
4
- * @Version: 0.0.18-beta.3
4
+ * @Version: 0.0.18-beta.5
5
5
  * @Author: zhanghongyang.mocha
6
- * @Date: 2025-08-27 21:41:32
6
+ * @Date: 2025-08-27 21:58:09
7
7
  * ==========================================
8
8
  */
9
- "use strict";var e=require("fs"),t=require("winston"),n=require("path");require("crypto"),require("zlib");var s=require("acorn"),o=require("glob"),i=require("magic-string");function c(e){var t=Object.create(null);return e&&Object.keys(e).forEach(function(n){if("default"!==n){var s=Object.getOwnPropertyDescriptor(e,n);Object.defineProperty(t,n,s.get?s:{enumerable:!0,get:function(){return e[n]}})}}),t.default=e,Object.freeze(t)}var r=c(e),a=c(n);function u(t){e.existsSync(t)||e.mkdirSync(t,{recursive:!0})}const f=new class{constructor(){this.logger=null}init(e,s=!0){this.enableLog=s,this.logger=t.createLogger({level:"info",format:t.format.combine(t.format.timestamp(),t.format.json()),transports:[new t.transports.File({filename:n.join(e,"pack.log")}),new t.transports.Console]})}info(e){this.enableLog&&this.logger.info(e)}error(e){this.enableLog&&this.logger.error(e)}warn(e){this.enableLog&&this.logger.warn(e)}debug(e){this.enableLog&&this.logger.debug(e)}log(e){this.logger.info(e)}setOutput(e){this.output=e}};function l(t){if(!e.existsSync(t))return!1;if(!e.statSync(t).isDirectory())return!1;let s=!0;const o=e.readdirSync(t);for(const i of o){const o=n.join(t,i);if(e.statSync(o).isDirectory()){l(o)||(s=!1)}else s=!1}return!!s&&(e.rmSync(t,{recursive:!0}),!0)}const m="packageConfig.json",d="project.config.json",g="game.json",p="__GAME__",y=".txt",k="_code",h="_asset",j=62914560,S=["ttmg_odr_tag_1","ttmg_odr_tag_2","ttmg_odr_tag_3"],b=31457280,_=4194304;function $(t,s=[]){let o=0;if(!e.existsSync(t))return 0;const i=e.readdirSync(t,{withFileTypes:!0});for(const c of i){const i=n.join(t,c.name);c.isDirectory()?s.includes(c.name)||(o+=$(i,s)):c.isFile()&&(o+=e.statSync(i).size)}return o}function w(t){const s=e.readdirSync(t,{withFileTypes:!0});for(const o of s){const s=n.join(t,o.name);if(o.isDirectory())w(s);else if(o.isFile()&&(s.endsWith(".js")||s.endsWith(".js.map")))try{e.unlinkSync(s)}catch(e){}}}function x(t){const s=e.readdirSync(t,{withFileTypes:!0});for(const o of s){const s=n.join(t,o.name);if(o.isDirectory())x(s);else if(o.isFile()&&!s.endsWith(".js")&&!s.endsWith(".js.map"))try{e.unlinkSync(s)}catch(e){}}}const O=["node_modules","__MACOSX"],v=[".DS_Store","Thumbs.db"];const F={__GAME__:{code_url:"https://connect.tiktok-minis.com/test/__GAME___code.txt",asset_url:"https://connect.tiktok-minis.com/test/__GAME___asset.txt",asset_md5:"8e3cc2ebd5536e9183b93fda34c1e7f4",root:"",main:"game.js"},configBundle:{root:"subpackages/configBundle",main:"subpackages/configBundle/game.js",asset_md5:"f624a1b37fb35d4dfb1975105a51ebd4",code_url:"https://connect.tiktok-minis.com/test/configBundle_code.txt",asset_url:"https://connect.tiktok-minis.com/test/configBundle_asset.txt"},gameBundle:{main:"subpackages/gameBundle/game.js",root:"subpackages/gameBundle",asset_md5:"fcc9f686bbc8d6f983263e7132598b26",code_url:"https://connect.tiktok-minis.com/test/gameBundle_code.txt",asset_url:"https://connect.tiktok-minis.com/test/gameBundle_asset.txt"},mainBundle:{main:"subpackages/mainBundle/game.js",root:"subpackages/mainBundle",asset_md5:"36b9674d48f9fbb0de6e8f48b194aebc",code_url:"https://connect.tiktok-minis.com/test/mainBundle_code.txt",asset_url:"https://connect.tiktok-minis.com/test/mainBundle_asset.txt"},resources:{root:"subpackages/resources",main:"subpackages/resources/game.js",asset_md5:"1c4608d7690750f7bd5c00663013fc70",code_url:"https://connect.tiktok-minis.com/test/resources_code.txt",asset_url:"https://connect.tiktok-minis.com/test/resources_asset.txt"},iceAndFire:{root:"subpackages/iceAndFire",main:"subpackages/iceAndFire/game.js",asset_md5:"c7fd8e8506dd59725269027a52e2cae4",code_url:"https://connect.tiktok-minis.com/test/iceAndFire_code.txt",asset_url:"https://connect.tiktok-minis.com/test/iceAndFire_asset.txt"},pushstone:{root:"subpackages/pushstone",main:"subpackages/pushstone/game.js",asset_md5:"f5061dba94ad53e1f578325a7cb26a01",code_url:"https://connect.tiktok-minis.com/test/pushstone_code.txt",asset_url:"https://connect.tiktok-minis.com/test/pushstone_asset.txt"},main:{root:"subpackages/main",main:"subpackages/main/game.js",asset_md5:"aece2f69cc168d7cefb248e8574c3371",code_url:"https://connect.tiktok-minis.com/test/main_code.txt",asset_url:"https://connect.tiktok-minis.com/test/main_asset.txt"}},M=[{id:""+1e17*Math.random(),packages:F},{id:""+1e17*Math.random(),packages:F}];function D(t,s){f.info("start check project"),function(e,t){f.info("start check project size");const{projectSizeLimit:n,isDev:s}=t,o=$(e,[]),i=Math.round(o/1048576),c=Math.round(n/1048576);if(o>n){const e=`Game size ${i}MB, must not exceed ${c}MB`;if(f.error(e),!s)throw new Error(e);console.log("%s",e)}else f.info(`Game size ${i}MB, check successfully!`),f.info("Game size check successfully!")}(t,s),function(t){f.info("start check config");const s=n.join(t,g);if(!e.existsSync(s)){const e="can not find game.json in game source code, please check it";throw f.error(e),new Error(e)}f.info("check config successfully")}(t),function(t){f.info("start check project config");const s=n.join(t,d);if(!e.existsSync(s)){const e="can not find project.config.json in game source code, please check it";throw f.error(e),new Error(e)}f.info("check project.config.json successfully");let o;try{o=JSON.parse(e.readFileSync(s,"utf-8"))}catch(e){throw f.error("check project.config.json failed, your project.config.json is invalid json format"),e}if(!(null==o?void 0:o.appid)){const e="can not find appid in project.config.json, you should add appid(client_key) in project.config.json";throw f.error(e),new Error(e)}}(t),f.info("check project successfully")}function E(e,t){f.info("start check main package"),f.info("check game.js");const n=a.join(e,"game.js");if(!r.existsSync(n)){const e="game.js must exist in main package!";throw f.error(e),new Error(e)}f.info("check game.js successfully"),function({entryDir:e,config:t}){f.info("start check main package size");const{mainPackageSizeLimit:n,isDev:s}=t,o=$(e,["subpackages"]),i=Math.round(o/1048576),c=Math.round(n/1048576);if(o>n){const e=`Main package size ${i}MB, must not exceed ${c}MB`;if(!s)throw f.error(e),new Error(e);console.log("%s",e)}else f.info(`Main package size ${i}MB, check successfully!`)}({entryDir:e,config:t}),f.info("check main package successfully")}function P(t,s){const{subpackageSizeLimit:o}=s||{};f.info("start check subpackages in game.json");const i=n.join(t,g),c=JSON.parse(e.readFileSync(i,"utf-8")).subpackages||[];(null==c?void 0:c.length)?c.forEach(s=>{if(!s.name||!s.root){const e="The subpackages configuration in game.json is invalid: the root or name field in one or more subpackages is empty. Please ensure that all subpackages have non-empty root and name values.";throw f.error(e),new Error(e)}const i=n.join(t,s.root);if(!e.existsSync(i)){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${s.name} entry dir, please check it`;throw f.error(e),new Error(e)}const c=s.root;if(!e.existsSync(n.join(t,c))){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${s.name} root, please check it`;throw f.error(e),new Error(e)}o&&function({entryDir:e,subPkgName:t,limit:s}){const o=n.join(e,"subpackages",t),i=$(o,[]),c=Math.round(i/1048576);if(c>s){const e=`${t} Sub package size ${c}MB, must not exceed ${s/1048576}MB`;throw f.error(e),new Error(e)}f.info(`${t} Sub package size ${c}MB, check successfully!`)}({entryDir:t,subPkgName:s.name,limit:o})}):f.info("subpackages is empty"),f.info("check subpackages in game.json successfully")}async function B({entryDir:e,config:t}){f.info("开始校验项目"),D(e,t),E(e,t),P(e,t),f.info("校验项目完成")}async function z(t,s){f.info("开始基于源代码中的 game.json 生成 packageConfig.json");const o=n.join(t,g),i=JSON.parse(e.readFileSync(o,"utf-8")),c={};c[p]={url:"",md5:"",root:"",main:"game.js",output:`${p}${y}`},function(e){if(!e||0===e.length)return[];function t(e){return String(e).replace(/^\/+|\/+$/g,"")}function s(e){return o(e)?n.join(e,"game.js"):e}function o(e){return e.endsWith("/")||!/\.[a-zA-Z0-9]+$/.test(e)}function i(e){return e.replace(/^subpackages\/([^/]+).*$/,"subpackages/$1")}return e.map(e=>{const n=t(e.name);return{main:s(e.root),root:i(e.root),name:n}})}(i.subpackages).forEach(({name:e,root:t,main:n})=>{c[e]={url:"",md5:"",root:t,main:n,output:`${e}${y}`}});const r={packages:c};return Object.keys(c).forEach(s=>{const o=n.join(t,c[s].main);if(!e.existsSync(o)){const e=`开发者源代码包中的子包 ${s} 中的 main 文件 ${c[s].main} 不存在`;throw f.error(e),new Error(e)}}),e.existsSync(s)&&e.rmSync(s,{recursive:!0}),e.mkdirSync(s),e.writeFileSync(n.join(s,m),JSON.stringify(r,null,2)),f.info("基于源代码中的 game.json 生成 packageConfig.json 成功"),r}function L(e){const t=[];return Object.values(e).forEach(e=>{Object.values(e).forEach(e=>{t.push(...e)})}),t}function A(t,s=[".js",".ts",".jsx",".tsx"],o=[]){let i=[];if(!e.existsSync(t))return i;const c=e.readdirSync(t,{withFileTypes:!0});for(const e of c){const c=n.join(t,e.name);if(e.isDirectory()){if(o.includes(e.name))continue;i=i.concat(A(c,s,[]))}else s.some(t=>e.name.endsWith(t))&&i.push(c)}return i}function T(t,i){const c=function(t){return e.readdirSync(t).filter(s=>{const o=n.join(t,s);return e.statSync(o).isDirectory()&&!["node_modules","dist","build","output"].includes(s)}).map(e=>e+"/")}(t),r=o.sync("**/*.js",{cwd:t,ignore:[]}),a=[];r.forEach(o=>{const i=n.join(t,o),r=e.readFileSync(i,"utf-8");let u;try{u=s.parse(r,{ecmaVersion:"latest",sourceType:"module"})}catch(e){return}N(u,s=>{if("CallExpression"===s.type&&"Identifier"===s.callee.type&&("require"===s.callee.name||"requireAsync"===s.callee.name)&&s.arguments.length>0&&"Literal"===s.arguments[0].type){const o=s.arguments[0].value;let r=null;if(o.startsWith("."))r=n.resolve(n.dirname(i),o);else if(o.startsWith("/"))r=n.resolve(t,"."+o);else for(const e of c)if(o.startsWith(e)){r=n.resolve(t,o);break}r&&!function(t){if(e.existsSync(t)&&e.statSync(t).isFile())return!0;const n=[".js",".ts",".jsx",".tsx"];for(const s of n)if(e.existsSync(t+s)&&e.statSync(t+s).isFile())return!0;return!1}(r)&&(r=null),a.push({file:n.relative(t,i),callee:s.callee.name,requirePath:o,resolvedPath:r?n.relative(t,r):""})}})});const u=a.map(e=>{let t="";Object.keys(i).forEach(n=>{e.resolvedPath&&e.resolvedPath.startsWith(i[n].root)&&(t=n)});let n="";return Object.keys(i).forEach(t=>{e.file.startsWith(i[t].root)&&(n=t)}),Object.assign(Object.assign({},e),{curModule:n,depModule:t})}),f=[".js",".ts",".jsx",".tsx"];return u.reduce((e,t)=>{return e[t.curModule]||(e[t.curModule]={}),t.curModule===t.depModule||t.resolvedPath&&(n=t.resolvedPath,f.some(e=>n.endsWith(e)))&&(e[t.curModule]=Object.assign(Object.assign({},e[t.curModule]),{[t.resolvedPath]:t.depModule})),e;var n},{})}function N(e,t){t(e);for(let n in e)if(e.hasOwnProperty(n)){const s=e[n];Array.isArray(s)?s.forEach(e=>e&&"string"==typeof e.type&&N(e,t)):s&&"string"==typeof s.type&&N(s,t)}}function C({gameEntry:t,gameOutput:s,pkgName:o,allDeps:c,allMaps:r,pkgConfig:a,destRoot:l,enableSourcemap:m}){let d=Date.now();f.info(`pack start,startTime:${d}`);const g=r[o],p={main:a.main,deps:c[o]||{},map:r[o]};u(l),e.writeFileSync(n.join(l,"moduleConfig.json"),JSON.stringify(p));for(const s in g){const c=g[s];if(!Array.isArray(c)||0===c.length)continue;const r=new i.Bundle({separator:"\n"});for(const o of c){const c=n.join(t,o);if(e.existsSync(c)&&(n.basename(c)!==s&&/\.js$/i.test(o))){const t=e.readFileSync(c,"utf-8"),n=o.replace(/\\/g,"/"),s=`define("game:${n}", ["require", "requireAsync", "module", "exports", "sandboxGlobal"], function(require, requireAsync, module, exports, sandboxGlobal){\nwith(sandboxGlobal){\n`,a="\n}\n});\n",u=new i(t,{filename:n});u.prepend(s),u.append(a),r.addSource(u)}}const a=n.join(l,n.basename(s)),u=r.toString();if(m){const t=r.generateMap({hires:!0,includeContent:!0,file:s}),n=u+`\n//# sourceMappingURL=${`data:application/json;charset=utf-8;base64,${Buffer.from(t.toString(),"utf-8").toString("base64")}`}`+`\n//# sourceURL=${s}`;e.writeFileSync(a,n,"utf-8")}else{const t=u+`\n//# sourceURL=${s}`;e.writeFileSync(a,t,"utf-8")}f.info(`pack end,packName:${o}`)}}function G({pkgRoot:t,destRoot:s,packedFiles:o,gameEntry:i,gameOutput:c,skipDirs:r=[],isRoot:a=!0}){if(!e.existsSync(t))return;const f=e.readdirSync(t,{withFileTypes:!0});for(const l of f){const f=n.join(t,l.name),m=n.join(s,l.name);a&&l.isDirectory()&&r.includes(l.name)||(l.isDirectory()?G({pkgRoot:f,destRoot:m,gameOutput:c,packedFiles:o,gameEntry:i,skipDirs:r,isRoot:!1}):(u(n.dirname(m)),o.includes(n.relative(i,f))||e.copyFileSync(f,m)))}}async function R({gameEntry:t,gameOutput:s,enableSourcemap:o}){let i=Date.now();!function(t){!function t(s){let o;try{o=e.readdirSync(s)}catch(e){return void f.warn(`无法访问目录: ${s}, ${e.message}`)}o.forEach(o=>{const i=n.join(s,o);try{const n=e.statSync(i);n.isDirectory()?O.includes(o)?(e.rmSync(i,{recursive:!0,force:!0}),f.info(`已删除无用目录: ${i}`)):t(i):n.isFile()&&v.includes(o)&&(e.rmSync(i,{force:!0}),f.info(`已删除无用文件: ${i}`))}catch(e){f.warn(`无法处理: ${i}, ${e.message}`)}})}(t)}(t),f.info(`pack start,startTime:${i}`);const{packages:c}=await z(t,s),r=T(t,c),a=function(e,t){f.info("开始基于游戏源代码收集分包中的 JS 文件");const s={},o=Object.values(t).filter(e=>e.root&&""!==e.root).map(e=>e.root.split(n.sep)[0]).filter(Boolean),i=Array.from(new Set(o));for(const o in t){const c=t[o];let r,a;"__GAME__"===o?(a=A(e,[".js",".ts",".jsx",".tsx"],i),r=e):(r=n.join(e,c.root),a=A(r));const u=[];a.forEach(t=>{u.push(n.relative(e,t).replace(/\\/g,"/"))});const f=t[o].root;s[o]={[f?`${f}/game.pack.js`:"game.pack.js"]:u}}return f.info("基于游戏源代码收集分包中的 JS 文件完成"),s}(t,c);for(const e in a){f.info(`开始基于游戏源代码打包,包名:${e}`);let i=Date.now();const u=c[e],m=n.join(t,u.root),d=n.join(s,e,u.root);C({gameEntry:t,gameOutput:s,allDeps:r,allMaps:a,pkgName:e,pkgConfig:u,destRoot:d,enableSourcemap:o}),G({gameEntry:t,gameOutput:s,pkgRoot:m,destRoot:d,packedFiles:L(a),skipDirs:["subpackages"]});l(n.join(s,e)),f.info(`基于游戏源代码打包分包完成,包名:${e}`),f.info(`基于游戏源代码打包分包耗时,包名:${e},耗时:${Date.now()-i}ms`)}return f.info(`pack end,duration:${Date.now()-i}ms`),{}}exports.buildPkgs=async function(t){const{entry:s,output:o,rules:i}=t;let c=Date.now();f.init(o,!0),f.info("TTMG_PACK_VERSION: 0.0.18-beta.3"),f.info(`pack start, startTime:${c}`),await B({entryDir:s,config:{projectSizeLimit:(null==i?void 0:i.projectSizeLimit)||b,mainPackageSizeLimit:(null==i?void 0:i.mainPackageSizeLimit)||_,subpackageSizeLimit:null==i?void 0:i.subpackageSizeLimit}}),await R({gameEntry:s,gameOutput:o,enableSourcemap:!1}),await async function(t){let s=Date.now();f.info(`makeOdrPkgs start, ${s}`);const o=n.join(t,m),i=JSON.parse(e.readFileSync(o,"utf-8")),c=i.packages;for(const s in c){const o=n.join(t,s),i=n.join(t,`${s}${k}`),r=n.join(t,`${s}${h}`);u(i),u(r),e.cpSync(o,i,{recursive:!0}),e.cpSync(o,r,{recursive:!0}),x(i),w(r),c[`${s}${k}`]=Object.assign(Object.assign({},c[s]),{code_url:"",code_md5:"",output:`${s}${k}${y}`}),c[`${s}${h}`]=Object.assign(Object.assign({},c[s]),{asset_url:"",asset_md5:"",output:`${s}${h}${y}`})}e.writeFileSync(o,JSON.stringify(i,null,2)),f.info(`makeOdrPkgs end, ${Date.now()-s}ms`)}(o),f.info(`pack end:${Date.now()-c}ms`)},exports.debugPkgs=async function(t){const{entry:s,output:o,rules:i,enableLog:c=!1}=t;f.init(o,c);try{await B({entryDir:s,config:{projectSizeLimit:(null==i?void 0:i.projectSizeLimit)||b,mainPackageSizeLimit:(null==i?void 0:i.mainPackageSizeLimit)||_,isDev:!0}}),await R({gameEntry:s,gameOutput:o,enableSourcemap:!0}),await function(t){f.info("mergePkgs start");const s=n.join(t,m);let o;try{o=JSON.parse(e.readFileSync(s,"utf-8"))}catch(e){return void f.error(`读取配置文件失败: ${s}, err: ${e.message}`)}const i=o.packages;if(i&&"object"==typeof i){for(const s of Object.keys(i)){const o=n.join(t,s);if(!e.existsSync(o)){f.warn(`package 目录不存在: ${o}`);continue}const i=e.readdirSync(o);for(const s of i){const i=n.join(o,s),c=n.join(t,s);try{const t=e.statSync(i);if(t.isFile())e.copyFileSync(i,c);else if(t.isDirectory())e.cpSync(i,c,{recursive:!0});else if(t.isSymbolicLink()){const n=e.readlinkSync(i);e.existsSync(c)&&e.rmSync(c,{force:!0}),e.symlinkSync(n,c,t.isDirectory()?"dir":"file")}f.info(`合并文件/目录: ${i} -> ${c}`)}catch(e){f.error(`处理文件失败: ${i}, err: ${e.message}`)}}try{e.rmSync(o,{recursive:!0,force:!0}),f.info(`已删除 package 目录: ${o}`)}catch(e){f.error(`删除 package 目录失败: ${o}, err: ${e.message}`)}}f.info("mergePkgs 完成")}else f.error("配置文件中 packages 字段无效, 请检查配置")}(o);const t=e.readFileSync(n.join(o,m),"utf-8");return{isSuccess:!0,packages:JSON.parse(t).packages}}catch(e){return{isSuccess:!1,errorMsg:e.message}}},exports.downloadOdrPkgs=async function(){const t=n.join(__dirname,"temp");e.existsSync(t)||e.mkdirSync(t);const s=await async function(){return M}();f.info("odr_games"),function(e,t){for(const n of e){const{id:e,packages:s}=n,o=a.join(t,e);r.existsSync(o)||r.mkdirSync(o);for(const e in s){const t=s[e],{code_url:n}=t,i=require("child_process").execSync(`curl -s "${n}"`).toString();f.info(`downloadAndSaveCodesSync code, ${n}`);const c=`${e}.txt`,u=a.join(o,c);r.writeFileSync(u,i)}}}(s,t);const o={};for(const e of s){const{id:s,packages:i}=e,c=n.join(t,s);o[s]=$(c)}const i={};S.forEach(e=>{i[e]=[]});const c={};for(const r of s){const{id:s,packages:a}=r,u=o[s];if(u>j)continue;let l=null;for(const c of S){const r=n.join(t,c);e.existsSync(r)||e.mkdirSync(r);if(i[c].reduce((e,t)=>e+o[t],0)+u<=j){const o=n.join(t,s),a=n.join(r,s);e.cpSync(o,a,{recursive:!0}),i[c].push(s),l=c;break}}if(l){c[s]={tag:l,packages:{}};for(const e in a){const t=a[e];f.info(`TTMGODRConfig[gameId].packages, ${t}, ${e}`),c[s].packages[e]={root:t.root,main:t.main,asset_md5:t.asset_md5,asset_url:t.asset_url}}}}const u=n.join(t,"TTMG_ODR_CONFIG.json");e.writeFileSync(u,JSON.stringify(c,null,2),"utf-8"),f.info(`TTMG_ODR_CONFIG.json 已生成: ${u}`)},exports.writeOdrConfig=async function(e){f.info("writeOdrPackagesConfig start");const t=a.join(e,m),n=JSON.parse(r.readFileSync(t,"utf-8")),s=n.packages,o=n.odr_packages||{};for(const e in s){if(e.endsWith(k)){const t=s[e],n=e.split(k)[0];o[n]||(o[n]={}),o[n]=Object.assign(Object.assign({},o[n]),{root:t.root,code_output:t.output,code_md5:t.md5,main:t.main}),delete s[e]}if(e.endsWith(h)){const t=e.split(h)[0],n=s[e];o[t]=Object.assign(Object.assign({},o[t]),{root:n.root,asset_output:n.output,asset_md5:n.md5}),delete s[e]}}n.odr_packages=o,r.writeFileSync(t,JSON.stringify(n,null,2)),f.info("writeOdrPackagesConfig end")};
9
+ "use strict";var e=require("fs"),t=require("winston"),n=require("path");require("crypto"),require("zlib");var o=require("acorn"),s=require("glob"),i=require("magic-string");function c(e){var t=Object.create(null);return e&&Object.keys(e).forEach(function(n){if("default"!==n){var o=Object.getOwnPropertyDescriptor(e,n);Object.defineProperty(t,n,o.get?o:{enumerable:!0,get:function(){return e[n]}})}}),t.default=e,Object.freeze(t)}var r=c(e),a=c(n);function u(t){e.existsSync(t)||e.mkdirSync(t,{recursive:!0})}const f=new class{constructor(){this.logger=null}init(e,o=!0){this.enableLog=o,this.logger=t.createLogger({level:"info",format:t.format.combine(t.format.timestamp(),t.format.json()),transports:[new t.transports.File({filename:n.join(e,"pack.log")}),new t.transports.Console]})}info(e){this.enableLog&&this.logger.info(e)}error(e){this.enableLog&&this.logger.error(e)}warn(e){this.enableLog&&this.logger.warn(e)}debug(e){this.enableLog&&this.logger.debug(e)}log(e){this.logger.info(e)}setOutput(e){this.output=e}};function l(t){if(!e.existsSync(t))return!1;if(!e.statSync(t).isDirectory())return!1;let o=!0;const s=e.readdirSync(t);for(const i of s){const s=n.join(t,i);if(e.statSync(s).isDirectory()){l(s)||(o=!1)}else o=!1}return!!o&&(e.rmSync(t,{recursive:!0}),!0)}const m="packageConfig.json",d="project.config.json",g="game.json",p="__GAME__",y=".txt",k="_code",h="_asset",j=62914560,S=["ttmg_odr_tag_1","ttmg_odr_tag_2","ttmg_odr_tag_3"],b=31457280,_=4194304;function $(t,o=[]){let s=0;if(!e.existsSync(t))return 0;const i=e.readdirSync(t,{withFileTypes:!0});for(const c of i){const i=n.join(t,c.name);c.isDirectory()?o.includes(c.name)||(s+=$(i,o)):c.isFile()&&(s+=e.statSync(i).size)}return s}function w(t){const o=e.readdirSync(t,{withFileTypes:!0});for(const s of o){const o=n.join(t,s.name);if(s.isDirectory())w(o);else if(s.isFile()&&(o.endsWith(".js")||o.endsWith(".js.map")))try{e.unlinkSync(o)}catch(e){}}}function x(t){const o=e.readdirSync(t,{withFileTypes:!0});for(const s of o){const o=n.join(t,s.name);if(s.isDirectory())x(o);else if(s.isFile()&&!o.endsWith(".js")&&!o.endsWith(".js.map"))try{e.unlinkSync(o)}catch(e){}}}const O=["node_modules","__MACOSX"],v=[".DS_Store","Thumbs.db"];const F={__GAME__:{code_url:"https://connect.tiktok-minis.com/test/__GAME___code.txt",asset_url:"https://connect.tiktok-minis.com/test/__GAME___asset.txt",asset_md5:"8e3cc2ebd5536e9183b93fda34c1e7f4",root:"",main:"game.js"},configBundle:{root:"subpackages/configBundle",main:"subpackages/configBundle/game.js",asset_md5:"f624a1b37fb35d4dfb1975105a51ebd4",code_url:"https://connect.tiktok-minis.com/test/configBundle_code.txt",asset_url:"https://connect.tiktok-minis.com/test/configBundle_asset.txt"},gameBundle:{main:"subpackages/gameBundle/game.js",root:"subpackages/gameBundle",asset_md5:"fcc9f686bbc8d6f983263e7132598b26",code_url:"https://connect.tiktok-minis.com/test/gameBundle_code.txt",asset_url:"https://connect.tiktok-minis.com/test/gameBundle_asset.txt"},mainBundle:{main:"subpackages/mainBundle/game.js",root:"subpackages/mainBundle",asset_md5:"36b9674d48f9fbb0de6e8f48b194aebc",code_url:"https://connect.tiktok-minis.com/test/mainBundle_code.txt",asset_url:"https://connect.tiktok-minis.com/test/mainBundle_asset.txt"},resources:{root:"subpackages/resources",main:"subpackages/resources/game.js",asset_md5:"1c4608d7690750f7bd5c00663013fc70",code_url:"https://connect.tiktok-minis.com/test/resources_code.txt",asset_url:"https://connect.tiktok-minis.com/test/resources_asset.txt"},iceAndFire:{root:"subpackages/iceAndFire",main:"subpackages/iceAndFire/game.js",asset_md5:"c7fd8e8506dd59725269027a52e2cae4",code_url:"https://connect.tiktok-minis.com/test/iceAndFire_code.txt",asset_url:"https://connect.tiktok-minis.com/test/iceAndFire_asset.txt"},pushstone:{root:"subpackages/pushstone",main:"subpackages/pushstone/game.js",asset_md5:"f5061dba94ad53e1f578325a7cb26a01",code_url:"https://connect.tiktok-minis.com/test/pushstone_code.txt",asset_url:"https://connect.tiktok-minis.com/test/pushstone_asset.txt"},main:{root:"subpackages/main",main:"subpackages/main/game.js",asset_md5:"aece2f69cc168d7cefb248e8574c3371",code_url:"https://connect.tiktok-minis.com/test/main_code.txt",asset_url:"https://connect.tiktok-minis.com/test/main_asset.txt"}},M=[{id:""+1e17*Math.random(),packages:F},{id:""+1e17*Math.random(),packages:F}];function D(t,o){f.info("start check project"),function(e,t){f.info("start check project size");const{projectSizeLimit:n,isDev:o}=t,s=$(e,[]),i=Math.round(s/1048576),c=Math.round(n/1048576);if(s>n){const e=`Game size ${i}MB, must not exceed ${c}MB`;if(f.error(e),!o)throw new Error(e);console.log("%s",e)}else f.info(`Game size ${i}MB, check successfully!`),f.info("Game size check successfully!")}(t,o),function(t){f.info("start check config");const o=n.join(t,g);if(!e.existsSync(o)){const e="can not find game.json in game source code, please check it";throw f.error(e),new Error(e)}f.info("check config successfully")}(t),function(t){f.info("start check project config");const o=n.join(t,d);if(!e.existsSync(o)){const e="can not find project.config.json in game source code, please check it";throw f.error(e),new Error(e)}f.info("check project.config.json successfully");let s;try{s=JSON.parse(e.readFileSync(o,"utf-8"))}catch(e){throw f.error("check project.config.json failed, your project.config.json is invalid json format"),e}if(!(null==s?void 0:s.appid)){const e="can not find appid in project.config.json, you should add appid(client_key) in project.config.json";throw f.error(e),new Error(e)}}(t),f.info("check project successfully")}function E(e,t){f.info("start check main package"),f.info("check game.js");const n=a.join(e,"game.js");if(!r.existsSync(n)){const e="game.js must exist in main package!";throw f.error(e),new Error(e)}f.info("check game.js successfully"),function({entryDir:e,config:t}){f.info("start check main package size");const{mainPackageSizeLimit:n,isDev:o}=t,s=$(e,["subpackages"]),i=Math.round(s/1048576),c=Math.round(n/1048576);if(s>n){const e=`Main package size ${i}MB, must not exceed ${c}MB`;if(!o)throw f.error(e),new Error(e);console.log("%s",e)}else f.info(`Main package size ${i}MB, check successfully!`)}({entryDir:e,config:t}),f.info("check main package successfully")}function P(t,o){const{subpackageSizeLimit:s}=o||{};f.info("start check subpackages in game.json");const i=n.join(t,g),c=JSON.parse(e.readFileSync(i,"utf-8")).subpackages||[];(null==c?void 0:c.length)?c.forEach(o=>{if(!o.name||!o.root){const e="The subpackages configuration in game.json is invalid: the root or name field in one or more subpackages is empty. Please ensure that all subpackages have non-empty root and name values.";throw f.error(e),new Error(e)}const i=n.join(t,o.root);if(!e.existsSync(i)){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${o.name} entry dir, please check it`;throw f.error(e),new Error(e)}const c=o.root;if(!e.existsSync(n.join(t,c))){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${o.name} root, please check it`;throw f.error(e),new Error(e)}s&&function({entryDir:e,subPkgName:t,limit:o}){const s=n.join(e,"subpackages",t),i=$(s,[]),c=Math.round(i/1048576);if(c>o){const e=`${t} Sub package size ${c}MB, must not exceed ${o/1048576}MB`;throw f.error(e),new Error(e)}f.info(`${t} Sub package size ${c}MB, check successfully!`)}({entryDir:t,subPkgName:o.name,limit:s})}):f.info("subpackages is empty"),f.info("check subpackages in game.json successfully")}async function B({entryDir:e,config:t}){f.info("开始校验项目"),D(e,t),E(e,t),P(e,t),f.info("校验项目完成")}async function z(t,o){f.info("开始基于源代码中的 game.json 生成 packageConfig.json");const s=n.join(t,g),i=JSON.parse(e.readFileSync(s,"utf-8")),c={};c[p]={url:"",md5:"",root:"",main:"game.js",output:`${p}${y}`},function(e){if(!e||0===e.length)return[];function t(e){return String(e).replace(/^\/+|\/+$/g,"")}function o(e){return s(e)?n.join(e,"game.js"):e}function s(e){return e.endsWith("/")||!/\.[a-zA-Z0-9]+$/.test(e)}function i(e){return e.replace(/^subpackages\/([^/]+).*$/,"subpackages/$1")}return e.map(e=>{const n=t(e.name);return{main:o(e.root),root:i(e.root),name:n}})}(i.subpackages).forEach(({name:e,root:t,main:n})=>{c[e]={url:"",md5:"",root:t,main:n,output:`${e}${y}`}});const r={packages:c};return Object.keys(c).forEach(o=>{const s=n.join(t,c[o].main);if(!e.existsSync(s)){const e=`开发者源代码包中的子包 ${o} 中的 main 文件 ${c[o].main} 不存在`;throw f.error(e),new Error(e)}}),e.existsSync(o)&&e.rmSync(o,{recursive:!0}),e.mkdirSync(o),e.writeFileSync(n.join(o,m),JSON.stringify(r,null,2)),f.info("基于源代码中的 game.json 生成 packageConfig.json 成功"),r}function L(e){const t=[];return Object.values(e).forEach(e=>{Object.values(e).forEach(e=>{t.push(...e)})}),t}function A(t,o=[".js",".ts",".jsx",".tsx"],s=[]){let i=[];if(!e.existsSync(t))return i;const c=e.readdirSync(t,{withFileTypes:!0});for(const e of c){const c=n.join(t,e.name);if(e.isDirectory()){if(s.includes(e.name))continue;i=i.concat(A(c,o,[]))}else o.some(t=>e.name.endsWith(t))&&i.push(c)}return i}function T(t,i){const c=function(t){return e.readdirSync(t).filter(o=>{const s=n.join(t,o);return e.statSync(s).isDirectory()&&!["node_modules","dist","build","output"].includes(o)}).map(e=>e+"/")}(t),r=s.sync("**/*.js",{cwd:t,ignore:[]}),a=[];r.forEach(s=>{const i=n.join(t,s),r=e.readFileSync(i,"utf-8");let u;try{u=o.parse(r,{ecmaVersion:"latest",sourceType:"module"})}catch(e){return}N(u,o=>{if("CallExpression"===o.type&&"Identifier"===o.callee.type&&("require"===o.callee.name||"requireAsync"===o.callee.name)&&o.arguments.length>0&&"Literal"===o.arguments[0].type){const s=o.arguments[0].value;let r=null;if(s.startsWith("."))r=n.resolve(n.dirname(i),s);else if(s.startsWith("/"))r=n.resolve(t,"."+s);else for(const e of c)if(s.startsWith(e)){r=n.resolve(t,s);break}r&&!function(t){if(e.existsSync(t)&&e.statSync(t).isFile())return!0;const n=[".js",".ts",".jsx",".tsx"];for(const o of n)if(e.existsSync(t+o)&&e.statSync(t+o).isFile())return!0;return!1}(r)&&(r=null),a.push({file:n.relative(t,i),callee:o.callee.name,requirePath:s,resolvedPath:r?n.relative(t,r):""})}})});const u=a.map(e=>{let t="";Object.keys(i).forEach(n=>{e.resolvedPath&&e.resolvedPath.startsWith(i[n].root)&&(t=n)});let n="";return Object.keys(i).forEach(t=>{e.file.startsWith(i[t].root)&&(n=t)}),Object.assign(Object.assign({},e),{curModule:n,depModule:t})}),f=[".js",".ts",".jsx",".tsx"];return u.reduce((e,t)=>{return e[t.curModule]||(e[t.curModule]={}),t.curModule===t.depModule||t.resolvedPath&&(n=t.resolvedPath,f.some(e=>n.endsWith(e)))&&(e[t.curModule]=Object.assign(Object.assign({},e[t.curModule]),{[t.resolvedPath]:t.depModule})),e;var n},{})}function N(e,t){t(e);for(let n in e)if(e.hasOwnProperty(n)){const o=e[n];Array.isArray(o)?o.forEach(e=>e&&"string"==typeof e.type&&N(e,t)):o&&"string"==typeof o.type&&N(o,t)}}function C({gameEntry:t,gameOutput:o,pkgName:s,allDeps:c,allMaps:r,pkgConfig:a,destRoot:l,enableSourcemap:m}){let d=Date.now();f.info(`pack start,startTime:${d}`);const g=r[s],p={main:a.main,deps:c[s]||{},map:r[s]};u(l),e.writeFileSync(n.join(l,"moduleConfig.json"),JSON.stringify(p));for(const o in g){const c=g[o];if(!Array.isArray(c)||0===c.length)continue;const r=new i.Bundle({separator:"\n"});for(const s of c){const c=n.join(t,s);if(e.existsSync(c)&&(n.basename(c)!==o&&/\.js$/i.test(s))){const t=e.readFileSync(c,"utf-8"),n=s.replace(/\\/g,"/"),o=`define("game:${n}", ["require", "requireAsync", "module", "exports", "sandboxGlobal"], function(require, requireAsync, module, exports, sandboxGlobal){\nwith(sandboxGlobal){\n`,a="\n}\n});\n",u=new i(t,{filename:n}),f=new i(o).appendLeft(o.length,u.toString()).append(a);r.addSource({content:f,filename:n})}}const a=n.join(l,n.basename(o)),u=r.toString();if(m){const t=r.generateMap({hires:!0,includeContent:!0,file:o}),n=u+`\n//# sourceMappingURL=${`data:application/json;charset=utf-8;base64,${Buffer.from(t.toString(),"utf-8").toString("base64")}`}`+`\n//# sourceURL=${o}`;e.writeFileSync(a,n,"utf-8")}else{const t=u+`\n//# sourceURL=${o}`;e.writeFileSync(a,t,"utf-8")}f.info(`pack end,packName:${s}`)}}function G({pkgRoot:t,destRoot:o,packedFiles:s,gameEntry:i,gameOutput:c,skipDirs:r=[],isRoot:a=!0}){if(!e.existsSync(t))return;const f=e.readdirSync(t,{withFileTypes:!0});for(const l of f){const f=n.join(t,l.name),m=n.join(o,l.name);a&&l.isDirectory()&&r.includes(l.name)||(l.isDirectory()?G({pkgRoot:f,destRoot:m,gameOutput:c,packedFiles:s,gameEntry:i,skipDirs:r,isRoot:!1}):(u(n.dirname(m)),s.includes(n.relative(i,f))||e.copyFileSync(f,m)))}}async function R({gameEntry:t,gameOutput:o,enableSourcemap:s}){let i=Date.now();!function(t){!function t(o){let s;try{s=e.readdirSync(o)}catch(e){return void f.warn(`无法访问目录: ${o}, ${e.message}`)}s.forEach(s=>{const i=n.join(o,s);try{const n=e.statSync(i);n.isDirectory()?O.includes(s)?(e.rmSync(i,{recursive:!0,force:!0}),f.info(`已删除无用目录: ${i}`)):t(i):n.isFile()&&v.includes(s)&&(e.rmSync(i,{force:!0}),f.info(`已删除无用文件: ${i}`))}catch(e){f.warn(`无法处理: ${i}, ${e.message}`)}})}(t)}(t),f.info(`pack start,startTime:${i}`);const{packages:c}=await z(t,o),r=T(t,c),a=function(e,t){f.info("开始基于游戏源代码收集分包中的 JS 文件");const o={},s=Object.values(t).filter(e=>e.root&&""!==e.root).map(e=>e.root.split(n.sep)[0]).filter(Boolean),i=Array.from(new Set(s));for(const s in t){const c=t[s];let r,a;"__GAME__"===s?(a=A(e,[".js",".ts",".jsx",".tsx"],i),r=e):(r=n.join(e,c.root),a=A(r));const u=[];a.forEach(t=>{u.push(n.relative(e,t).replace(/\\/g,"/"))});const f=t[s].root;o[s]={[f?`${f}/game.pack.js`:"game.pack.js"]:u}}return f.info("基于游戏源代码收集分包中的 JS 文件完成"),o}(t,c);for(const e in a){f.info(`开始基于游戏源代码打包,包名:${e}`);let i=Date.now();const u=c[e],m=n.join(t,u.root),d=n.join(o,e,u.root);C({gameEntry:t,gameOutput:o,allDeps:r,allMaps:a,pkgName:e,pkgConfig:u,destRoot:d,enableSourcemap:s}),G({gameEntry:t,gameOutput:o,pkgRoot:m,destRoot:d,packedFiles:L(a),skipDirs:["subpackages"]});l(n.join(o,e)),f.info(`基于游戏源代码打包分包完成,包名:${e}`),f.info(`基于游戏源代码打包分包耗时,包名:${e},耗时:${Date.now()-i}ms`)}return f.info(`pack end,duration:${Date.now()-i}ms`),{}}exports.buildPkgs=async function(t){const{entry:o,output:s,rules:i}=t;let c=Date.now();f.init(s,!0),f.info("TTMG_PACK_VERSION: 0.0.18-beta.5"),f.info(`pack start, startTime:${c}`),await B({entryDir:o,config:{projectSizeLimit:(null==i?void 0:i.projectSizeLimit)||b,mainPackageSizeLimit:(null==i?void 0:i.mainPackageSizeLimit)||_,subpackageSizeLimit:null==i?void 0:i.subpackageSizeLimit}}),await R({gameEntry:o,gameOutput:s,enableSourcemap:!1}),await async function(t){let o=Date.now();f.info(`makeOdrPkgs start, ${o}`);const s=n.join(t,m),i=JSON.parse(e.readFileSync(s,"utf-8")),c=i.packages;for(const o in c){const s=n.join(t,o),i=n.join(t,`${o}${k}`),r=n.join(t,`${o}${h}`);u(i),u(r),e.cpSync(s,i,{recursive:!0}),e.cpSync(s,r,{recursive:!0}),x(i),w(r),c[`${o}${k}`]=Object.assign(Object.assign({},c[o]),{code_url:"",code_md5:"",output:`${o}${k}${y}`}),c[`${o}${h}`]=Object.assign(Object.assign({},c[o]),{asset_url:"",asset_md5:"",output:`${o}${h}${y}`})}e.writeFileSync(s,JSON.stringify(i,null,2)),f.info(`makeOdrPkgs end, ${Date.now()-o}ms`)}(s),f.info(`pack end:${Date.now()-c}ms`)},exports.debugPkgs=async function(t){const{entry:o,output:s,rules:i,enableLog:c=!1}=t;f.init(s,c);try{await B({entryDir:o,config:{projectSizeLimit:(null==i?void 0:i.projectSizeLimit)||b,mainPackageSizeLimit:(null==i?void 0:i.mainPackageSizeLimit)||_,isDev:!0}}),await R({gameEntry:o,gameOutput:s,enableSourcemap:!0}),await function(t){f.info("mergePkgs start");const o=n.join(t,m);let s;try{s=JSON.parse(e.readFileSync(o,"utf-8"))}catch(e){return void f.error(`读取配置文件失败: ${o}, err: ${e.message}`)}const i=s.packages;if(i&&"object"==typeof i){for(const o of Object.keys(i)){const s=n.join(t,o);if(!e.existsSync(s)){f.warn(`package 目录不存在: ${s}`);continue}const i=e.readdirSync(s);for(const o of i){const i=n.join(s,o),c=n.join(t,o);try{const t=e.statSync(i);if(t.isFile())e.copyFileSync(i,c);else if(t.isDirectory())e.cpSync(i,c,{recursive:!0});else if(t.isSymbolicLink()){const n=e.readlinkSync(i);e.existsSync(c)&&e.rmSync(c,{force:!0}),e.symlinkSync(n,c,t.isDirectory()?"dir":"file")}f.info(`合并文件/目录: ${i} -> ${c}`)}catch(e){f.error(`处理文件失败: ${i}, err: ${e.message}`)}}try{e.rmSync(s,{recursive:!0,force:!0}),f.info(`已删除 package 目录: ${s}`)}catch(e){f.error(`删除 package 目录失败: ${s}, err: ${e.message}`)}}f.info("mergePkgs 完成")}else f.error("配置文件中 packages 字段无效, 请检查配置")}(s);const t=e.readFileSync(n.join(s,m),"utf-8");return{isSuccess:!0,packages:JSON.parse(t).packages}}catch(e){return{isSuccess:!1,errorMsg:e.message}}},exports.downloadOdrPkgs=async function(){const t=n.join(__dirname,"temp");e.existsSync(t)||e.mkdirSync(t);const o=await async function(){return M}();f.info("odr_games"),function(e,t){for(const n of e){const{id:e,packages:o}=n,s=a.join(t,e);r.existsSync(s)||r.mkdirSync(s);for(const e in o){const t=o[e],{code_url:n}=t,i=require("child_process").execSync(`curl -s "${n}"`).toString();f.info(`downloadAndSaveCodesSync code, ${n}`);const c=`${e}.txt`,u=a.join(s,c);r.writeFileSync(u,i)}}}(o,t);const s={};for(const e of o){const{id:o,packages:i}=e,c=n.join(t,o);s[o]=$(c)}const i={};S.forEach(e=>{i[e]=[]});const c={};for(const r of o){const{id:o,packages:a}=r,u=s[o];if(u>j)continue;let l=null;for(const c of S){const r=n.join(t,c);e.existsSync(r)||e.mkdirSync(r);if(i[c].reduce((e,t)=>e+s[t],0)+u<=j){const s=n.join(t,o),a=n.join(r,o);e.cpSync(s,a,{recursive:!0}),i[c].push(o),l=c;break}}if(l){c[o]={tag:l,packages:{}};for(const e in a){const t=a[e];f.info(`TTMGODRConfig[gameId].packages, ${t}, ${e}`),c[o].packages[e]={root:t.root,main:t.main,asset_md5:t.asset_md5,asset_url:t.asset_url}}}}const u=n.join(t,"TTMG_ODR_CONFIG.json");e.writeFileSync(u,JSON.stringify(c,null,2),"utf-8"),f.info(`TTMG_ODR_CONFIG.json 已生成: ${u}`)},exports.writeOdrConfig=async function(e){f.info("writeOdrPackagesConfig start");const t=a.join(e,m),n=JSON.parse(r.readFileSync(t,"utf-8")),o=n.packages,s=n.odr_packages||{};for(const e in o){if(e.endsWith(k)){const t=o[e],n=e.split(k)[0];s[n]||(s[n]={}),s[n]=Object.assign(Object.assign({},s[n]),{root:t.root,code_output:t.output,code_md5:t.md5,main:t.main}),delete o[e]}if(e.endsWith(h)){const t=e.split(h)[0],n=o[e];s[t]=Object.assign(Object.assign({},s[t]),{root:n.root,asset_output:n.output,asset_md5:n.md5}),delete o[e]}}n.odr_packages=s,r.writeFileSync(t,JSON.stringify(n,null,2)),f.info("writeOdrPackagesConfig end")};
10
10
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/utils/ensureDirSync.ts","../src/utils/logger.ts","../src/utils/removeEmptyDir.ts","../src/constants/index.ts","../src/utils/getDirSizeSync.ts","../src/utils/deleteJsFilesSync.ts","../src/utils/deleteNoJsFilesSync.ts","../src/utils/removeSystemUseless.ts","../src/libs/odrPkgs/mock.ts","../src/libs/checkPkgs/checkProject.ts","../src/libs/checkPkgs/checkMainpackage.ts","../src/libs/checkPkgs/checkSubpackages.ts","../src/libs/checkPkgs/index.ts","../src/libs/makePkgs/setup.ts","../src/libs/makePkgs/collectMaps.ts","../src/libs/makePkgs/collectDeps.ts","../src/libs/makePkgs/pack.ts","../src/libs/makePkgs/partition.ts","../src/libs/makePkgs/index.ts","../src/libs/buildPkgs/index.ts","../src/libs/odrPkgs/makeOdrPkgs.ts","../src/libs/debugPkgs/index.ts","../src/libs/mergePkgs/index.ts","../src/libs/odrPkgs/downloadOdrPkgs.ts","../src/libs/odrPkgs/utils.ts","../src/libs/odrPkgs/writeOdrConfig.ts"],"sourcesContent":["import fs from 'fs';\nexport function ensureDirSync(dir) {\n if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true });\n}\n","import winston from 'winston';\nimport path from 'path';\n\nclass Logger {\n logger: winston.Logger;\n enableLog: boolean;\n output: any;\n constructor() {\n this.logger = null;\n }\n init(output: string, enableLog = true) {\n this.enableLog = enableLog;\n this.logger = winston.createLogger({\n level: 'info',\n format: winston.format.combine(\n winston.format.timestamp(),\n winston.format.json(), // 输出为 JSON 格式\n ),\n transports: [\n new winston.transports.File({\n filename: path.join(output, 'pack.log'),\n }),\n new winston.transports.Console(),\n ],\n });\n }\n info(msg: string) {\n if (this.enableLog) {\n this.logger.info(msg);\n }\n }\n error(msg) {\n if (this.enableLog) {\n this.logger.error(msg);\n }\n }\n warn(msg) {\n if (this.enableLog) {\n this.logger.warn(msg);\n }\n }\n debug(msg) {\n if (this.enableLog) {\n this.logger.debug(msg);\n }\n }\n log(msg) {\n this.logger.info(msg);\n }\n setOutput(output) {\n this.output = output;\n }\n}\n\nexport const logger = new Logger();\n","import fs from 'fs';\nimport path from 'path';\n\n/**\n * 移除指定目录下所有空文件夹(递归)\n * @param {string} dir 目标目录\n * @returns {boolean} 返回当前目录是否已被删除(即是否为空)\n */\nexport function removeEmptyDir(dir) {\n if (!fs.existsSync(dir)) return false;\n if (!fs.statSync(dir).isDirectory()) return false;\n\n let isEmpty = true;\n const files = fs.readdirSync(dir);\n for (const file of files) {\n const fullPath = path.join(dir, file);\n if (fs.statSync(fullPath).isDirectory()) {\n // 递归处理子目录\n const childIsEmpty = removeEmptyDir(fullPath);\n if (!childIsEmpty) isEmpty = false;\n } else {\n // 有文件,当前目录不为空\n isEmpty = false;\n }\n }\n\n // 如果当前目录为空,删除它\n if (isEmpty) {\n fs.rmSync(dir, { recursive: true });\n return true;\n }\n return false;\n}\n\n","export const GAME_PACK_CONFIG_FILE_NAME = 'packageConfig.json';\nexport const GAME_PROJECT_CONFIG_FILE_NAME = 'project.config.json';\nexport const GAME_ORIGIN_CONFIG_FILE_NAME = 'game.json';\nexport const GAME_MODULE_CONFIG_FILE_NAME = 'moduleConfig.json';\nexport const GAME_OUTPUT_CONFIG_FILE_NAME = 'output.json';\nexport const GAME_MAIN_PACKAGE_NAME = '__GAME__';\nexport const TTPKG_TAG = 'TPKG';\nexport const STTPKG_TAG = 'SPKG';\nexport const TTPKG_VERSION = 2;\nexport const ARK_KEY_LEN = 32;\nexport const ARK_INT_LEN = 4;\nexport const UNIT32_LEN = 4;\nexport const UINT16_LEN = 2;\nexport const STTPKG_VERSION = 1;\nexport const ZSTD_COMPRESS_LEVEL = 19;\nexport const ROLL_FILE_EXT = [\n '.js',\n '.json',\n '.wxml',\n '.ttml',\n '.wxss',\n '.ttss',\n];\nexport const EXT = '';\nexport const BROTLI_COMPRESS_TYPE = 1;\nexport const ZSTD_COMPRESS_TYPE = 3;\n\nexport const STTPKG_EXT = '.txt';\n\nexport const SUBPACKAGES_DIR_NAME = 'subpackages';\n\nexport const ODR_CODE_DIR_SUFFIX = '_code';\n\nexport const ODR_ASSET_DIR_SUFFIX = '_asset';\n\n// 60MB const ONE_ODR_PKG_LIMIT = 1024 * 1024;\nexport const ONE_ODR_PKG_LIMIT = 1024 * 1024 * 60;\n\nexport const AVAILABLE_ODR_TAG_LIST = [\n 'ttmg_odr_tag_1',\n 'ttmg_odr_tag_2',\n 'ttmg_odr_tag_3',\n];\n\nexport const PROJECT_SIZE_LIMIT = 30 * 1024 * 1024;\n\n\nexport const MAINPACKAGE_SIZE_LIMIT = 4 * 1024 * 1024;\n\n","import fs from 'fs';\nimport path from 'path';\n// 递归统计目录下所有文件的大小,支持我过滤某些文件夹\nexport function getDirSizeSync(dir: string, filterDirs: string[] = []) {\n let totalSize = 0;\n if (!fs.existsSync(dir)) return 0;\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n if (entry.isDirectory()) {\n if (!filterDirs.includes(entry.name)) {\n totalSize += getDirSizeSync(fullPath, filterDirs);\n }\n } else if (entry.isFile()) {\n totalSize += fs.statSync(fullPath).size;\n }\n }\n return totalSize;\n}\n","import fs from 'fs';\nimport path from 'path';\n\n/**\n * 同步删除指定目录及其子目录下所有 .js 文件\n * @param {string} dir 目标目录路径\n */\nexport function deleteJsFilesSync(dir) {\n // 读取目录内容\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n\n if (entry.isDirectory()) {\n // 递归处理子目录\n deleteJsFilesSync(fullPath);\n } else if (\n entry.isFile() &&\n (fullPath.endsWith('.js') || fullPath.endsWith('.js.map'))\n ) {\n try {\n fs.unlinkSync(fullPath);\n } catch (err) {}\n }\n }\n}\n","import fs from 'fs';\nimport path from 'path';\n\nexport function deleteNoJsFilesSync(dir) {\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n\n if (entry.isDirectory()) {\n // 递归处理子目录\n deleteNoJsFilesSync(fullPath);\n } else if (\n entry.isFile() &&\n !(fullPath.endsWith('.js') || fullPath.endsWith('.js.map'))\n ) {\n try {\n fs.unlinkSync(fullPath);\n } catch (err) {}\n }\n }\n}\n","import fs from 'fs';\nimport path from 'path';\n\nimport { logger } from './logger';\n\nconst uselessDirs = ['node_modules', '__MACOSX'];\nconst uselessFiles = ['.DS_Store', 'Thumbs.db'];\n\nexport function removeSystemUseless(gameEntry: string) {\n /**\n * 递归删除目录下的无用文件夹和无用文件\n */\n function removeUselessDirs(dir: string) {\n let files: string[];\n try {\n files = fs.readdirSync(dir);\n } catch (err) {\n // 目录不可访问,跳过\n logger.warn(`无法访问目录: ${dir}, ${err.message}`);\n return;\n }\n files.forEach(file => {\n const filePath = path.join(dir, file);\n try {\n const stat = fs.statSync(filePath);\n if (stat.isDirectory()) {\n if (uselessDirs.includes(file)) {\n // 删除无用目录\n fs.rmSync(filePath, { recursive: true, force: true });\n logger.info(`已删除无用目录: ${filePath}`);\n } else {\n // 递归处理子目录\n removeUselessDirs(filePath);\n }\n } else if (stat.isFile()) {\n if (uselessFiles.includes(file)) {\n // 删除无用文件\n fs.rmSync(filePath, { force: true });\n logger.info(`已删除无用文件: ${filePath}`);\n }\n }\n } catch (err) {\n // 某个文件/目录不可访问,跳过\n logger.warn(`无法处理: ${filePath}, ${err.message}`);\n }\n });\n }\n removeUselessDirs(gameEntry);\n}\n","interface ODRPackage {\n // md5: string;\n root: string;\n main: string;\n code_url: string;\n // code_md5: string;\n asset_md5: string;\n asset_url: string;\n}\n\ntype ODRPackages = Record<string, ODRPackage>;\nconst odrPackages: ODRPackages = {\n __GAME__: {\n // md5: '6eac4e8a7a7f54a20bcca6c67d71ebad',\n code_url: 'https://connect.tiktok-minis.com/test/__GAME___code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/__GAME___asset.txt',\n asset_md5: '8e3cc2ebd5536e9183b93fda34c1e7f4',\n root: '',\n main: 'game.js',\n },\n configBundle: {\n // md5: '575a0a1f0191e682fc27af897515740b',\n root: 'subpackages/configBundle',\n main: 'subpackages/configBundle/game.js',\n asset_md5: 'f624a1b37fb35d4dfb1975105a51ebd4',\n code_url: 'https://connect.tiktok-minis.com/test/configBundle_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/configBundle_asset.txt',\n },\n gameBundle: {\n // md5: '02e5d043be3156cfe2553e8af0f1b3d3',\n main: 'subpackages/gameBundle/game.js',\n root: 'subpackages/gameBundle',\n asset_md5: 'fcc9f686bbc8d6f983263e7132598b26',\n code_url: 'https://connect.tiktok-minis.com/test/gameBundle_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/gameBundle_asset.txt',\n },\n mainBundle: {\n // md5: 'f28aae73fbb0cc3b8876c2773a1c5789',\n main: 'subpackages/mainBundle/game.js',\n root: 'subpackages/mainBundle',\n asset_md5: '36b9674d48f9fbb0de6e8f48b194aebc',\n code_url: 'https://connect.tiktok-minis.com/test/mainBundle_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/mainBundle_asset.txt',\n },\n resources: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/resources',\n main: 'subpackages/resources/game.js',\n asset_md5: '1c4608d7690750f7bd5c00663013fc70',\n code_url: 'https://connect.tiktok-minis.com/test/resources_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/resources_asset.txt',\n },\n iceAndFire: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/iceAndFire',\n main: 'subpackages/iceAndFire/game.js',\n asset_md5: 'c7fd8e8506dd59725269027a52e2cae4',\n code_url: 'https://connect.tiktok-minis.com/test/iceAndFire_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/iceAndFire_asset.txt',\n },\n pushstone: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/pushstone',\n main: 'subpackages/pushstone/game.js',\n asset_md5: 'f5061dba94ad53e1f578325a7cb26a01',\n code_url: 'https://connect.tiktok-minis.com/test/pushstone_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/pushstone_asset.txt',\n },\n main: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/main',\n main: 'subpackages/main/game.js',\n asset_md5: 'aece2f69cc168d7cefb248e8574c3371',\n code_url: 'https://connect.tiktok-minis.com/test/main_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/main_asset.txt',\n },\n};\n\nexport const gameOdrPackages = [\n { id: `${Math.random() * 100000000000000000}`, packages: odrPackages },\n { id: `${Math.random() * 100000000000000000}`, packages: odrPackages },\n];\n","import path from 'path';\nimport fs from 'fs';\nimport { CheckConfig } from '@typings';\nimport {\n GAME_ORIGIN_CONFIG_FILE_NAME,\n GAME_PROJECT_CONFIG_FILE_NAME,\n} from '@constants';\nimport { logger, getDirSizeSync } from '@utils';\n\nexport function checkGameJson(entryDir: string) {\n logger.info('start check config');\n const gameJsonPath = path.join(entryDir, GAME_ORIGIN_CONFIG_FILE_NAME);\n if (!fs.existsSync(gameJsonPath)) {\n const errMsg =\n 'can not find game.json in game source code, please check it';\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n logger.info('check config successfully');\n }\n}\n\n/**\n * 检查游戏大小是否超出限制\n * @param param0\n * @returns\n */\n\nexport function checkProjectSize(entryDir: string, config: CheckConfig) {\n logger.info('start check project size');\n const { projectSizeLimit, isDev } = config;\n const gameSize = getDirSizeSync(entryDir, []);\n const gameSizeMB = Math.round(gameSize / (1024 * 1024));\n const limitMB = Math.round(projectSizeLimit / (1024 * 1024));\n if (gameSize > projectSizeLimit) {\n const errMsg = `Game size ${gameSizeMB}MB, must not exceed ${limitMB}MB`;\n logger.error(errMsg);\n if (!isDev) {\n throw new Error(errMsg);\n } else {\n console.log('\\x1b[1m\\x1b[33m%s\\x1b[0m', errMsg);\n }\n } else {\n /**\n * 游戏大小检查通过\n */\n logger.info(`Game size ${gameSizeMB}MB, check successfully!`);\n logger.info('Game size check successfully!');\n }\n}\n\nexport function checkProject(entryDir: string, config: CheckConfig) {\n logger.info('start check project');\n checkProjectSize(entryDir, config);\n checkGameJson(entryDir);\n checkProjectConfig(entryDir);\n logger.info('check project successfully');\n}\n\nexport function checkProjectConfig(entryDir: string) {\n logger.info('start check project config');\n const gameJsonConfigPath = path.join(entryDir, GAME_PROJECT_CONFIG_FILE_NAME);\n if (!fs.existsSync(gameJsonConfigPath)) {\n const errMsg =\n 'can not find project.config.json in game source code, please check it';\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n logger.info('check project.config.json successfully');\n }\n let gameJsonConfig: any;\n try {\n gameJsonConfig = JSON.parse(fs.readFileSync(gameJsonConfigPath, 'utf-8'));\n } catch (error) {\n logger.error(\n 'check project.config.json failed, your project.config.json is invalid json format',\n );\n throw error;\n }\n const appid = gameJsonConfig?.appid;\n if (!appid) {\n const errMsg =\n 'can not find appid in project.config.json, you should add appid(client_key) in project.config.json';\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n}\n","import * as path from 'path';\nimport * as fs from 'fs';\nimport { CheckConfig } from '@typings';\nimport { logger, getDirSizeSync } from '@utils';\n/**\n * 检查主包大小是否超出限制\n * @param param0\n * @returns\n */\nexport function checkMainPackageSize({\n entryDir,\n config,\n}: {\n entryDir: string;\n config: CheckConfig;\n}) {\n logger.info('start check main package size');\n const { mainPackageSizeLimit, isDev } = config;\n const mainPkgSize = getDirSizeSync(entryDir, ['subpackages']);\n const mainPkgSizeMB = Math.round(mainPkgSize / (1024 * 1024));\n const limitMB = Math.round(mainPackageSizeLimit / (1024 * 1024));\n if (mainPkgSize > mainPackageSizeLimit) {\n const errMsg = `Main package size ${mainPkgSizeMB}MB, must not exceed ${limitMB}MB`;\n if (!isDev) {\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n console.log('\\x1b[1m\\x1b[33m%s\\x1b[0m', errMsg);\n }\n } else {\n /**\n * 主包大小检查通过\n */\n logger.info(`Main package size ${mainPkgSizeMB}MB, check successfully!`);\n }\n}\n\nexport function checkMainPackage(entryDir: string, config: CheckConfig) {\n logger.info('start check main package');\n /**\n * 检查 game.js 是否存在\n */\n\n logger.info('check game.js');\n const gameJsPath = path.join(entryDir, 'game.js');\n if (!fs.existsSync(gameJsPath)) {\n const errMsg = 'game.js must exist in main package!';\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n logger.info('check game.js successfully');\n\n /**\n * 检查主包大小是否超出限制\n */\n checkMainPackageSize({\n entryDir,\n config,\n });\n logger.info('check main package successfully');\n}\n","import { logger, getDirSizeSync } from '@utils';\nimport { CheckConfig } from '@typings';\nimport { GAME_ORIGIN_CONFIG_FILE_NAME } from '@constants';\nimport path from 'path';\nimport fs from 'fs';\n\nexport function checkSubpackages(\n entryDir: string,\n config?: Pick<CheckConfig, 'subpackageSizeLimit'>,\n) {\n const { subpackageSizeLimit } = config || {};\n /**\n * 开始校验 subpackages 配置是否有效\n */\n logger.info('start check subpackages in game.json');\n const gameJsonPath = path.join(entryDir, GAME_ORIGIN_CONFIG_FILE_NAME);\n const gameJson = JSON.parse(fs.readFileSync(gameJsonPath, 'utf-8'));\n const subpackages: Array<{\n name: string;\n root: string;\n }> = gameJson.subpackages || [];\n if (!subpackages?.length) {\n logger.info('subpackages is empty');\n } else {\n subpackages.forEach(sub => {\n /**\n * 校验 subpackage 配置是否为空\n */\n if (!sub.name || !sub.root) {\n const errMsg =\n 'The subpackages configuration in game.json is invalid: the root or name field in one or more subpackages is empty. Please ensure that all subpackages have non-empty root and name values.';\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n /**\n * 校验 subpackage 配置是否有效\n */\n const subpackageEntryDir = path.join(entryDir, sub.root);\n // 检查是文件或者文件夹是否存在\n if (!fs.existsSync(subpackageEntryDir)) {\n const errMsg = `The subpackages configuration in game.json is invalid: can not find subpackage ${sub.name} entry dir, please check it`;\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n const subpackageRoot = sub.root;\n if (!fs.existsSync(path.join(entryDir, subpackageRoot))) {\n const errMsg = `The subpackages configuration in game.json is invalid: can not find subpackage ${sub.name} root, please check it`;\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n /**\n * 校验 subpackage 包大小\n */\n\n if (subpackageSizeLimit) {\n checkSubPackageSize({\n entryDir,\n subPkgName: sub.name,\n limit: subpackageSizeLimit,\n });\n }\n });\n }\n logger.info('check subpackages in game.json successfully');\n}\n\n/**\n * 检查子包大小是否超出限制\n * @param param0\n * @returns\n */\n\nfunction checkSubPackageSize({\n entryDir,\n subPkgName,\n limit,\n}: {\n entryDir: string;\n subPkgName: string;\n limit: number;\n}) {\n const subPkgDir = path.join(entryDir, 'subpackages', subPkgName);\n const subPkgSize = getDirSizeSync(subPkgDir, []);\n const sizeMB = Math.round(subPkgSize / (1024 * 1024));\n if (sizeMB > limit) {\n const errMsg = `${subPkgName} Sub package size ${sizeMB}MB, must not exceed ${\n limit / (1024 * 1024)\n }MB`;\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n /**\n * 子包大小检查通过\n */\n logger.info(\n `${subPkgName} Sub package size ${sizeMB}MB, check successfully!`,\n );\n }\n}\n","import { logger } from '@utils';\nimport { CheckConfig } from '@typings';\nimport { checkProject } from './checkProject';\nimport { checkMainPackage } from './checkMainpackage';\nimport { checkSubpackages } from './checkSubpackages';\n\nexport async function checkPkgs({\n entryDir,\n config,\n}: {\n entryDir: string;\n config: CheckConfig;\n}) {\n logger.info('开始校验项目');\n /**\n * 1. 校验项目\n */\n checkProject(entryDir, config);\n /**\n * 2. 校验主包\n */\n checkMainPackage(entryDir, config);\n /**\n * 3. 如果有,校验分包\n */\n checkSubpackages(entryDir, config);\n logger.info('校验项目完成');\n}\n","import fs from 'fs';\nimport path from 'path';\nimport {\n GAME_PACK_CONFIG_FILE_NAME,\n GAME_ORIGIN_CONFIG_FILE_NAME,\n GAME_MAIN_PACKAGE_NAME,\n STTPKG_EXT,\n} from '@constants';\n\nimport { logger } from '@utils';\n\nexport async function setup(entryDir: string, outputDir: string) {\n logger.info('开始基于源代码中的 game.json 生成 packageConfig.json');\n const gameJsonPath = path.join(entryDir, GAME_ORIGIN_CONFIG_FILE_NAME);\n const gameJson = JSON.parse(fs.readFileSync(gameJsonPath, 'utf-8'));\n\n const packages = {};\n // 主包\n packages[GAME_MAIN_PACKAGE_NAME] = {\n url: '',\n md5: '',\n root: '',\n main: 'game.js',\n output: `${GAME_MAIN_PACKAGE_NAME}${STTPKG_EXT}`,\n };\n\n transformSubPackages(gameJson.subpackages).forEach(({ name, root, main }) => {\n // root 是目录\n packages[name] = {\n url: '',\n md5: '',\n root,\n main,\n output: `${name}${STTPKG_EXT}`,\n };\n });\n\n const result = {\n packages,\n };\n\n /**\n * 校验子包的 main 文件是否存在\n */\n Object.keys(packages).forEach(pkg => {\n const pkgPath = path.join(entryDir, packages[pkg].main);\n if (!fs.existsSync(pkgPath)) {\n const errMsg = `开发者源代码包中的子包 ${pkg} 中的 main 文件 ${packages[pkg].main} 不存在`;\n logger.error(errMsg);\n\n throw new Error(errMsg);\n }\n });\n\n if (fs.existsSync(outputDir)) {\n fs.rmSync(outputDir, { recursive: true });\n }\n fs.mkdirSync(outputDir);\n fs.writeFileSync(\n path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME),\n JSON.stringify(result, null, 2),\n );\n logger.info('基于源代码中的 game.json 生成 packageConfig.json 成功');\n return result;\n}\n\nfunction transformSubPackages(\n configs: Array<{\n name: string;\n root: string;\n }>,\n) {\n if (!configs || configs.length === 0) {\n return [];\n }\n function normalizeName(name) {\n return String(name).replace(/^\\/+|\\/+$/g, '');\n }\n\n function getMainFromRoot(root: string) {\n if (isDirectory(root)) {\n return path.join(root, 'game.js');\n }\n return root;\n }\n\n function isDirectory(str) {\n // 以 / 结尾,或没有 .js 结尾且没有 . 号\n return str.endsWith('/') || !/\\.[a-zA-Z0-9]+$/.test(str);\n }\n\n function normalizeRoot(root: string): string {\n /**\n * 只保留 subpackages/xxx 两层\n * 例如:\n * 'subpackages/moduleA/pages/index' -> 'subpackages/moduleA'\n * 'subpackages/moduleB' -> 'subpackages/moduleB'\n * 'subpackages/moduleC/' -> 'subpackages/moduleC'\n */\n // 将捕获组从匹配两层目录 ([^/]+\\/[^/]+) 改为只匹配一层 ([^/]+)\n return root.replace(/^subpackages\\/([^/]+).*$/, 'subpackages/$1');\n }\n\n return configs.map(config => {\n const name = normalizeName(config.name);\n let main = getMainFromRoot(config.root);\n // test\n let root = normalizeRoot(config.root);\n return { main, root, name };\n });\n}\n","import fs from 'fs';\nimport path from 'path';\nimport { logger } from '@utils';\n\nexport function flattenMaps(allMaps) {\n const list = [];\n Object.values(allMaps).forEach(map => {\n Object.values(map).forEach(arr => {\n list.push(...arr);\n });\n });\n return list;\n}\n\n// 递归收集 JS 文件,但可排除指定目录\nfunction collectAllJsFiles(\n dir,\n exts = ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs = [],\n) {\n let files = [];\n if (!fs.existsSync(dir)) return files;\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n if (entry.isDirectory()) {\n if (excludeDirs.includes(entry.name)) continue; // 跳过排除目录\n files = files.concat(collectAllJsFiles(fullPath, exts, [])); // 只排除一级\n } else if (exts.some(ext => entry.name.endsWith(ext))) {\n files.push(fullPath);\n }\n }\n return files;\n}\n\n// 构建每个包的 map\nexport function collectMaps(\n entryDir,\n packages: Record<string, { root: string }>,\n) {\n logger.info('开始基于游戏源代码收集分包中的 JS 文件');\n const result = {};\n // 1. 先收集所有分包根目录名(如 subpackages/level1、subpackages/level2)\n const subRoots = Object.values(packages)\n .filter(pkg => pkg.root && pkg.root !== '')\n .map(pkg => pkg.root.split(path.sep)[0]) // 只提取一级目录名\n .filter(Boolean);\n const uniqueSubRoots = Array.from(new Set(subRoots));\n\n for (const pkgName in packages) {\n const pkg = packages[pkgName];\n let srcRoot;\n let jsFiles;\n if (pkgName === '__GAME__') {\n // 主包:只收集 entryDir 下非分包目录\n jsFiles = collectAllJsFiles(\n entryDir,\n ['.js', '.ts', '.jsx', '.tsx'],\n uniqueSubRoots,\n );\n srcRoot = entryDir;\n } else {\n // 分包:只收集自己目录\n srcRoot = path.join(entryDir, pkg.root);\n jsFiles = collectAllJsFiles(srcRoot);\n }\n // 以包的 srcRoot 为基准生成相对路径\n const list = [];\n jsFiles.forEach(absPath => {\n list.push(path.relative(entryDir, absPath).replace(/\\\\/g, '/'));\n });\n const packNameRoot = packages[pkgName].root;\n result[pkgName] = {\n [packNameRoot ? `${packNameRoot}/game.pack.js` : 'game.pack.js']: list,\n };\n }\n logger.info('基于游戏源代码收集分包中的 JS 文件完成');\n return result;\n}\n","import fs from 'fs';\nimport path from 'path';\nimport acorn from 'acorn';\nimport glob from 'glob';\n\nexport function collectDeps(gameEntry, packages) {\n // 自动扫描 game 目录下的一级文件夹作为根前缀\n const rootPrefixes = getRootPrefixes(gameEntry);\n\n // 查找所有 JS 文件\n const files = glob.sync('**/*.js', {\n cwd: gameEntry,\n ignore: []\n });\n\n const requireCalls = [];\n\n files.forEach(file => {\n const fullPath = path.join(gameEntry, file);\n const code = fs.readFileSync(fullPath, 'utf-8');\n\n let ast;\n try {\n ast = acorn.parse(code, { ecmaVersion: 'latest', sourceType: 'module' });\n } catch (e) {\n // 跳过无法解析的文件\n return;\n }\n\n walkAst(ast, node => {\n // require('xxx') 或 requireAsync('xxx')\n if (\n node.type === 'CallExpression' &&\n node.callee.type === 'Identifier' &&\n (node.callee.name === 'require' ||\n node.callee.name === 'requireAsync') &&\n node.arguments.length > 0 &&\n node.arguments[0].type === 'Literal'\n ) {\n const requirePath = node.arguments[0].value;\n let resolvedPath = null;\n\n if (requirePath.startsWith('.')) {\n resolvedPath = path.resolve(path.dirname(fullPath), requirePath);\n } else if (requirePath.startsWith('/')) {\n resolvedPath = path.resolve(gameEntry, '.' + requirePath);\n } else {\n for (const prefix of rootPrefixes) {\n if (requirePath.startsWith(prefix)) {\n resolvedPath = path.resolve(gameEntry, requirePath);\n break;\n }\n }\n }\n\n // 校验文件是否存在(支持 .js/.ts/.jsx/.tsx 后缀自动补全)\n if (resolvedPath && !fileExistsWithExtensions(resolvedPath)) {\n resolvedPath = null;\n }\n\n requireCalls.push({\n file: path.relative(gameEntry, fullPath),\n callee: node.callee.name,\n requirePath,\n resolvedPath: resolvedPath\n ? path.relative(gameEntry, resolvedPath)\n : '',\n });\n }\n });\n });\n\n const newRequireCalls = requireCalls.map(i => {\n /**\n * 基于 packages 来判断\n */\n let depModule = '';\n Object.keys(packages).forEach(pkg => {\n if (i.resolvedPath && i.resolvedPath.startsWith(packages[pkg].root)) {\n depModule = pkg;\n }\n });\n\n let curModule = '';\n Object.keys(packages).forEach(pkg => {\n if (i.file.startsWith(packages[pkg].root)) {\n curModule = pkg;\n }\n });\n return {\n ...i,\n curModule,\n depModule,\n };\n });\n\n // 只统计 JS/TS/JSX/TSX 文件依赖\n const exts = ['.js', '.ts', '.jsx', '.tsx'];\n const isJsLike = f => exts.some(ext => f.endsWith(ext));\n\n const result = newRequireCalls.reduce((acc, cur) => {\n if (!acc[cur.curModule]) {\n acc[cur.curModule] = {};\n }\n if (cur.curModule === cur.depModule) {\n return acc;\n }\n if (cur.resolvedPath && isJsLike(cur.resolvedPath)) {\n acc[cur.curModule] = {\n ...acc[cur.curModule],\n [cur.resolvedPath]: cur.depModule,\n };\n }\n return acc;\n }, {});\n return result;\n}\n\nfunction getRootPrefixes(gameEntry) {\n return fs\n .readdirSync(gameEntry)\n .filter(name => {\n const fullPath = path.join(gameEntry, name);\n return (\n fs.statSync(fullPath).isDirectory() &&\n !['node_modules', 'dist', 'build', 'output'].includes(name)\n );\n })\n .map(name => name + '/');\n}\n\nfunction walkAst(node, cb) {\n cb(node);\n for (let key in node) {\n if (node.hasOwnProperty(key)) {\n const child = node[key];\n if (Array.isArray(child)) {\n child.forEach(c => c && typeof c.type === 'string' && walkAst(c, cb));\n } else if (child && typeof child.type === 'string') {\n walkAst(child, cb);\n }\n }\n }\n}\n\n// 支持 .js/.ts/.jsx/.tsx 自动补全\nfunction fileExistsWithExtensions(filePath) {\n if (fs.existsSync(filePath) && fs.statSync(filePath).isFile()) return true;\n const exts = ['.js', '.ts', '.jsx', '.tsx'];\n for (const ext of exts) {\n if (fs.existsSync(filePath + ext) && fs.statSync(filePath + ext).isFile())\n return true;\n }\n return false;\n}\n","import fs from 'fs';\nimport path from 'path';\nimport { GAME_MODULE_CONFIG_FILE_NAME } from '@constants';\nimport { logger, ensureDirSync } from '@utils';\nimport MagicString, { Bundle } from 'magic-string';\n\nexport function pack({\n gameEntry,\n gameOutput,\n pkgName,\n allDeps,\n allMaps,\n pkgConfig,\n destRoot,\n enableSourcemap,\n}: {\n gameOutput: string;\n gameEntry: string;\n pkgName: string;\n allDeps: Record<string, string[]>;\n allMaps: Record<string, Record<string, string[]>>;\n pkgConfig: {\n main: string;\n root: string;\n };\n destRoot: string;\n enableSourcemap?: boolean;\n}) {\n let startTime = Date.now();\n logger.info(`pack start,startTime:${startTime}`);\n const packMap = allMaps[pkgName];\n const moduleConfig = {\n main: pkgConfig.main,\n deps: allDeps[pkgName] || {},\n map: allMaps[pkgName],\n };\n\n ensureDirSync(destRoot);\n fs.writeFileSync(\n path.join(destRoot, GAME_MODULE_CONFIG_FILE_NAME),\n JSON.stringify(moduleConfig),\n );\n\n for (const packFileName in packMap) {\n const relPaths = packMap[packFileName];\n if (!Array.isArray(relPaths) || relPaths.length === 0) continue;\n\n const concat = new Bundle({ separator: '\\n' });\n for (const relPath of relPaths) {\n const absPath = path.join(gameEntry, relPath);\n if (!fs.existsSync(absPath)) continue;\n if (path.basename(absPath) === packFileName) continue;\n\n /**\n * 只匹配 js 文件\n */\n if (/\\.js$/i.test(relPath)) {\n const code = fs.readFileSync(absPath, 'utf-8');\n const fileId = relPath.replace(/\\\\/g, '/');\n const defineHeader = `define(\"game:${fileId}\", [\"require\", \"requireAsync\", \"module\", \"exports\", \"sandboxGlobal\"], function(require, requireAsync, module, exports, sandboxGlobal){\\nwith(sandboxGlobal){\\n`;\n const defineFooter = `\\n}\\n});\\n`;\n\n const fileMagic = new MagicString(code, { filename: fileId });\n fileMagic.prepend(defineHeader);\n fileMagic.append(defineFooter);\n concat.addSource(fileMagic);\n }\n }\n\n const outFile = path.join(destRoot, path.basename(packFileName));\n const bundledCode = concat.toString();\n\n if (enableSourcemap) {\n const map = concat.generateMap({\n hires: true,\n includeContent: true,\n file: packFileName,\n });\n\n const mapBase64 = Buffer.from(map.toString(), 'utf-8').toString('base64');\n const sourceMapDataURI = `data:application/json;charset=utf-8;base64,${mapBase64}`;\n\n const fileContent =\n bundledCode +\n `\\n//# sourceMappingURL=${sourceMapDataURI}` +\n `\\n//# sourceURL=${packFileName}`;\n\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n } else {\n const fileContent = bundledCode + `\\n//# sourceURL=${packFileName}`;\n\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n }\n\n logger.info(`pack end,packName:${pkgName}`);\n }\n}\n","import path from 'path';\nimport fs from 'fs';\nimport { ensureDirSync } from '@utils';\n\nexport function partition({\n pkgRoot,\n destRoot,\n packedFiles,\n gameEntry,\n gameOutput,\n skipDirs = [],\n isRoot = true\n}) {\n if (!fs.existsSync(pkgRoot)) return;\n const entries = fs.readdirSync(pkgRoot, { withFileTypes: true });\n for (const entry of entries) {\n const srcPath = path.join(pkgRoot, entry.name);\n const destPath = path.join(destRoot, entry.name);\n // 只在第一层判断是否跳过分包目录\n if (isRoot && entry.isDirectory() && skipDirs.includes(entry.name)) {\n continue; // 跳过分包目录\n }\n\n if (entry.isDirectory()) {\n partition({\n pkgRoot: srcPath,\n destRoot: destPath,\n gameOutput,\n packedFiles,\n gameEntry,\n skipDirs,\n isRoot: false, // 递归下去不再判断\n });\n } else {\n ensureDirSync(path.dirname(destPath));\n if (!packedFiles.includes(path.relative(gameEntry, srcPath))) {\n fs.copyFileSync(srcPath, destPath);\n }\n }\n }\n}\n","import fs from 'fs';\nimport path from 'path';\nimport { SUBPACKAGES_DIR_NAME } from '@constants';\nimport { logger, removeEmptyDir, removeSystemUseless } from '@utils';\nimport { setup } from './setup';\nimport { flattenMaps, collectMaps } from './collectMaps';\nimport { collectDeps } from './collectDeps';\nimport { pack } from './pack';\nimport { partition } from './partition';\nexport async function makePkgs({ gameEntry, gameOutput, enableSourcemap }) {\n let startTime = Date.now();\n removeSystemUseless(gameEntry);\n logger.info(`pack start,startTime:${startTime}`);\n const { packages } = await setup(gameEntry, gameOutput);\n const allDeps = collectDeps(gameEntry, packages);\n const allMaps = collectMaps(gameEntry, packages);\n const pkgOutput = {};\n\n for (const pkgName in allMaps) {\n logger.info(`开始基于游戏源代码打包,包名:${pkgName}`);\n let startTime = Date.now();\n const pkgConfig = packages[pkgName];\n const pkgRoot = path.join(gameEntry, pkgConfig.root);\n const destRoot = path.join(gameOutput, pkgName, pkgConfig.root);\n /**\n * 打包\n */\n pack({\n gameEntry,\n gameOutput,\n allDeps,\n allMaps,\n pkgName,\n pkgConfig,\n destRoot,\n enableSourcemap,\n });\n\n /**\n * 分包\n */\n partition({\n gameEntry,\n gameOutput,\n pkgRoot,\n destRoot,\n packedFiles: flattenMaps(allMaps),\n skipDirs: [SUBPACKAGES_DIR_NAME],\n });\n\n const pkgRootDir = path.join(gameOutput, pkgName);\n removeEmptyDir(pkgRootDir);\n logger.info(`基于游戏源代码打包分包完成,包名:${pkgName}`);\n logger.info(\n `基于游戏源代码打包分包耗时,包名:${pkgName},耗时:${\n Date.now() - startTime\n }ms`,\n );\n }\n // splitOdrPackages(gameOutput);\n logger.info(`pack end,duration:${Date.now() - startTime}ms`);\n\n return pkgOutput;\n}\n","import { logger } from '@utils';\nimport { PROJECT_SIZE_LIMIT, MAINPACKAGE_SIZE_LIMIT } from '@constants';\nimport { BuildConfig } from '@typings';\nimport { checkPkgs } from '../checkPkgs';\nimport { makePkgs } from '../makePkgs';\nimport { makeOdrPkgs } from '../odrPkgs';\n\nexport async function buildPkgs(config: BuildConfig) {\n const { entry: entryDir, output: outputDir, rules } = config;\n let startTime = Date.now();\n logger.init(outputDir, true);\n logger.info(`TTMG_PACK_VERSION: ${process.env.TTMG_PACK_VERSION}`);\n logger.info(`pack start, startTime:${startTime}`);\n\n /**\n * 校验\n */\n await checkPkgs({\n entryDir,\n config: {\n projectSizeLimit: rules?.projectSizeLimit || PROJECT_SIZE_LIMIT,\n mainPackageSizeLimit:\n rules?.mainPackageSizeLimit || MAINPACKAGE_SIZE_LIMIT,\n subpackageSizeLimit: rules?.subpackageSizeLimit,\n },\n });\n /**\n * 打包\n */\n await makePkgs({\n gameEntry: entryDir,\n gameOutput: outputDir,\n enableSourcemap: false,\n });\n\n /**\n * 分拆 odr 包\n */\n await makeOdrPkgs(outputDir);\n\n logger.info(`pack end:${Date.now() - startTime}ms`);\n}\n","import path from 'path';\nimport fs from 'fs';\nimport {\n ensureDirSync,\n deleteNoJsFilesSync,\n logger,\n deleteJsFilesSync,\n} from '@utils';\n\nimport {\n GAME_PACK_CONFIG_FILE_NAME,\n ODR_ASSET_DIR_SUFFIX,\n ODR_CODE_DIR_SUFFIX,\n STTPKG_EXT,\n} from '@constants';\n\nexport async function makeOdrPkgs(outputDir: string) {\n let startTime = Date.now();\n logger.info(`makeOdrPkgs start, ${startTime}`);\n const outputPath = path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME);\n const config = JSON.parse(fs.readFileSync(outputPath, 'utf-8'));\n const packages = config.packages;\n for (const pkgName in packages) {\n /**\n * 将原本的分包配置,转换为 odr 的配置,\n * 将原本产物按照原路径,分别拆分成两个文件夹:js 部分 package_name_code,asset 部分 package_name_asset\n */\n const pkgOutput = path.join(outputDir, pkgName);\n const codeOutput = path.join(outputDir, `${pkgName}${ODR_CODE_DIR_SUFFIX}`);\n const assetOutput = path.join(\n outputDir,\n `${pkgName}${ODR_ASSET_DIR_SUFFIX}`,\n );\n ensureDirSync(codeOutput);\n ensureDirSync(assetOutput);\n fs.cpSync(pkgOutput, codeOutput, { recursive: true });\n fs.cpSync(pkgOutput, assetOutput, { recursive: true });\n deleteNoJsFilesSync(codeOutput);\n deleteJsFilesSync(assetOutput);\n\n /**\n * 写入配置\n */\n packages[`${pkgName}${ODR_CODE_DIR_SUFFIX}`] = {\n ...packages[pkgName],\n code_url: '',\n code_md5: '',\n output: `${pkgName}${ODR_CODE_DIR_SUFFIX}${STTPKG_EXT}`,\n };\n packages[`${pkgName}${ODR_ASSET_DIR_SUFFIX}`] = {\n ...packages[pkgName],\n asset_url: '',\n asset_md5: '',\n output: `${pkgName}${ODR_ASSET_DIR_SUFFIX}${STTPKG_EXT}`,\n };\n }\n fs.writeFileSync(outputPath, JSON.stringify(config, null, 2));\n logger.info(`makeOdrPkgs end, ${Date.now() - startTime}ms`);\n}\n","import fs from 'fs';\nimport path from 'path';\n\nimport { logger } from '@utils';\nimport {\n PROJECT_SIZE_LIMIT,\n MAINPACKAGE_SIZE_LIMIT,\n GAME_PACK_CONFIG_FILE_NAME,\n} from '@constants';\nimport { BuildConfig } from '@typings';\nimport { checkPkgs } from '../checkPkgs';\nimport { mergePkgs } from '../mergePkgs';\nimport { makePkgs } from '../makePkgs';\n\nexport async function debugPkgs(config: BuildConfig): Promise<{\n isSuccess: boolean;\n errorMsg?: string;\n packages?: any;\n}> {\n const {\n entry: entryDir,\n output: outputDir,\n rules,\n enableLog = false,\n } = config;\n logger.init(outputDir, enableLog);\n try {\n /**\n * 校验\n */\n await checkPkgs({\n entryDir,\n config: {\n projectSizeLimit: rules?.projectSizeLimit || PROJECT_SIZE_LIMIT,\n mainPackageSizeLimit:\n rules?.mainPackageSizeLimit || MAINPACKAGE_SIZE_LIMIT,\n isDev: true,\n },\n });\n /**\n * 打包\n */\n await makePkgs({\n gameEntry: entryDir,\n gameOutput: outputDir,\n enableSourcemap: true,\n });\n\n /**\n * 合并\n */\n await mergePkgs(outputDir);\n\n /**\n * 返回 packageConfig\n */\n const packageConfig = fs.readFileSync(\n path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME),\n 'utf-8',\n );\n const packageConfigJson = JSON.parse(packageConfig);\n return {\n isSuccess: true,\n packages: packageConfigJson.packages,\n };\n } catch (error) {\n return {\n isSuccess: false,\n errorMsg: error.message,\n };\n }\n}\n","import path from 'path';\nimport fs from 'fs';\nimport { logger } from '@utils';\nimport { ProjectConfig } from '@typings';\nimport { GAME_PACK_CONFIG_FILE_NAME } from '@constants';\n\n/**\n * 将 packages 中的每个 package 下的文件内容合并到 outputDir 下,不保留 package name\n * @param outputDir 输出目录\n */\nexport function mergePkgs(outputDir: string) {\n logger.info('mergePkgs start');\n const gamePackConfigPath = path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME);\n let gamePackConfig: ProjectConfig;\n try {\n gamePackConfig = JSON.parse(fs.readFileSync(gamePackConfigPath, 'utf-8'));\n } catch (err) {\n logger.error(\n `读取配置文件失败: ${gamePackConfigPath}, err: ${err.message}`,\n );\n return;\n }\n\n const packages = gamePackConfig.packages;\n if (!packages || typeof packages !== 'object') {\n logger.error('配置文件中 packages 字段无效, 请检查配置');\n return;\n }\n\n for (const pkgName of Object.keys(packages)) {\n const pkgDir = path.join(outputDir, pkgName);\n if (!fs.existsSync(pkgDir)) {\n logger.warn(`package 目录不存在: ${pkgDir}`);\n continue;\n }\n\n const files = fs.readdirSync(pkgDir);\n for (const file of files) {\n const srcPath = path.join(pkgDir, file);\n const destPath = path.join(outputDir, file);\n\n try {\n const stat = fs.statSync(srcPath);\n if (stat.isFile()) {\n fs.copyFileSync(srcPath, destPath);\n } else if (stat.isDirectory()) {\n fs.cpSync(srcPath, destPath, { recursive: true });\n } else if (stat.isSymbolicLink()) {\n const linkTarget = fs.readlinkSync(srcPath);\n // 判断目标是否已存在符号链接\n if (fs.existsSync(destPath)) {\n fs.rmSync(destPath, { force: true });\n }\n fs.symlinkSync(\n linkTarget,\n destPath,\n stat.isDirectory() ? 'dir' : 'file',\n );\n }\n logger.info(`合并文件/目录: ${srcPath} -> ${destPath}`);\n } catch (err) {\n logger.error(`处理文件失败: ${srcPath}, err: ${err.message}`);\n }\n }\n\n // 删除 package 目录\n try {\n fs.rmSync(pkgDir, { recursive: true, force: true });\n logger.info(`已删除 package 目录: ${pkgDir}`);\n } catch (err) {\n logger.error(`删除 package 目录失败: ${pkgDir}, err: ${err.message}`);\n }\n }\n\n logger.info('mergePkgs 完成');\n}\n","/**\n * 合并请求拆包\n */\nimport fs from 'fs';\nimport path from 'path';\nimport { getDirSizeSync, logger } from '@utils';\nimport { fetchAllGames, downloadAndSaveCodesSync } from './utils';\nimport { AVAILABLE_ODR_TAG_LIST, ONE_ODR_PKG_LIMIT } from '@constants';\n\nexport async function downloadOdrPkgs() {\n const TEMP_DIR = path.join(__dirname, 'temp');\n if (!fs.existsSync(TEMP_DIR)) fs.mkdirSync(TEMP_DIR);\n const games = await fetchAllGames();\n logger.info('odr_games');\n downloadAndSaveCodesSync(games, TEMP_DIR);\n const gamesContentSize = {};\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameDir = path.join(TEMP_DIR, gameId);\n gamesContentSize[gameId] = getDirSizeSync(gameDir);\n }\n // ODR TAG 分配\n const odrTagGameMap = {};\n AVAILABLE_ODR_TAG_LIST.forEach(tag => {\n odrTagGameMap[tag] = [];\n });\n // 用于产出 TTMGODRConfig\n const TTMGODRConfig = {};\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameSize = gamesContentSize[gameId];\n if (gameSize > ONE_ODR_PKG_LIMIT) continue;\n let assignedTag = null;\n for (const odrTag of AVAILABLE_ODR_TAG_LIST) {\n const odrTagDir = path.join(TEMP_DIR, odrTag);\n if (!fs.existsSync(odrTagDir)) fs.mkdirSync(odrTagDir);\n const currentTagSize = odrTagGameMap[odrTag].reduce(\n (sum, gid) => sum + gamesContentSize[gid],\n 0,\n );\n if (currentTagSize + gameSize <= ONE_ODR_PKG_LIMIT) {\n // 复制资源\n const srcDir = path.join(TEMP_DIR, gameId);\n const destDir = path.join(odrTagDir, gameId);\n fs.cpSync(srcDir, destDir, { recursive: true });\n odrTagGameMap[odrTag].push(gameId);\n assignedTag = odrTag;\n break;\n }\n }\n if (!assignedTag) continue; // 没有分配成功就跳过\n // 构建 TTMGODRConfig\n\n TTMGODRConfig[gameId] = {\n tag: assignedTag,\n packages: {},\n };\n for (const packageName in packages) {\n const pkg = packages[packageName];\n logger.info(`TTMGODRConfig[gameId].packages, ${pkg}, ${packageName}`);\n TTMGODRConfig[gameId].packages[packageName] = {\n root: pkg.root,\n main: pkg.main,\n // code_md5: pkg.code_md5,\n asset_md5: pkg.asset_md5,\n asset_url: pkg.asset_url,\n };\n }\n }\n // 写入 JSON 文件\n const configPath = path.join(TEMP_DIR, 'TTMG_ODR_CONFIG.json');\n fs.writeFileSync(configPath, JSON.stringify(TTMGODRConfig, null, 2), 'utf-8');\n logger.info(`TTMG_ODR_CONFIG.json 已生成: ${configPath}`);\n}\n","\nimport * as fs from 'fs';\nimport { logger } from '@utils';\nimport * as path from 'path';\n\nimport { ODRPackages } from '@typings';\nimport { gameOdrPackages } from './mock';\n\nexport function fetchGames(params: { size: number; offset: number }): Promise<{\n data: Array<{\n id: string;\n packages: ODRPackages;\n }>;\n total: number;\n}> {\n const { size, offset } = params;\n const url = `https://game-odr.bytedance.net/odr/api/v1/games?size=${size}&offset=${offset}`;\n return fetch(url).then(res => res.json());\n}\n\n\n\nexport async function fetchAllGames(): Promise<Array<{\n id: string;\n packages: ODRPackages;\n }>> {\n return gameOdrPackages;\n const size = 100;\n let offset = 0;\n let total = 0;\n const games: Array<{\n id: string;\n packages: ODRPackages;\n }> = [];\n while (true) {\n const res = await fetchGames({ size, offset });\n games.push(...res.data);\n total = res.total;\n if (games.length >= total) {\n break;\n }\n offset += size;\n }\n return games;\n}\n\n\n// 同步下载并写入所有 code_url 文件\nexport function downloadAndSaveCodesSync(\n games: Array<{\n id: string;\n packages: ODRPackages;\n }>,\n tempDir: string,\n) {\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameDir = path.join(tempDir, gameId);\n if (!fs.existsSync(gameDir)) fs.mkdirSync(gameDir);\n\n for (const packageName in packages) {\n const packageInfo = packages[packageName];\n const { code_url } = packageInfo;\n // 同步下载\n const code = require('child_process')\n .execSync(`curl -s \"${code_url}\"`)\n .toString();\n logger.info(`downloadAndSaveCodesSync code, ${code_url}`);\n const fileName = `${packageName}.txt`;\n const filePath = path.join(gameDir, fileName);\n fs.writeFileSync(filePath, code);\n }\n }\n}","import { logger } from '@utils';\nimport * as path from 'path';\nimport * as fs from 'fs';\n\nimport {\n GAME_PACK_CONFIG_FILE_NAME,\n ODR_ASSET_DIR_SUFFIX,\n ODR_CODE_DIR_SUFFIX,\n} from '@constants';\n\n/**\n * 将 packageConfig packages 中 分拆的包 配置迁移到 odr_packages 中\n */\n\nexport async function writeOdrConfig(outputDir: string) {\n logger.info('writeOdrPackagesConfig start');\n const outputPath = path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME);\n const config = JSON.parse(fs.readFileSync(outputPath, 'utf-8'));\n const packages = config.packages;\n const odrPackages = config.odr_packages || {};\n for (const pkgName in packages) {\n if (pkgName.endsWith(ODR_CODE_DIR_SUFFIX)) {\n /**\n * 找到 assets 对应的包\n */\n const pkgConfig = packages[pkgName];\n const relatedPkgName = pkgName.split(ODR_CODE_DIR_SUFFIX)[0];\n if (!odrPackages[relatedPkgName]) {\n odrPackages[relatedPkgName] = {};\n }\n odrPackages[relatedPkgName] = {\n ...odrPackages[relatedPkgName],\n root: pkgConfig.root,\n code_output: pkgConfig.output,\n code_md5: pkgConfig.md5,\n main: pkgConfig.main,\n };\n delete packages[pkgName];\n }\n if (pkgName.endsWith(ODR_ASSET_DIR_SUFFIX)) {\n const relatedPkgName = pkgName.split(ODR_ASSET_DIR_SUFFIX)[0];\n const pkgConfig = packages[pkgName];\n odrPackages[relatedPkgName] = {\n ...odrPackages[relatedPkgName],\n root: pkgConfig.root,\n asset_output: pkgConfig.output,\n asset_md5: pkgConfig.md5,\n };\n delete packages[pkgName];\n }\n }\n config.odr_packages = odrPackages;\n fs.writeFileSync(outputPath, JSON.stringify(config, null, 2));\n logger.info('writeOdrPackagesConfig end');\n}\n"],"names":["ensureDirSync","dir","fs","existsSync","mkdirSync","recursive","logger","constructor","this","init","output","enableLog","winston","createLogger","level","format","combine","timestamp","json","transports","File","filename","path","join","Console","info","msg","error","warn","debug","log","setOutput","removeEmptyDir","statSync","isDirectory","isEmpty","files","readdirSync","file","fullPath","rmSync","GAME_PACK_CONFIG_FILE_NAME","GAME_PROJECT_CONFIG_FILE_NAME","GAME_ORIGIN_CONFIG_FILE_NAME","GAME_MAIN_PACKAGE_NAME","STTPKG_EXT","ODR_CODE_DIR_SUFFIX","ODR_ASSET_DIR_SUFFIX","ONE_ODR_PKG_LIMIT","AVAILABLE_ODR_TAG_LIST","PROJECT_SIZE_LIMIT","MAINPACKAGE_SIZE_LIMIT","getDirSizeSync","filterDirs","totalSize","entries","withFileTypes","entry","name","includes","isFile","size","deleteJsFilesSync","endsWith","unlinkSync","err","deleteNoJsFilesSync","uselessDirs","uselessFiles","odrPackages","__GAME__","code_url","asset_url","asset_md5","root","main","configBundle","gameBundle","mainBundle","resources","iceAndFire","pushstone","gameOdrPackages","id","Math","random","packages","checkProject","entryDir","config","projectSizeLimit","isDev","gameSize","gameSizeMB","round","limitMB","errMsg","Error","console","checkProjectSize","gameJsonPath","checkGameJson","gameJsonConfigPath","gameJsonConfig","JSON","parse","readFileSync","appid","checkProjectConfig","checkMainPackage","gameJsPath","mainPackageSizeLimit","mainPkgSize","mainPkgSizeMB","checkMainPackageSize","checkSubpackages","subpackageSizeLimit","subpackages","length","forEach","sub","subpackageEntryDir","subpackageRoot","subPkgName","limit","subPkgDir","subPkgSize","sizeMB","checkSubPackageSize","async","checkPkgs","setup","outputDir","gameJson","url","md5","configs","normalizeName","String","replace","getMainFromRoot","str","test","normalizeRoot","map","transformSubPackages","result","Object","keys","pkg","pkgPath","writeFileSync","stringify","flattenMaps","allMaps","list","values","arr","push","collectAllJsFiles","exts","excludeDirs","concat","some","ext","collectDeps","gameEntry","rootPrefixes","filter","getRootPrefixes","glob","sync","cwd","ignore","requireCalls","code","ast","acorn","ecmaVersion","sourceType","e","walkAst","node","type","callee","arguments","requirePath","value","resolvedPath","startsWith","resolve","dirname","prefix","filePath","fileExistsWithExtensions","relative","newRequireCalls","i","depModule","curModule","assign","reduce","acc","cur","f","cb","key","hasOwnProperty","child","Array","isArray","c","pack","gameOutput","pkgName","allDeps","pkgConfig","destRoot","enableSourcemap","startTime","Date","now","packMap","moduleConfig","deps","packFileName","relPaths","Bundle","separator","relPath","absPath","basename","fileId","defineHeader","defineFooter","fileMagic","MagicString","prepend","append","addSource","outFile","bundledCode","toString","generateMap","hires","includeContent","fileContent","Buffer","from","partition","pkgRoot","packedFiles","skipDirs","isRoot","srcPath","destPath","copyFileSync","makePkgs","removeUselessDirs","message","stat","force","removeSystemUseless","subRoots","split","sep","Boolean","uniqueSubRoots","Set","srcRoot","jsFiles","packNameRoot","collectMaps","rules","outputPath","pkgOutput","codeOutput","assetOutput","cpSync","code_md5","makeOdrPkgs","gamePackConfigPath","gamePackConfig","pkgDir","isSymbolicLink","linkTarget","readlinkSync","symlinkSync","mergePkgs","packageConfig","isSuccess","errorMsg","TEMP_DIR","__dirname","games","fetchAllGames","tempDir","game","gameId","gameDir","packageName","packageInfo","require","execSync","fileName","downloadAndSaveCodesSync","gamesContentSize","odrTagGameMap","tag","TTMGODRConfig","assignedTag","odrTag","odrTagDir","sum","gid","srcDir","destDir","configPath","odr_packages","relatedPkgName","code_output","asset_output"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../src/utils/ensureDirSync.ts","../src/utils/logger.ts","../src/utils/removeEmptyDir.ts","../src/constants/index.ts","../src/utils/getDirSizeSync.ts","../src/utils/deleteJsFilesSync.ts","../src/utils/deleteNoJsFilesSync.ts","../src/utils/removeSystemUseless.ts","../src/libs/odrPkgs/mock.ts","../src/libs/checkPkgs/checkProject.ts","../src/libs/checkPkgs/checkMainpackage.ts","../src/libs/checkPkgs/checkSubpackages.ts","../src/libs/checkPkgs/index.ts","../src/libs/makePkgs/setup.ts","../src/libs/makePkgs/collectMaps.ts","../src/libs/makePkgs/collectDeps.ts","../src/libs/makePkgs/pack.ts","../src/libs/makePkgs/partition.ts","../src/libs/makePkgs/index.ts","../src/libs/buildPkgs/index.ts","../src/libs/odrPkgs/makeOdrPkgs.ts","../src/libs/debugPkgs/index.ts","../src/libs/mergePkgs/index.ts","../src/libs/odrPkgs/downloadOdrPkgs.ts","../src/libs/odrPkgs/utils.ts","../src/libs/odrPkgs/writeOdrConfig.ts"],"sourcesContent":["import fs from 'fs';\nexport function ensureDirSync(dir) {\n if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true });\n}\n","import winston from 'winston';\nimport path from 'path';\n\nclass Logger {\n logger: winston.Logger;\n enableLog: boolean;\n output: any;\n constructor() {\n this.logger = null;\n }\n init(output: string, enableLog = true) {\n this.enableLog = enableLog;\n this.logger = winston.createLogger({\n level: 'info',\n format: winston.format.combine(\n winston.format.timestamp(),\n winston.format.json(), // 输出为 JSON 格式\n ),\n transports: [\n new winston.transports.File({\n filename: path.join(output, 'pack.log'),\n }),\n new winston.transports.Console(),\n ],\n });\n }\n info(msg: string) {\n if (this.enableLog) {\n this.logger.info(msg);\n }\n }\n error(msg) {\n if (this.enableLog) {\n this.logger.error(msg);\n }\n }\n warn(msg) {\n if (this.enableLog) {\n this.logger.warn(msg);\n }\n }\n debug(msg) {\n if (this.enableLog) {\n this.logger.debug(msg);\n }\n }\n log(msg) {\n this.logger.info(msg);\n }\n setOutput(output) {\n this.output = output;\n }\n}\n\nexport const logger = new Logger();\n","import fs from 'fs';\nimport path from 'path';\n\n/**\n * 移除指定目录下所有空文件夹(递归)\n * @param {string} dir 目标目录\n * @returns {boolean} 返回当前目录是否已被删除(即是否为空)\n */\nexport function removeEmptyDir(dir) {\n if (!fs.existsSync(dir)) return false;\n if (!fs.statSync(dir).isDirectory()) return false;\n\n let isEmpty = true;\n const files = fs.readdirSync(dir);\n for (const file of files) {\n const fullPath = path.join(dir, file);\n if (fs.statSync(fullPath).isDirectory()) {\n // 递归处理子目录\n const childIsEmpty = removeEmptyDir(fullPath);\n if (!childIsEmpty) isEmpty = false;\n } else {\n // 有文件,当前目录不为空\n isEmpty = false;\n }\n }\n\n // 如果当前目录为空,删除它\n if (isEmpty) {\n fs.rmSync(dir, { recursive: true });\n return true;\n }\n return false;\n}\n\n","export const GAME_PACK_CONFIG_FILE_NAME = 'packageConfig.json';\nexport const GAME_PROJECT_CONFIG_FILE_NAME = 'project.config.json';\nexport const GAME_ORIGIN_CONFIG_FILE_NAME = 'game.json';\nexport const GAME_MODULE_CONFIG_FILE_NAME = 'moduleConfig.json';\nexport const GAME_OUTPUT_CONFIG_FILE_NAME = 'output.json';\nexport const GAME_MAIN_PACKAGE_NAME = '__GAME__';\nexport const TTPKG_TAG = 'TPKG';\nexport const STTPKG_TAG = 'SPKG';\nexport const TTPKG_VERSION = 2;\nexport const ARK_KEY_LEN = 32;\nexport const ARK_INT_LEN = 4;\nexport const UNIT32_LEN = 4;\nexport const UINT16_LEN = 2;\nexport const STTPKG_VERSION = 1;\nexport const ZSTD_COMPRESS_LEVEL = 19;\nexport const ROLL_FILE_EXT = [\n '.js',\n '.json',\n '.wxml',\n '.ttml',\n '.wxss',\n '.ttss',\n];\nexport const EXT = '';\nexport const BROTLI_COMPRESS_TYPE = 1;\nexport const ZSTD_COMPRESS_TYPE = 3;\n\nexport const STTPKG_EXT = '.txt';\n\nexport const SUBPACKAGES_DIR_NAME = 'subpackages';\n\nexport const ODR_CODE_DIR_SUFFIX = '_code';\n\nexport const ODR_ASSET_DIR_SUFFIX = '_asset';\n\n// 60MB const ONE_ODR_PKG_LIMIT = 1024 * 1024;\nexport const ONE_ODR_PKG_LIMIT = 1024 * 1024 * 60;\n\nexport const AVAILABLE_ODR_TAG_LIST = [\n 'ttmg_odr_tag_1',\n 'ttmg_odr_tag_2',\n 'ttmg_odr_tag_3',\n];\n\nexport const PROJECT_SIZE_LIMIT = 30 * 1024 * 1024;\n\n\nexport const MAINPACKAGE_SIZE_LIMIT = 4 * 1024 * 1024;\n\n","import fs from 'fs';\nimport path from 'path';\n// 递归统计目录下所有文件的大小,支持我过滤某些文件夹\nexport function getDirSizeSync(dir: string, filterDirs: string[] = []) {\n let totalSize = 0;\n if (!fs.existsSync(dir)) return 0;\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n if (entry.isDirectory()) {\n if (!filterDirs.includes(entry.name)) {\n totalSize += getDirSizeSync(fullPath, filterDirs);\n }\n } else if (entry.isFile()) {\n totalSize += fs.statSync(fullPath).size;\n }\n }\n return totalSize;\n}\n","import fs from 'fs';\nimport path from 'path';\n\n/**\n * 同步删除指定目录及其子目录下所有 .js 文件\n * @param {string} dir 目标目录路径\n */\nexport function deleteJsFilesSync(dir) {\n // 读取目录内容\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n\n if (entry.isDirectory()) {\n // 递归处理子目录\n deleteJsFilesSync(fullPath);\n } else if (\n entry.isFile() &&\n (fullPath.endsWith('.js') || fullPath.endsWith('.js.map'))\n ) {\n try {\n fs.unlinkSync(fullPath);\n } catch (err) {}\n }\n }\n}\n","import fs from 'fs';\nimport path from 'path';\n\nexport function deleteNoJsFilesSync(dir) {\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n\n if (entry.isDirectory()) {\n // 递归处理子目录\n deleteNoJsFilesSync(fullPath);\n } else if (\n entry.isFile() &&\n !(fullPath.endsWith('.js') || fullPath.endsWith('.js.map'))\n ) {\n try {\n fs.unlinkSync(fullPath);\n } catch (err) {}\n }\n }\n}\n","import fs from 'fs';\nimport path from 'path';\n\nimport { logger } from './logger';\n\nconst uselessDirs = ['node_modules', '__MACOSX'];\nconst uselessFiles = ['.DS_Store', 'Thumbs.db'];\n\nexport function removeSystemUseless(gameEntry: string) {\n /**\n * 递归删除目录下的无用文件夹和无用文件\n */\n function removeUselessDirs(dir: string) {\n let files: string[];\n try {\n files = fs.readdirSync(dir);\n } catch (err) {\n // 目录不可访问,跳过\n logger.warn(`无法访问目录: ${dir}, ${err.message}`);\n return;\n }\n files.forEach(file => {\n const filePath = path.join(dir, file);\n try {\n const stat = fs.statSync(filePath);\n if (stat.isDirectory()) {\n if (uselessDirs.includes(file)) {\n // 删除无用目录\n fs.rmSync(filePath, { recursive: true, force: true });\n logger.info(`已删除无用目录: ${filePath}`);\n } else {\n // 递归处理子目录\n removeUselessDirs(filePath);\n }\n } else if (stat.isFile()) {\n if (uselessFiles.includes(file)) {\n // 删除无用文件\n fs.rmSync(filePath, { force: true });\n logger.info(`已删除无用文件: ${filePath}`);\n }\n }\n } catch (err) {\n // 某个文件/目录不可访问,跳过\n logger.warn(`无法处理: ${filePath}, ${err.message}`);\n }\n });\n }\n removeUselessDirs(gameEntry);\n}\n","interface ODRPackage {\n // md5: string;\n root: string;\n main: string;\n code_url: string;\n // code_md5: string;\n asset_md5: string;\n asset_url: string;\n}\n\ntype ODRPackages = Record<string, ODRPackage>;\nconst odrPackages: ODRPackages = {\n __GAME__: {\n // md5: '6eac4e8a7a7f54a20bcca6c67d71ebad',\n code_url: 'https://connect.tiktok-minis.com/test/__GAME___code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/__GAME___asset.txt',\n asset_md5: '8e3cc2ebd5536e9183b93fda34c1e7f4',\n root: '',\n main: 'game.js',\n },\n configBundle: {\n // md5: '575a0a1f0191e682fc27af897515740b',\n root: 'subpackages/configBundle',\n main: 'subpackages/configBundle/game.js',\n asset_md5: 'f624a1b37fb35d4dfb1975105a51ebd4',\n code_url: 'https://connect.tiktok-minis.com/test/configBundle_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/configBundle_asset.txt',\n },\n gameBundle: {\n // md5: '02e5d043be3156cfe2553e8af0f1b3d3',\n main: 'subpackages/gameBundle/game.js',\n root: 'subpackages/gameBundle',\n asset_md5: 'fcc9f686bbc8d6f983263e7132598b26',\n code_url: 'https://connect.tiktok-minis.com/test/gameBundle_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/gameBundle_asset.txt',\n },\n mainBundle: {\n // md5: 'f28aae73fbb0cc3b8876c2773a1c5789',\n main: 'subpackages/mainBundle/game.js',\n root: 'subpackages/mainBundle',\n asset_md5: '36b9674d48f9fbb0de6e8f48b194aebc',\n code_url: 'https://connect.tiktok-minis.com/test/mainBundle_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/mainBundle_asset.txt',\n },\n resources: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/resources',\n main: 'subpackages/resources/game.js',\n asset_md5: '1c4608d7690750f7bd5c00663013fc70',\n code_url: 'https://connect.tiktok-minis.com/test/resources_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/resources_asset.txt',\n },\n iceAndFire: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/iceAndFire',\n main: 'subpackages/iceAndFire/game.js',\n asset_md5: 'c7fd8e8506dd59725269027a52e2cae4',\n code_url: 'https://connect.tiktok-minis.com/test/iceAndFire_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/iceAndFire_asset.txt',\n },\n pushstone: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/pushstone',\n main: 'subpackages/pushstone/game.js',\n asset_md5: 'f5061dba94ad53e1f578325a7cb26a01',\n code_url: 'https://connect.tiktok-minis.com/test/pushstone_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/pushstone_asset.txt',\n },\n main: {\n // md5: '238c98f80a43d0603ac4a371c1b6184b',\n root: 'subpackages/main',\n main: 'subpackages/main/game.js',\n asset_md5: 'aece2f69cc168d7cefb248e8574c3371',\n code_url: 'https://connect.tiktok-minis.com/test/main_code.txt',\n asset_url: 'https://connect.tiktok-minis.com/test/main_asset.txt',\n },\n};\n\nexport const gameOdrPackages = [\n { id: `${Math.random() * 100000000000000000}`, packages: odrPackages },\n { id: `${Math.random() * 100000000000000000}`, packages: odrPackages },\n];\n","import path from 'path';\nimport fs from 'fs';\nimport { CheckConfig } from '@typings';\nimport {\n GAME_ORIGIN_CONFIG_FILE_NAME,\n GAME_PROJECT_CONFIG_FILE_NAME,\n} from '@constants';\nimport { logger, getDirSizeSync } from '@utils';\n\nexport function checkGameJson(entryDir: string) {\n logger.info('start check config');\n const gameJsonPath = path.join(entryDir, GAME_ORIGIN_CONFIG_FILE_NAME);\n if (!fs.existsSync(gameJsonPath)) {\n const errMsg =\n 'can not find game.json in game source code, please check it';\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n logger.info('check config successfully');\n }\n}\n\n/**\n * 检查游戏大小是否超出限制\n * @param param0\n * @returns\n */\n\nexport function checkProjectSize(entryDir: string, config: CheckConfig) {\n logger.info('start check project size');\n const { projectSizeLimit, isDev } = config;\n const gameSize = getDirSizeSync(entryDir, []);\n const gameSizeMB = Math.round(gameSize / (1024 * 1024));\n const limitMB = Math.round(projectSizeLimit / (1024 * 1024));\n if (gameSize > projectSizeLimit) {\n const errMsg = `Game size ${gameSizeMB}MB, must not exceed ${limitMB}MB`;\n logger.error(errMsg);\n if (!isDev) {\n throw new Error(errMsg);\n } else {\n console.log('\\x1b[1m\\x1b[33m%s\\x1b[0m', errMsg);\n }\n } else {\n /**\n * 游戏大小检查通过\n */\n logger.info(`Game size ${gameSizeMB}MB, check successfully!`);\n logger.info('Game size check successfully!');\n }\n}\n\nexport function checkProject(entryDir: string, config: CheckConfig) {\n logger.info('start check project');\n checkProjectSize(entryDir, config);\n checkGameJson(entryDir);\n checkProjectConfig(entryDir);\n logger.info('check project successfully');\n}\n\nexport function checkProjectConfig(entryDir: string) {\n logger.info('start check project config');\n const gameJsonConfigPath = path.join(entryDir, GAME_PROJECT_CONFIG_FILE_NAME);\n if (!fs.existsSync(gameJsonConfigPath)) {\n const errMsg =\n 'can not find project.config.json in game source code, please check it';\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n logger.info('check project.config.json successfully');\n }\n let gameJsonConfig: any;\n try {\n gameJsonConfig = JSON.parse(fs.readFileSync(gameJsonConfigPath, 'utf-8'));\n } catch (error) {\n logger.error(\n 'check project.config.json failed, your project.config.json is invalid json format',\n );\n throw error;\n }\n const appid = gameJsonConfig?.appid;\n if (!appid) {\n const errMsg =\n 'can not find appid in project.config.json, you should add appid(client_key) in project.config.json';\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n}\n","import * as path from 'path';\nimport * as fs from 'fs';\nimport { CheckConfig } from '@typings';\nimport { logger, getDirSizeSync } from '@utils';\n/**\n * 检查主包大小是否超出限制\n * @param param0\n * @returns\n */\nexport function checkMainPackageSize({\n entryDir,\n config,\n}: {\n entryDir: string;\n config: CheckConfig;\n}) {\n logger.info('start check main package size');\n const { mainPackageSizeLimit, isDev } = config;\n const mainPkgSize = getDirSizeSync(entryDir, ['subpackages']);\n const mainPkgSizeMB = Math.round(mainPkgSize / (1024 * 1024));\n const limitMB = Math.round(mainPackageSizeLimit / (1024 * 1024));\n if (mainPkgSize > mainPackageSizeLimit) {\n const errMsg = `Main package size ${mainPkgSizeMB}MB, must not exceed ${limitMB}MB`;\n if (!isDev) {\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n console.log('\\x1b[1m\\x1b[33m%s\\x1b[0m', errMsg);\n }\n } else {\n /**\n * 主包大小检查通过\n */\n logger.info(`Main package size ${mainPkgSizeMB}MB, check successfully!`);\n }\n}\n\nexport function checkMainPackage(entryDir: string, config: CheckConfig) {\n logger.info('start check main package');\n /**\n * 检查 game.js 是否存在\n */\n\n logger.info('check game.js');\n const gameJsPath = path.join(entryDir, 'game.js');\n if (!fs.existsSync(gameJsPath)) {\n const errMsg = 'game.js must exist in main package!';\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n logger.info('check game.js successfully');\n\n /**\n * 检查主包大小是否超出限制\n */\n checkMainPackageSize({\n entryDir,\n config,\n });\n logger.info('check main package successfully');\n}\n","import { logger, getDirSizeSync } from '@utils';\nimport { CheckConfig } from '@typings';\nimport { GAME_ORIGIN_CONFIG_FILE_NAME } from '@constants';\nimport path from 'path';\nimport fs from 'fs';\n\nexport function checkSubpackages(\n entryDir: string,\n config?: Pick<CheckConfig, 'subpackageSizeLimit'>,\n) {\n const { subpackageSizeLimit } = config || {};\n /**\n * 开始校验 subpackages 配置是否有效\n */\n logger.info('start check subpackages in game.json');\n const gameJsonPath = path.join(entryDir, GAME_ORIGIN_CONFIG_FILE_NAME);\n const gameJson = JSON.parse(fs.readFileSync(gameJsonPath, 'utf-8'));\n const subpackages: Array<{\n name: string;\n root: string;\n }> = gameJson.subpackages || [];\n if (!subpackages?.length) {\n logger.info('subpackages is empty');\n } else {\n subpackages.forEach(sub => {\n /**\n * 校验 subpackage 配置是否为空\n */\n if (!sub.name || !sub.root) {\n const errMsg =\n 'The subpackages configuration in game.json is invalid: the root or name field in one or more subpackages is empty. Please ensure that all subpackages have non-empty root and name values.';\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n /**\n * 校验 subpackage 配置是否有效\n */\n const subpackageEntryDir = path.join(entryDir, sub.root);\n // 检查是文件或者文件夹是否存在\n if (!fs.existsSync(subpackageEntryDir)) {\n const errMsg = `The subpackages configuration in game.json is invalid: can not find subpackage ${sub.name} entry dir, please check it`;\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n const subpackageRoot = sub.root;\n if (!fs.existsSync(path.join(entryDir, subpackageRoot))) {\n const errMsg = `The subpackages configuration in game.json is invalid: can not find subpackage ${sub.name} root, please check it`;\n logger.error(errMsg);\n throw new Error(errMsg);\n }\n /**\n * 校验 subpackage 包大小\n */\n\n if (subpackageSizeLimit) {\n checkSubPackageSize({\n entryDir,\n subPkgName: sub.name,\n limit: subpackageSizeLimit,\n });\n }\n });\n }\n logger.info('check subpackages in game.json successfully');\n}\n\n/**\n * 检查子包大小是否超出限制\n * @param param0\n * @returns\n */\n\nfunction checkSubPackageSize({\n entryDir,\n subPkgName,\n limit,\n}: {\n entryDir: string;\n subPkgName: string;\n limit: number;\n}) {\n const subPkgDir = path.join(entryDir, 'subpackages', subPkgName);\n const subPkgSize = getDirSizeSync(subPkgDir, []);\n const sizeMB = Math.round(subPkgSize / (1024 * 1024));\n if (sizeMB > limit) {\n const errMsg = `${subPkgName} Sub package size ${sizeMB}MB, must not exceed ${\n limit / (1024 * 1024)\n }MB`;\n logger.error(errMsg);\n throw new Error(errMsg);\n } else {\n /**\n * 子包大小检查通过\n */\n logger.info(\n `${subPkgName} Sub package size ${sizeMB}MB, check successfully!`,\n );\n }\n}\n","import { logger } from '@utils';\nimport { CheckConfig } from '@typings';\nimport { checkProject } from './checkProject';\nimport { checkMainPackage } from './checkMainpackage';\nimport { checkSubpackages } from './checkSubpackages';\n\nexport async function checkPkgs({\n entryDir,\n config,\n}: {\n entryDir: string;\n config: CheckConfig;\n}) {\n logger.info('开始校验项目');\n /**\n * 1. 校验项目\n */\n checkProject(entryDir, config);\n /**\n * 2. 校验主包\n */\n checkMainPackage(entryDir, config);\n /**\n * 3. 如果有,校验分包\n */\n checkSubpackages(entryDir, config);\n logger.info('校验项目完成');\n}\n","import fs from 'fs';\nimport path from 'path';\nimport {\n GAME_PACK_CONFIG_FILE_NAME,\n GAME_ORIGIN_CONFIG_FILE_NAME,\n GAME_MAIN_PACKAGE_NAME,\n STTPKG_EXT,\n} from '@constants';\n\nimport { logger } from '@utils';\n\nexport async function setup(entryDir: string, outputDir: string) {\n logger.info('开始基于源代码中的 game.json 生成 packageConfig.json');\n const gameJsonPath = path.join(entryDir, GAME_ORIGIN_CONFIG_FILE_NAME);\n const gameJson = JSON.parse(fs.readFileSync(gameJsonPath, 'utf-8'));\n\n const packages = {};\n // 主包\n packages[GAME_MAIN_PACKAGE_NAME] = {\n url: '',\n md5: '',\n root: '',\n main: 'game.js',\n output: `${GAME_MAIN_PACKAGE_NAME}${STTPKG_EXT}`,\n };\n\n transformSubPackages(gameJson.subpackages).forEach(({ name, root, main }) => {\n // root 是目录\n packages[name] = {\n url: '',\n md5: '',\n root,\n main,\n output: `${name}${STTPKG_EXT}`,\n };\n });\n\n const result = {\n packages,\n };\n\n /**\n * 校验子包的 main 文件是否存在\n */\n Object.keys(packages).forEach(pkg => {\n const pkgPath = path.join(entryDir, packages[pkg].main);\n if (!fs.existsSync(pkgPath)) {\n const errMsg = `开发者源代码包中的子包 ${pkg} 中的 main 文件 ${packages[pkg].main} 不存在`;\n logger.error(errMsg);\n\n throw new Error(errMsg);\n }\n });\n\n if (fs.existsSync(outputDir)) {\n fs.rmSync(outputDir, { recursive: true });\n }\n fs.mkdirSync(outputDir);\n fs.writeFileSync(\n path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME),\n JSON.stringify(result, null, 2),\n );\n logger.info('基于源代码中的 game.json 生成 packageConfig.json 成功');\n return result;\n}\n\nfunction transformSubPackages(\n configs: Array<{\n name: string;\n root: string;\n }>,\n) {\n if (!configs || configs.length === 0) {\n return [];\n }\n function normalizeName(name) {\n return String(name).replace(/^\\/+|\\/+$/g, '');\n }\n\n function getMainFromRoot(root: string) {\n if (isDirectory(root)) {\n return path.join(root, 'game.js');\n }\n return root;\n }\n\n function isDirectory(str) {\n // 以 / 结尾,或没有 .js 结尾且没有 . 号\n return str.endsWith('/') || !/\\.[a-zA-Z0-9]+$/.test(str);\n }\n\n function normalizeRoot(root: string): string {\n /**\n * 只保留 subpackages/xxx 两层\n * 例如:\n * 'subpackages/moduleA/pages/index' -> 'subpackages/moduleA'\n * 'subpackages/moduleB' -> 'subpackages/moduleB'\n * 'subpackages/moduleC/' -> 'subpackages/moduleC'\n */\n // 将捕获组从匹配两层目录 ([^/]+\\/[^/]+) 改为只匹配一层 ([^/]+)\n return root.replace(/^subpackages\\/([^/]+).*$/, 'subpackages/$1');\n }\n\n return configs.map(config => {\n const name = normalizeName(config.name);\n let main = getMainFromRoot(config.root);\n // test\n let root = normalizeRoot(config.root);\n return { main, root, name };\n });\n}\n","import fs from 'fs';\nimport path from 'path';\nimport { logger } from '@utils';\n\nexport function flattenMaps(allMaps) {\n const list = [];\n Object.values(allMaps).forEach(map => {\n Object.values(map).forEach(arr => {\n list.push(...arr);\n });\n });\n return list;\n}\n\n// 递归收集 JS 文件,但可排除指定目录\nfunction collectAllJsFiles(\n dir,\n exts = ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs = [],\n) {\n let files = [];\n if (!fs.existsSync(dir)) return files;\n const entries = fs.readdirSync(dir, { withFileTypes: true });\n for (const entry of entries) {\n const fullPath = path.join(dir, entry.name);\n if (entry.isDirectory()) {\n if (excludeDirs.includes(entry.name)) continue; // 跳过排除目录\n files = files.concat(collectAllJsFiles(fullPath, exts, [])); // 只排除一级\n } else if (exts.some(ext => entry.name.endsWith(ext))) {\n files.push(fullPath);\n }\n }\n return files;\n}\n\n// 构建每个包的 map\nexport function collectMaps(\n entryDir,\n packages: Record<string, { root: string }>,\n) {\n logger.info('开始基于游戏源代码收集分包中的 JS 文件');\n const result = {};\n // 1. 先收集所有分包根目录名(如 subpackages/level1、subpackages/level2)\n const subRoots = Object.values(packages)\n .filter(pkg => pkg.root && pkg.root !== '')\n .map(pkg => pkg.root.split(path.sep)[0]) // 只提取一级目录名\n .filter(Boolean);\n const uniqueSubRoots = Array.from(new Set(subRoots));\n\n for (const pkgName in packages) {\n const pkg = packages[pkgName];\n let srcRoot;\n let jsFiles;\n if (pkgName === '__GAME__') {\n // 主包:只收集 entryDir 下非分包目录\n jsFiles = collectAllJsFiles(\n entryDir,\n ['.js', '.ts', '.jsx', '.tsx'],\n uniqueSubRoots,\n );\n srcRoot = entryDir;\n } else {\n // 分包:只收集自己目录\n srcRoot = path.join(entryDir, pkg.root);\n jsFiles = collectAllJsFiles(srcRoot);\n }\n // 以包的 srcRoot 为基准生成相对路径\n const list = [];\n jsFiles.forEach(absPath => {\n list.push(path.relative(entryDir, absPath).replace(/\\\\/g, '/'));\n });\n const packNameRoot = packages[pkgName].root;\n result[pkgName] = {\n [packNameRoot ? `${packNameRoot}/game.pack.js` : 'game.pack.js']: list,\n };\n }\n logger.info('基于游戏源代码收集分包中的 JS 文件完成');\n return result;\n}\n","import fs from 'fs';\nimport path from 'path';\nimport acorn from 'acorn';\nimport glob from 'glob';\n\nexport function collectDeps(gameEntry, packages) {\n // 自动扫描 game 目录下的一级文件夹作为根前缀\n const rootPrefixes = getRootPrefixes(gameEntry);\n\n // 查找所有 JS 文件\n const files = glob.sync('**/*.js', {\n cwd: gameEntry,\n ignore: []\n });\n\n const requireCalls = [];\n\n files.forEach(file => {\n const fullPath = path.join(gameEntry, file);\n const code = fs.readFileSync(fullPath, 'utf-8');\n\n let ast;\n try {\n ast = acorn.parse(code, { ecmaVersion: 'latest', sourceType: 'module' });\n } catch (e) {\n // 跳过无法解析的文件\n return;\n }\n\n walkAst(ast, node => {\n // require('xxx') 或 requireAsync('xxx')\n if (\n node.type === 'CallExpression' &&\n node.callee.type === 'Identifier' &&\n (node.callee.name === 'require' ||\n node.callee.name === 'requireAsync') &&\n node.arguments.length > 0 &&\n node.arguments[0].type === 'Literal'\n ) {\n const requirePath = node.arguments[0].value;\n let resolvedPath = null;\n\n if (requirePath.startsWith('.')) {\n resolvedPath = path.resolve(path.dirname(fullPath), requirePath);\n } else if (requirePath.startsWith('/')) {\n resolvedPath = path.resolve(gameEntry, '.' + requirePath);\n } else {\n for (const prefix of rootPrefixes) {\n if (requirePath.startsWith(prefix)) {\n resolvedPath = path.resolve(gameEntry, requirePath);\n break;\n }\n }\n }\n\n // 校验文件是否存在(支持 .js/.ts/.jsx/.tsx 后缀自动补全)\n if (resolvedPath && !fileExistsWithExtensions(resolvedPath)) {\n resolvedPath = null;\n }\n\n requireCalls.push({\n file: path.relative(gameEntry, fullPath),\n callee: node.callee.name,\n requirePath,\n resolvedPath: resolvedPath\n ? path.relative(gameEntry, resolvedPath)\n : '',\n });\n }\n });\n });\n\n const newRequireCalls = requireCalls.map(i => {\n /**\n * 基于 packages 来判断\n */\n let depModule = '';\n Object.keys(packages).forEach(pkg => {\n if (i.resolvedPath && i.resolvedPath.startsWith(packages[pkg].root)) {\n depModule = pkg;\n }\n });\n\n let curModule = '';\n Object.keys(packages).forEach(pkg => {\n if (i.file.startsWith(packages[pkg].root)) {\n curModule = pkg;\n }\n });\n return {\n ...i,\n curModule,\n depModule,\n };\n });\n\n // 只统计 JS/TS/JSX/TSX 文件依赖\n const exts = ['.js', '.ts', '.jsx', '.tsx'];\n const isJsLike = f => exts.some(ext => f.endsWith(ext));\n\n const result = newRequireCalls.reduce((acc, cur) => {\n if (!acc[cur.curModule]) {\n acc[cur.curModule] = {};\n }\n if (cur.curModule === cur.depModule) {\n return acc;\n }\n if (cur.resolvedPath && isJsLike(cur.resolvedPath)) {\n acc[cur.curModule] = {\n ...acc[cur.curModule],\n [cur.resolvedPath]: cur.depModule,\n };\n }\n return acc;\n }, {});\n return result;\n}\n\nfunction getRootPrefixes(gameEntry) {\n return fs\n .readdirSync(gameEntry)\n .filter(name => {\n const fullPath = path.join(gameEntry, name);\n return (\n fs.statSync(fullPath).isDirectory() &&\n !['node_modules', 'dist', 'build', 'output'].includes(name)\n );\n })\n .map(name => name + '/');\n}\n\nfunction walkAst(node, cb) {\n cb(node);\n for (let key in node) {\n if (node.hasOwnProperty(key)) {\n const child = node[key];\n if (Array.isArray(child)) {\n child.forEach(c => c && typeof c.type === 'string' && walkAst(c, cb));\n } else if (child && typeof child.type === 'string') {\n walkAst(child, cb);\n }\n }\n }\n}\n\n// 支持 .js/.ts/.jsx/.tsx 自动补全\nfunction fileExistsWithExtensions(filePath) {\n if (fs.existsSync(filePath) && fs.statSync(filePath).isFile()) return true;\n const exts = ['.js', '.ts', '.jsx', '.tsx'];\n for (const ext of exts) {\n if (fs.existsSync(filePath + ext) && fs.statSync(filePath + ext).isFile())\n return true;\n }\n return false;\n}\n","import fs from 'fs';\nimport path from 'path';\nimport { GAME_MODULE_CONFIG_FILE_NAME } from '@constants';\nimport { logger, ensureDirSync } from '@utils';\nimport MagicString, { Bundle } from 'magic-string';\n\nexport function pack({\n gameEntry,\n gameOutput,\n pkgName,\n allDeps,\n allMaps,\n pkgConfig,\n destRoot,\n enableSourcemap,\n}: {\n gameOutput: string;\n gameEntry: string;\n pkgName: string;\n allDeps: Record<string, string[]>;\n allMaps: Record<string, Record<string, string[]>>;\n pkgConfig: {\n main: string;\n root: string;\n };\n destRoot: string;\n enableSourcemap?: boolean;\n}) {\n let startTime = Date.now();\n logger.info(`pack start,startTime:${startTime}`);\n const packMap = allMaps[pkgName];\n const moduleConfig = {\n main: pkgConfig.main,\n deps: allDeps[pkgName] || {},\n map: allMaps[pkgName],\n };\n\n ensureDirSync(destRoot);\n fs.writeFileSync(\n path.join(destRoot, GAME_MODULE_CONFIG_FILE_NAME),\n JSON.stringify(moduleConfig),\n );\n\n for (const packFileName in packMap) {\n const relPaths = packMap[packFileName];\n if (!Array.isArray(relPaths) || relPaths.length === 0) continue;\n\n const concat = new Bundle({ separator: '\\n' });\n for (const relPath of relPaths) {\n const absPath = path.join(gameEntry, relPath);\n if (!fs.existsSync(absPath)) continue;\n if (path.basename(absPath) === packFileName) continue;\n\n /**\n * 只匹配 js 文件\n */\n if (/\\.js$/i.test(relPath)) {\n const code = fs.readFileSync(absPath, 'utf-8');\n const fileId = relPath.replace(/\\\\/g, '/');\n const defineHeader = `define(\"game:${fileId}\", [\"require\", \"requireAsync\", \"module\", \"exports\", \"sandboxGlobal\"], function(require, requireAsync, module, exports, sandboxGlobal){\\nwith(sandboxGlobal){\\n`;\n const defineFooter = `\\n}\\n});\\n`;\n\n const fileMagic = new MagicString(code, { filename: fileId });\n const wrapped = new MagicString(defineHeader)\n .appendLeft(defineHeader.length, fileMagic.toString())\n .append(defineFooter);\n concat.addSource({ content: wrapped, filename: fileId });\n // fileMagic.prepend(defineHeader);\n // fileMagic.append(defineFooter);\n // concat.addSource(fileMagic);\n }\n }\n\n const outFile = path.join(destRoot, path.basename(packFileName));\n const bundledCode = concat.toString();\n\n if (enableSourcemap) {\n const map = concat.generateMap({\n hires: true,\n includeContent: true,\n file: packFileName,\n });\n\n const mapBase64 = Buffer.from(map.toString(), 'utf-8').toString('base64');\n const sourceMapDataURI = `data:application/json;charset=utf-8;base64,${mapBase64}`;\n\n const fileContent =\n bundledCode +\n `\\n//# sourceMappingURL=${sourceMapDataURI}` +\n `\\n//# sourceURL=${packFileName}`;\n\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n } else {\n const fileContent = bundledCode + `\\n//# sourceURL=${packFileName}`;\n\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n }\n\n logger.info(`pack end,packName:${pkgName}`);\n }\n}\n","import path from 'path';\nimport fs from 'fs';\nimport { ensureDirSync } from '@utils';\n\nexport function partition({\n pkgRoot,\n destRoot,\n packedFiles,\n gameEntry,\n gameOutput,\n skipDirs = [],\n isRoot = true\n}) {\n if (!fs.existsSync(pkgRoot)) return;\n const entries = fs.readdirSync(pkgRoot, { withFileTypes: true });\n for (const entry of entries) {\n const srcPath = path.join(pkgRoot, entry.name);\n const destPath = path.join(destRoot, entry.name);\n // 只在第一层判断是否跳过分包目录\n if (isRoot && entry.isDirectory() && skipDirs.includes(entry.name)) {\n continue; // 跳过分包目录\n }\n\n if (entry.isDirectory()) {\n partition({\n pkgRoot: srcPath,\n destRoot: destPath,\n gameOutput,\n packedFiles,\n gameEntry,\n skipDirs,\n isRoot: false, // 递归下去不再判断\n });\n } else {\n ensureDirSync(path.dirname(destPath));\n if (!packedFiles.includes(path.relative(gameEntry, srcPath))) {\n fs.copyFileSync(srcPath, destPath);\n }\n }\n }\n}\n","import fs from 'fs';\nimport path from 'path';\nimport { SUBPACKAGES_DIR_NAME } from '@constants';\nimport { logger, removeEmptyDir, removeSystemUseless } from '@utils';\nimport { setup } from './setup';\nimport { flattenMaps, collectMaps } from './collectMaps';\nimport { collectDeps } from './collectDeps';\nimport { pack } from './pack';\nimport { partition } from './partition';\nexport async function makePkgs({ gameEntry, gameOutput, enableSourcemap }) {\n let startTime = Date.now();\n removeSystemUseless(gameEntry);\n logger.info(`pack start,startTime:${startTime}`);\n const { packages } = await setup(gameEntry, gameOutput);\n const allDeps = collectDeps(gameEntry, packages);\n const allMaps = collectMaps(gameEntry, packages);\n const pkgOutput = {};\n\n for (const pkgName in allMaps) {\n logger.info(`开始基于游戏源代码打包,包名:${pkgName}`);\n let startTime = Date.now();\n const pkgConfig = packages[pkgName];\n const pkgRoot = path.join(gameEntry, pkgConfig.root);\n const destRoot = path.join(gameOutput, pkgName, pkgConfig.root);\n /**\n * 打包\n */\n pack({\n gameEntry,\n gameOutput,\n allDeps,\n allMaps,\n pkgName,\n pkgConfig,\n destRoot,\n enableSourcemap,\n });\n\n /**\n * 分包\n */\n partition({\n gameEntry,\n gameOutput,\n pkgRoot,\n destRoot,\n packedFiles: flattenMaps(allMaps),\n skipDirs: [SUBPACKAGES_DIR_NAME],\n });\n\n const pkgRootDir = path.join(gameOutput, pkgName);\n removeEmptyDir(pkgRootDir);\n logger.info(`基于游戏源代码打包分包完成,包名:${pkgName}`);\n logger.info(\n `基于游戏源代码打包分包耗时,包名:${pkgName},耗时:${\n Date.now() - startTime\n }ms`,\n );\n }\n // splitOdrPackages(gameOutput);\n logger.info(`pack end,duration:${Date.now() - startTime}ms`);\n\n return pkgOutput;\n}\n","import { logger } from '@utils';\nimport { PROJECT_SIZE_LIMIT, MAINPACKAGE_SIZE_LIMIT } from '@constants';\nimport { BuildConfig } from '@typings';\nimport { checkPkgs } from '../checkPkgs';\nimport { makePkgs } from '../makePkgs';\nimport { makeOdrPkgs } from '../odrPkgs';\n\nexport async function buildPkgs(config: BuildConfig) {\n const { entry: entryDir, output: outputDir, rules } = config;\n let startTime = Date.now();\n logger.init(outputDir, true);\n logger.info(`TTMG_PACK_VERSION: ${process.env.TTMG_PACK_VERSION}`);\n logger.info(`pack start, startTime:${startTime}`);\n\n /**\n * 校验\n */\n await checkPkgs({\n entryDir,\n config: {\n projectSizeLimit: rules?.projectSizeLimit || PROJECT_SIZE_LIMIT,\n mainPackageSizeLimit:\n rules?.mainPackageSizeLimit || MAINPACKAGE_SIZE_LIMIT,\n subpackageSizeLimit: rules?.subpackageSizeLimit,\n },\n });\n /**\n * 打包\n */\n await makePkgs({\n gameEntry: entryDir,\n gameOutput: outputDir,\n enableSourcemap: false,\n });\n\n /**\n * 分拆 odr 包\n */\n await makeOdrPkgs(outputDir);\n\n logger.info(`pack end:${Date.now() - startTime}ms`);\n}\n","import path from 'path';\nimport fs from 'fs';\nimport {\n ensureDirSync,\n deleteNoJsFilesSync,\n logger,\n deleteJsFilesSync,\n} from '@utils';\n\nimport {\n GAME_PACK_CONFIG_FILE_NAME,\n ODR_ASSET_DIR_SUFFIX,\n ODR_CODE_DIR_SUFFIX,\n STTPKG_EXT,\n} from '@constants';\n\nexport async function makeOdrPkgs(outputDir: string) {\n let startTime = Date.now();\n logger.info(`makeOdrPkgs start, ${startTime}`);\n const outputPath = path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME);\n const config = JSON.parse(fs.readFileSync(outputPath, 'utf-8'));\n const packages = config.packages;\n for (const pkgName in packages) {\n /**\n * 将原本的分包配置,转换为 odr 的配置,\n * 将原本产物按照原路径,分别拆分成两个文件夹:js 部分 package_name_code,asset 部分 package_name_asset\n */\n const pkgOutput = path.join(outputDir, pkgName);\n const codeOutput = path.join(outputDir, `${pkgName}${ODR_CODE_DIR_SUFFIX}`);\n const assetOutput = path.join(\n outputDir,\n `${pkgName}${ODR_ASSET_DIR_SUFFIX}`,\n );\n ensureDirSync(codeOutput);\n ensureDirSync(assetOutput);\n fs.cpSync(pkgOutput, codeOutput, { recursive: true });\n fs.cpSync(pkgOutput, assetOutput, { recursive: true });\n deleteNoJsFilesSync(codeOutput);\n deleteJsFilesSync(assetOutput);\n\n /**\n * 写入配置\n */\n packages[`${pkgName}${ODR_CODE_DIR_SUFFIX}`] = {\n ...packages[pkgName],\n code_url: '',\n code_md5: '',\n output: `${pkgName}${ODR_CODE_DIR_SUFFIX}${STTPKG_EXT}`,\n };\n packages[`${pkgName}${ODR_ASSET_DIR_SUFFIX}`] = {\n ...packages[pkgName],\n asset_url: '',\n asset_md5: '',\n output: `${pkgName}${ODR_ASSET_DIR_SUFFIX}${STTPKG_EXT}`,\n };\n }\n fs.writeFileSync(outputPath, JSON.stringify(config, null, 2));\n logger.info(`makeOdrPkgs end, ${Date.now() - startTime}ms`);\n}\n","import fs from 'fs';\nimport path from 'path';\n\nimport { logger } from '@utils';\nimport {\n PROJECT_SIZE_LIMIT,\n MAINPACKAGE_SIZE_LIMIT,\n GAME_PACK_CONFIG_FILE_NAME,\n} from '@constants';\nimport { BuildConfig } from '@typings';\nimport { checkPkgs } from '../checkPkgs';\nimport { mergePkgs } from '../mergePkgs';\nimport { makePkgs } from '../makePkgs';\n\nexport async function debugPkgs(config: BuildConfig): Promise<{\n isSuccess: boolean;\n errorMsg?: string;\n packages?: any;\n}> {\n const {\n entry: entryDir,\n output: outputDir,\n rules,\n enableLog = false,\n } = config;\n logger.init(outputDir, enableLog);\n try {\n /**\n * 校验\n */\n await checkPkgs({\n entryDir,\n config: {\n projectSizeLimit: rules?.projectSizeLimit || PROJECT_SIZE_LIMIT,\n mainPackageSizeLimit:\n rules?.mainPackageSizeLimit || MAINPACKAGE_SIZE_LIMIT,\n isDev: true,\n },\n });\n /**\n * 打包\n */\n await makePkgs({\n gameEntry: entryDir,\n gameOutput: outputDir,\n enableSourcemap: true,\n });\n\n /**\n * 合并\n */\n await mergePkgs(outputDir);\n\n /**\n * 返回 packageConfig\n */\n const packageConfig = fs.readFileSync(\n path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME),\n 'utf-8',\n );\n const packageConfigJson = JSON.parse(packageConfig);\n return {\n isSuccess: true,\n packages: packageConfigJson.packages,\n };\n } catch (error) {\n return {\n isSuccess: false,\n errorMsg: error.message,\n };\n }\n}\n","import path from 'path';\nimport fs from 'fs';\nimport { logger } from '@utils';\nimport { ProjectConfig } from '@typings';\nimport { GAME_PACK_CONFIG_FILE_NAME } from '@constants';\n\n/**\n * 将 packages 中的每个 package 下的文件内容合并到 outputDir 下,不保留 package name\n * @param outputDir 输出目录\n */\nexport function mergePkgs(outputDir: string) {\n logger.info('mergePkgs start');\n const gamePackConfigPath = path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME);\n let gamePackConfig: ProjectConfig;\n try {\n gamePackConfig = JSON.parse(fs.readFileSync(gamePackConfigPath, 'utf-8'));\n } catch (err) {\n logger.error(\n `读取配置文件失败: ${gamePackConfigPath}, err: ${err.message}`,\n );\n return;\n }\n\n const packages = gamePackConfig.packages;\n if (!packages || typeof packages !== 'object') {\n logger.error('配置文件中 packages 字段无效, 请检查配置');\n return;\n }\n\n for (const pkgName of Object.keys(packages)) {\n const pkgDir = path.join(outputDir, pkgName);\n if (!fs.existsSync(pkgDir)) {\n logger.warn(`package 目录不存在: ${pkgDir}`);\n continue;\n }\n\n const files = fs.readdirSync(pkgDir);\n for (const file of files) {\n const srcPath = path.join(pkgDir, file);\n const destPath = path.join(outputDir, file);\n\n try {\n const stat = fs.statSync(srcPath);\n if (stat.isFile()) {\n fs.copyFileSync(srcPath, destPath);\n } else if (stat.isDirectory()) {\n fs.cpSync(srcPath, destPath, { recursive: true });\n } else if (stat.isSymbolicLink()) {\n const linkTarget = fs.readlinkSync(srcPath);\n // 判断目标是否已存在符号链接\n if (fs.existsSync(destPath)) {\n fs.rmSync(destPath, { force: true });\n }\n fs.symlinkSync(\n linkTarget,\n destPath,\n stat.isDirectory() ? 'dir' : 'file',\n );\n }\n logger.info(`合并文件/目录: ${srcPath} -> ${destPath}`);\n } catch (err) {\n logger.error(`处理文件失败: ${srcPath}, err: ${err.message}`);\n }\n }\n\n // 删除 package 目录\n try {\n fs.rmSync(pkgDir, { recursive: true, force: true });\n logger.info(`已删除 package 目录: ${pkgDir}`);\n } catch (err) {\n logger.error(`删除 package 目录失败: ${pkgDir}, err: ${err.message}`);\n }\n }\n\n logger.info('mergePkgs 完成');\n}\n","/**\n * 合并请求拆包\n */\nimport fs from 'fs';\nimport path from 'path';\nimport { getDirSizeSync, logger } from '@utils';\nimport { fetchAllGames, downloadAndSaveCodesSync } from './utils';\nimport { AVAILABLE_ODR_TAG_LIST, ONE_ODR_PKG_LIMIT } from '@constants';\n\nexport async function downloadOdrPkgs() {\n const TEMP_DIR = path.join(__dirname, 'temp');\n if (!fs.existsSync(TEMP_DIR)) fs.mkdirSync(TEMP_DIR);\n const games = await fetchAllGames();\n logger.info('odr_games');\n downloadAndSaveCodesSync(games, TEMP_DIR);\n const gamesContentSize = {};\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameDir = path.join(TEMP_DIR, gameId);\n gamesContentSize[gameId] = getDirSizeSync(gameDir);\n }\n // ODR TAG 分配\n const odrTagGameMap = {};\n AVAILABLE_ODR_TAG_LIST.forEach(tag => {\n odrTagGameMap[tag] = [];\n });\n // 用于产出 TTMGODRConfig\n const TTMGODRConfig = {};\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameSize = gamesContentSize[gameId];\n if (gameSize > ONE_ODR_PKG_LIMIT) continue;\n let assignedTag = null;\n for (const odrTag of AVAILABLE_ODR_TAG_LIST) {\n const odrTagDir = path.join(TEMP_DIR, odrTag);\n if (!fs.existsSync(odrTagDir)) fs.mkdirSync(odrTagDir);\n const currentTagSize = odrTagGameMap[odrTag].reduce(\n (sum, gid) => sum + gamesContentSize[gid],\n 0,\n );\n if (currentTagSize + gameSize <= ONE_ODR_PKG_LIMIT) {\n // 复制资源\n const srcDir = path.join(TEMP_DIR, gameId);\n const destDir = path.join(odrTagDir, gameId);\n fs.cpSync(srcDir, destDir, { recursive: true });\n odrTagGameMap[odrTag].push(gameId);\n assignedTag = odrTag;\n break;\n }\n }\n if (!assignedTag) continue; // 没有分配成功就跳过\n // 构建 TTMGODRConfig\n\n TTMGODRConfig[gameId] = {\n tag: assignedTag,\n packages: {},\n };\n for (const packageName in packages) {\n const pkg = packages[packageName];\n logger.info(`TTMGODRConfig[gameId].packages, ${pkg}, ${packageName}`);\n TTMGODRConfig[gameId].packages[packageName] = {\n root: pkg.root,\n main: pkg.main,\n // code_md5: pkg.code_md5,\n asset_md5: pkg.asset_md5,\n asset_url: pkg.asset_url,\n };\n }\n }\n // 写入 JSON 文件\n const configPath = path.join(TEMP_DIR, 'TTMG_ODR_CONFIG.json');\n fs.writeFileSync(configPath, JSON.stringify(TTMGODRConfig, null, 2), 'utf-8');\n logger.info(`TTMG_ODR_CONFIG.json 已生成: ${configPath}`);\n}\n","\nimport * as fs from 'fs';\nimport { logger } from '@utils';\nimport * as path from 'path';\n\nimport { ODRPackages } from '@typings';\nimport { gameOdrPackages } from './mock';\n\nexport function fetchGames(params: { size: number; offset: number }): Promise<{\n data: Array<{\n id: string;\n packages: ODRPackages;\n }>;\n total: number;\n}> {\n const { size, offset } = params;\n const url = `https://game-odr.bytedance.net/odr/api/v1/games?size=${size}&offset=${offset}`;\n return fetch(url).then(res => res.json());\n}\n\n\n\nexport async function fetchAllGames(): Promise<Array<{\n id: string;\n packages: ODRPackages;\n }>> {\n return gameOdrPackages;\n const size = 100;\n let offset = 0;\n let total = 0;\n const games: Array<{\n id: string;\n packages: ODRPackages;\n }> = [];\n while (true) {\n const res = await fetchGames({ size, offset });\n games.push(...res.data);\n total = res.total;\n if (games.length >= total) {\n break;\n }\n offset += size;\n }\n return games;\n}\n\n\n// 同步下载并写入所有 code_url 文件\nexport function downloadAndSaveCodesSync(\n games: Array<{\n id: string;\n packages: ODRPackages;\n }>,\n tempDir: string,\n) {\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameDir = path.join(tempDir, gameId);\n if (!fs.existsSync(gameDir)) fs.mkdirSync(gameDir);\n\n for (const packageName in packages) {\n const packageInfo = packages[packageName];\n const { code_url } = packageInfo;\n // 同步下载\n const code = require('child_process')\n .execSync(`curl -s \"${code_url}\"`)\n .toString();\n logger.info(`downloadAndSaveCodesSync code, ${code_url}`);\n const fileName = `${packageName}.txt`;\n const filePath = path.join(gameDir, fileName);\n fs.writeFileSync(filePath, code);\n }\n }\n}","import { logger } from '@utils';\nimport * as path from 'path';\nimport * as fs from 'fs';\n\nimport {\n GAME_PACK_CONFIG_FILE_NAME,\n ODR_ASSET_DIR_SUFFIX,\n ODR_CODE_DIR_SUFFIX,\n} from '@constants';\n\n/**\n * 将 packageConfig packages 中 分拆的包 配置迁移到 odr_packages 中\n */\n\nexport async function writeOdrConfig(outputDir: string) {\n logger.info('writeOdrPackagesConfig start');\n const outputPath = path.join(outputDir, GAME_PACK_CONFIG_FILE_NAME);\n const config = JSON.parse(fs.readFileSync(outputPath, 'utf-8'));\n const packages = config.packages;\n const odrPackages = config.odr_packages || {};\n for (const pkgName in packages) {\n if (pkgName.endsWith(ODR_CODE_DIR_SUFFIX)) {\n /**\n * 找到 assets 对应的包\n */\n const pkgConfig = packages[pkgName];\n const relatedPkgName = pkgName.split(ODR_CODE_DIR_SUFFIX)[0];\n if (!odrPackages[relatedPkgName]) {\n odrPackages[relatedPkgName] = {};\n }\n odrPackages[relatedPkgName] = {\n ...odrPackages[relatedPkgName],\n root: pkgConfig.root,\n code_output: pkgConfig.output,\n code_md5: pkgConfig.md5,\n main: pkgConfig.main,\n };\n delete packages[pkgName];\n }\n if (pkgName.endsWith(ODR_ASSET_DIR_SUFFIX)) {\n const relatedPkgName = pkgName.split(ODR_ASSET_DIR_SUFFIX)[0];\n const pkgConfig = packages[pkgName];\n odrPackages[relatedPkgName] = {\n ...odrPackages[relatedPkgName],\n root: pkgConfig.root,\n asset_output: pkgConfig.output,\n asset_md5: pkgConfig.md5,\n };\n delete packages[pkgName];\n }\n }\n config.odr_packages = odrPackages;\n fs.writeFileSync(outputPath, JSON.stringify(config, null, 2));\n logger.info('writeOdrPackagesConfig end');\n}\n"],"names":["ensureDirSync","dir","fs","existsSync","mkdirSync","recursive","logger","constructor","this","init","output","enableLog","winston","createLogger","level","format","combine","timestamp","json","transports","File","filename","path","join","Console","info","msg","error","warn","debug","log","setOutput","removeEmptyDir","statSync","isDirectory","isEmpty","files","readdirSync","file","fullPath","rmSync","GAME_PACK_CONFIG_FILE_NAME","GAME_PROJECT_CONFIG_FILE_NAME","GAME_ORIGIN_CONFIG_FILE_NAME","GAME_MAIN_PACKAGE_NAME","STTPKG_EXT","ODR_CODE_DIR_SUFFIX","ODR_ASSET_DIR_SUFFIX","ONE_ODR_PKG_LIMIT","AVAILABLE_ODR_TAG_LIST","PROJECT_SIZE_LIMIT","MAINPACKAGE_SIZE_LIMIT","getDirSizeSync","filterDirs","totalSize","entries","withFileTypes","entry","name","includes","isFile","size","deleteJsFilesSync","endsWith","unlinkSync","err","deleteNoJsFilesSync","uselessDirs","uselessFiles","odrPackages","__GAME__","code_url","asset_url","asset_md5","root","main","configBundle","gameBundle","mainBundle","resources","iceAndFire","pushstone","gameOdrPackages","id","Math","random","packages","checkProject","entryDir","config","projectSizeLimit","isDev","gameSize","gameSizeMB","round","limitMB","errMsg","Error","console","checkProjectSize","gameJsonPath","checkGameJson","gameJsonConfigPath","gameJsonConfig","JSON","parse","readFileSync","appid","checkProjectConfig","checkMainPackage","gameJsPath","mainPackageSizeLimit","mainPkgSize","mainPkgSizeMB","checkMainPackageSize","checkSubpackages","subpackageSizeLimit","subpackages","length","forEach","sub","subpackageEntryDir","subpackageRoot","subPkgName","limit","subPkgDir","subPkgSize","sizeMB","checkSubPackageSize","async","checkPkgs","setup","outputDir","gameJson","url","md5","configs","normalizeName","String","replace","getMainFromRoot","str","test","normalizeRoot","map","transformSubPackages","result","Object","keys","pkg","pkgPath","writeFileSync","stringify","flattenMaps","allMaps","list","values","arr","push","collectAllJsFiles","exts","excludeDirs","concat","some","ext","collectDeps","gameEntry","rootPrefixes","filter","getRootPrefixes","glob","sync","cwd","ignore","requireCalls","code","ast","acorn","ecmaVersion","sourceType","e","walkAst","node","type","callee","arguments","requirePath","value","resolvedPath","startsWith","resolve","dirname","prefix","filePath","fileExistsWithExtensions","relative","newRequireCalls","i","depModule","curModule","assign","reduce","acc","cur","f","cb","key","hasOwnProperty","child","Array","isArray","c","pack","gameOutput","pkgName","allDeps","pkgConfig","destRoot","enableSourcemap","startTime","Date","now","packMap","moduleConfig","deps","packFileName","relPaths","Bundle","separator","relPath","absPath","basename","fileId","defineHeader","defineFooter","fileMagic","MagicString","wrapped","appendLeft","toString","append","addSource","content","outFile","bundledCode","generateMap","hires","includeContent","fileContent","Buffer","from","partition","pkgRoot","packedFiles","skipDirs","isRoot","srcPath","destPath","copyFileSync","makePkgs","removeUselessDirs","message","stat","force","removeSystemUseless","subRoots","split","sep","Boolean","uniqueSubRoots","Set","srcRoot","jsFiles","packNameRoot","collectMaps","rules","outputPath","pkgOutput","codeOutput","assetOutput","cpSync","code_md5","makeOdrPkgs","gamePackConfigPath","gamePackConfig","pkgDir","isSymbolicLink","linkTarget","readlinkSync","symlinkSync","mergePkgs","packageConfig","isSuccess","errorMsg","TEMP_DIR","__dirname","games","fetchAllGames","tempDir","game","gameId","gameDir","packageName","packageInfo","require","execSync","fileName","downloadAndSaveCodesSync","gamesContentSize","odrTagGameMap","tag","TTMGODRConfig","assignedTag","odrTag","odrTagDir","sum","gid","srcDir","destDir","configPath","odr_packages","relatedPkgName","code_output","asset_output"],"mappings":"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"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "ttmg-pack",
3
3
  "author": "ttmg",
4
- "version": "0.0.18-beta.3",
4
+ "version": "0.0.18-beta.5",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
7
7
  "scripts": {