ttmg-pack 0.0.33 → 0.0.34
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/CHANGELOG.md +5 -1
- package/dist/index.js +3 -3
- package/dist/index.js.map +1 -1
- package/dist/libs/mergePkgs/index.d.ts +1 -1
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
package/dist/index.js
CHANGED
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* ==========================================
|
|
3
3
|
* @Description: ttmg pack
|
|
4
|
-
* @Version: 0.0.
|
|
4
|
+
* @Version: 0.0.34
|
|
5
5
|
* @Author: zhanghongyang.mocha
|
|
6
|
-
* @Date: 2025-09-22
|
|
6
|
+
* @Date: 2025-09-22 07:11:34
|
|
7
7
|
* ==========================================
|
|
8
8
|
*/
|
|
9
|
-
"use strict";var e=require("fs"),s=require("winston"),t=require("path");require("crypto"),require("zlib");var a=require("fs/promises"),i=require("node-fetch"),c=require("acorn"),n=require("glob"),o=require("magic-string"),r=require("esbuild");function d(e){var s=Object.create(null);return e&&Object.keys(e).forEach(function(t){if("default"!==t){var a=Object.getOwnPropertyDescriptor(e,t);Object.defineProperty(s,t,a.get?a:{enumerable:!0,get:function(){return e[t]}})}}),s.default=e,Object.freeze(s)}var b=d(e),m=d(t);function u(s){e.existsSync(s)||e.mkdirSync(s,{recursive:!0})}const f=new class{constructor(){this.logger=null}init(e,a=!0){this.enableLog=a,this.logger=s.createLogger({level:"info",format:s.format.combine(s.format.timestamp(),s.format.json()),transports:[new s.transports.File({filename:t.join(e,"pack.log")}),new s.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 g(s){if(!e.existsSync(s))return!1;if(!e.statSync(s).isDirectory())return!1;let a=!0;const i=e.readdirSync(s);for(const c of i){const i=t.join(s,c);if(e.statSync(i).isDirectory()){g(i)||(a=!1)}else a=!1}return!!a&&(e.rmSync(s,{recursive:!0}),!0)}const k="packageConfig.json",l="project.config.json",p="game.json",_="__GAME__",h=".txt",y="_code",j="_asset",v=62914560,x=["ttmg_odr_high_channel","ttmg_odr_middle_channel","ttmg_odr_normal_channel"];function S(s,a=[]){let i=0;if(!e.existsSync(s))return 0;const c=e.readdirSync(s,{withFileTypes:!0});for(const n of c){const c=t.join(s,n.name);n.isDirectory()?a.includes(n.name)||(i+=S(c,a)):n.isFile()&&(i+=e.statSync(c).size)}return i}function w(s){const a=e.readdirSync(s,{withFileTypes:!0});for(const i of a){const a=t.join(s,i.name);if(i.isDirectory())w(a);else if(i.isFile()&&(a.endsWith(".js")||a.endsWith(".js.map")))try{e.unlinkSync(a)}catch(e){}}}function z(s){const a=e.readdirSync(s,{withFileTypes:!0});for(const i of a){const a=t.join(s,i.name);if(i.isDirectory())z(a);else if(i.isFile()&&!a.endsWith(".js")&&!a.endsWith(".js.map"))try{e.unlinkSync(a)}catch(e){}}}const q=["node_modules","__MACOSX"],$=[".DS_Store","Thumbs.db"];async function O(e,s,c=5){const n=[];for(const i of e){const{id:e,packages:c}=i,o=t.join(s,e);await a.mkdir(o,{recursive:!0});for(const e in c){const s=c[e],{code_url:a}=s,i=`${e}.txt`,r=t.join(o,i);n.push({code_url:a,filePath:r,packageName:e})}}await async function(e,s,t){const a=[],i=[];for(const c of s){const n=Promise.resolve().then(()=>t(c));if(a.push(n),e<=s.length){const s=n.then(()=>i.splice(i.indexOf(s),1));i.push(s),i.length>=e&&await Promise.race(i)}}return Promise.all(a)}(c,n,async e=>{const{code_url:s,filePath:t,packageName:c}=e;console.log(`start to download ${c}`);try{const e=await i(s);if(!e.ok)throw new Error(`Failed to fetch ${s}: ${e.statusText}`);const n=Buffer.from(await e.arrayBuffer());await a.writeFile(t,n),console.log(`finish to download ${c}`)}catch(e){console.error(`Error downloading ${c}:`,e)}})}var M={MinisPackagingInfoMap:{sbmg0cldak4bbz86dq:{ClientKey:"sbmg0cldak4bbz86dq",MinisStage:1,MinisURL:"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us",ODRPackages:'{"112":{"asset_md5":"9fe63b5fc31e9b4ccd1ee9fa9ed2913b","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/112_asset/9fe63b5fc31e9b4ccd1ee9fa9ed2913b.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/112_code/ebb1a6d341c0013a445f3dbf2600c9ae.txt","root":"subpackages/112","main":"subpackages/112/game.js"},"116":{"asset_md5":"690814b650564724a235a4f155cfe5e3","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/116_asset/690814b650564724a235a4f155cfe5e3.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/116_code/ddc0d442aef316e3e45cdbac9ccb88eb.txt","root":"subpackages/116","main":"subpackages/116/game.js"},"108":{"asset_md5":"b7b3aeafdd70aed9e77fb2f1554d6675","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/108_asset/b7b3aeafdd70aed9e77fb2f1554d6675.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/108_code/bdac7d266b4a15af9ca109d1673168a6.txt","root":"subpackages/108","main":"subpackages/108/game.js"},"game3":{"asset_md5":"bc4b09b1177a35fd8516b8fd5bb9b4e4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game3_asset/bc4b09b1177a35fd8516b8fd5bb9b4e4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game3_code/7f3c18114afd5b3d4fd8c39dd2481cf1.txt","root":"subpackages/game3","main":"subpackages/game3/game.js"},"game5":{"asset_md5":"6a75dd22dcf0d7c117cfcb03185696e6","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game5_asset/6a75dd22dcf0d7c117cfcb03185696e6.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game5_code/ed2c8ca8023418099d1232fdc6ac09ae.txt","root":"subpackages/game5","main":"subpackages/game5/game.js"},"main":{"asset_md5":"ecadb8f2a4138a73459e04a8a0c5f470","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/main_asset/ecadb8f2a4138a73459e04a8a0c5f470.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/main_code/3b7a7f0869e5392f1e244eef40ad8abe.txt","root":"subpackages/main","main":"subpackages/main/game.js"},"103":{"asset_md5":"069620ed43643956f6b6448c5d360471","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/103_asset/069620ed43643956f6b6448c5d360471.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/103_code/64a9caf580b6cd991888751e8ea51222.txt","root":"subpackages/103","main":"subpackages/103/game.js"},"other":{"asset_md5":"473ad574dee1813610988dba22fb0acc","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/other_asset/473ad574dee1813610988dba22fb0acc.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/other_code/003784b887ff99fb82d47899798e1ee0.txt","root":"subpackages/other","main":"subpackages/other/game.js"},"resSps":{"asset_md5":"4e72c930e3c8f77bda25be975e3c8019","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/resSps_asset/4e72c930e3c8f77bda25be975e3c8019.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/resSps_code/c6ac2ebde3c8c421ca75ff649280d90b.txt","root":"subpackages/resSps","main":"subpackages/resSps/game.js"},"111":{"asset_md5":"3452f073e30ecc6b3727f072a856068b","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/111_asset/3452f073e30ecc6b3727f072a856068b.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/111_code/3ca9e57876b82945faedce2ca0ab414e.txt","root":"subpackages/111","main":"subpackages/111/game.js"},"117":{"asset_md5":"60206e41273a6a1f6d21bbf634023962","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/117_asset/60206e41273a6a1f6d21bbf634023962.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/117_code/1a85a85048eca0d650ad4b8b198cefdc.txt","root":"subpackages/117","main":"subpackages/117/game.js"},"__GAME__":{"asset_md5":"dbee9c4f704800eb5bf482be01aecadc","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/__GAME___asset/dbee9c4f704800eb5bf482be01aecadc.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/__GAME___code/b42b2a68ce373eb274eedc22351aac02.txt","root":"","main":"game.js"},"jigsaw":{"asset_md5":"4782715dda5ea0d10897d866a84bca19","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/jigsaw_asset/4782715dda5ea0d10897d866a84bca19.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/jigsaw_code/799004f65797d4620549e0d4b27e3794.txt","root":"subpackages/jigsaw","main":"subpackages/jigsaw/game.js"},"newYear":{"asset_md5":"c2cb90fbaac0b4b86d1afa7caabd2f20","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/newYear_asset/c2cb90fbaac0b4b86d1afa7caabd2f20.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/newYear_code/a7d962b4f7a95e8114fcf00bbe27b8ac.txt","root":"subpackages/newYear","main":"subpackages/newYear/game.js"},"adapter":{"asset_md5":"a44f0cd93ad08da6db427c9dc3ab6fe4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/adapter_asset/a44f0cd93ad08da6db427c9dc3ab6fe4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/adapter_code/6c3090ba10c12ad637aefa993fe838fc.txt","root":"subpackages/adapter","main":"subpackages/adapter/game.js"},"101":{"asset_md5":"ad3f476d334e695c01f96eef67308f26","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/101_asset/ad3f476d334e695c01f96eef67308f26.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/101_code/dcb5f8547d5285c5e116a1726dc1ecdb.txt","root":"subpackages/101","main":"subpackages/101/game.js"},"102":{"asset_md5":"8b39712d9f0ae7549382da9041cc5f56","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/102_asset/8b39712d9f0ae7549382da9041cc5f56.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/102_code/b70fb4ea0d05b866a4a32ab4cae418c4.txt","root":"subpackages/102","main":"subpackages/102/game.js"},"110":{"asset_md5":"24da51736441ceb88b3d331775004551","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/110_asset/24da51736441ceb88b3d331775004551.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/110_code/f7c97cb3aed7acb4091feaf613c56dd3.txt","root":"subpackages/110","main":"subpackages/110/game.js"},"114":{"asset_md5":"c2f81aaf93311b9302995dba105d69d0","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/114_asset/c2f81aaf93311b9302995dba105d69d0.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/114_code/7574a218fb4053e2005497d84a56132c.txt","root":"subpackages/114","main":"subpackages/114/game.js"},"practice":{"asset_md5":"01d63b100e2cfb80d2d7568d11f346f8","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/practice_asset/01d63b100e2cfb80d2d7568d11f346f8.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/practice_code/2275ac8e93d1c5dd100427aa948cd23c.txt","root":"subpackages/practice","main":"subpackages/practice/game.js"},"zh_CN":{"asset_md5":"2b1c715a0c624c630acb29b4fdfcb240","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/zh_CN_asset/2b1c715a0c624c630acb29b4fdfcb240.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/zh_CN_code/bb44d46f30e57f29d20bf0678426c7c5.txt","root":"subpackages/zh_CN","main":"subpackages/zh_CN/game.js"},"107":{"asset_md5":"652c5385223642c4a19f823666c8a682","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/107_asset/652c5385223642c4a19f823666c8a682.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/107_code/9e717b17251bdc8eefb7b8135fcb6996.txt","root":"subpackages/107","main":"subpackages/107/game.js"},"113":{"asset_md5":"f53882a23326919614b0e9031d28ed39","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/113_asset/f53882a23326919614b0e9031d28ed39.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/113_code/666bb07aea74f0a408136c0ac16a04b8.txt","root":"subpackages/113","main":"subpackages/113/game.js"},"115":{"asset_md5":"6831f12abfeccc9ff720155d9ece2d53","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/115_asset/6831f12abfeccc9ff720155d9ece2d53.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/115_code/76b85e96a57c33c8ba9ff2f316290416.txt","root":"subpackages/115","main":"subpackages/115/game.js"},"118":{"asset_md5":"f787456ee45eccc4c13f20ee21c61858","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/118_asset/f787456ee45eccc4c13f20ee21c61858.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/118_code/e595f8ed244752fe2ddc72def8452cd5.txt","root":"subpackages/118","main":"subpackages/118/game.js"},"chapter":{"asset_md5":"9c4b6cb8dc0b39bfcb275efa0c6b8493","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/chapter_asset/9c4b6cb8dc0b39bfcb275efa0c6b8493.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/chapter_code/9ee175f09ca69c65b630cfee23c0e42c.txt","root":"subpackages/chapter","main":"subpackages/chapter/game.js"},"sound":{"asset_md5":"65c37817ffc86d942d1942443b5e3fb4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/sound_asset/65c37817ffc86d942d1942443b5e3fb4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/sound_code/2662512b5aaa6801b6742eb661c44828.txt","root":"subpackages/sound","main":"subpackages/sound/game.js"},"105":{"asset_md5":"82c8cde04fec4456ac6bab8adaef0028","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/105_asset/82c8cde04fec4456ac6bab8adaef0028.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/105_code/74e319194a7f5a204504fc3d83131112.txt","root":"subpackages/105","main":"subpackages/105/game.js"},"109":{"asset_md5":"926ebc459e3329a4d9cda7ddf9718bf4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/109_asset/926ebc459e3329a4d9cda7ddf9718bf4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/109_code/19cbdc3d88a72dbd0c5eb52e89e2635a.txt","root":"subpackages/109","main":"subpackages/109/game.js"},"game2":{"asset_md5":"9f1bb086f7587aada8312315f2dd94fc","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game2_asset/9f1bb086f7587aada8312315f2dd94fc.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game2_code/49d07e848cae8eb08428bfbf225a6f30.txt","root":"subpackages/game2","main":"subpackages/game2/game.js"},"game6":{"asset_md5":"b1407de78d19f60644b53436f1791ee6","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game6_asset/b1407de78d19f60644b53436f1791ee6.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game6_code/155651953a65a663f67b95475cd5c941.txt","root":"subpackages/game6","main":"subpackages/game6/game.js"},"prefab":{"asset_md5":"4c4ba8d442bae683575460e884ddba00","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/prefab_asset/4c4ba8d442bae683575460e884ddba00.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/prefab_code/7f037cd7f3b83e90765e749d28f32e3e.txt","root":"subpackages/prefab","main":"subpackages/prefab/game.js"},"104":{"asset_md5":"160c617e3e3adef19c94f57bc4520dba","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/104_asset/160c617e3e3adef19c94f57bc4520dba.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/104_code/ef86df3a24fca9ecffc55f7c33ff3747.txt","root":"subpackages/104","main":"subpackages/104/game.js"},"106":{"asset_md5":"b22052c8f569db2f00a2ae6fcb8872e6","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/106_asset/b22052c8f569db2f00a2ae6fcb8872e6.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/106_code/f13ed6653a2ebb49703d0bc882255cfd.txt","root":"subpackages/106","main":"subpackages/106/game.js"},"game4":{"asset_md5":"b98a8892ccb76e5999cbf7a8e81c4cb5","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game4_asset/b98a8892ccb76e5999cbf7a8e81c4cb5.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game4_code/ebbcf2b74998d340cb8cf85249748992.txt","root":"subpackages/game4","main":"subpackages/game4/game.js"}}',UpdatedTime:1756716749e3,AssetType:1}}};function D(s,a){f.info("start check project"),function(e,s){f.info("start check project size");const{build:{pkgSizeLimit:t},dev:a}=s,i=S(e,[]),c=Math.round(i/1048576),n=Math.round(t/1048576);if(i>t){const e=`Game size ${c}MB, must not exceed ${n}MB`;if(f.error(e),!a.enable)throw new Error(e);console.log("[1m[33m%s[0m",e)}else f.info(`Game size ${c}MB, check successfully!`),f.info("Game size check successfully!")}(s,a),function(s){f.info("start check config");const a=t.join(s,p);if(!e.existsSync(a)){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")}(s),function(s){f.info("start check project config");const a=t.join(s,l);if(!e.existsSync(a)){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 i;try{i=JSON.parse(e.readFileSync(a,"utf-8"))}catch(e){throw f.error("check project.config.json failed, your project.config.json is invalid json format"),e}if(!(null==i?void 0:i.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)}}(s),f.info("check project successfully")}function E(e,s){f.info("start check main package"),f.info("check game.js");const t=m.join(e,"game.js");if(!b.existsSync(t)){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:s}){f.info("start check main package size");const{build:t,dev:a}=s,i=S(e,["subpackages"]),c=Math.round(i/1048576),n=Math.round(t.pkgSizeLimit/1048576);if(i>t.pkgSizeLimit){const e=`Main package size ${c}MB, must not exceed ${n}MB`;if(!a.enable)throw f.error(e),new Error(e);console.log("[1m[33m%s[0m",e)}else f.info(`Main package size ${c}MB, check successfully!`)}({entryDir:e,config:s}),f.info("check main package successfully")}function F(s,a){const{build:i}=a||{},{subPkgSizeLimit:c}=i||{};f.info("start check subpackages in game.json");const n=t.join(s,p),o=JSON.parse(e.readFileSync(n,"utf-8")).subpackages||[];(null==o?void 0:o.length)?o.forEach(a=>{if(!a.name||!a.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=t.join(s,a.root);if(!e.existsSync(i)){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${a.name} entry dir, please check it`;throw f.error(e),new Error(e)}const n=a.root;if(!e.existsSync(t.join(s,n))){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${a.name} root, please check it`;throw f.error(e),new Error(e)}c&&function({entryDir:e,subPkgName:s,limit:a}){const i=t.join(e,"subpackages",s),c=S(i,[]),n=Math.round(c/1048576);if(n>a){const e=`${s} Sub package size ${n}MB, must not exceed ${a/1048576}MB`;throw f.error(e),new Error(e)}f.info(`${s} Sub package size ${n}MB, check successfully!`)}({entryDir:s,subPkgName:a.name,limit:c})}):f.info("subpackages is empty"),f.info("check subpackages in game.json successfully")}async function P({entryDir:e,config:s}){f.info("开始校验项目"),D(e,s),E(e,s),F(e,s),f.info("校验项目完成")}async function N(s,a){f.info("开始基于源代码中的 game.json 生成 packageConfig.json");const i=t.join(s,p),c=JSON.parse(e.readFileSync(i,"utf-8")),n={};n[_]={url:"",md5:"",root:"",main:"game.js",output:`${_}${h}`},function(e){if(!e||0===e.length)return[];function s(e){return String(e).replace(/^\/+|\/+$/g,"")}function t(e){return a(e)?`${e}game.js`:e}function a(e){return e.endsWith("/")||!/\.[a-zA-Z0-9]+$/.test(e)}function i(e){return e.replace(/^\/+|\/+$/g,"")}return e.map(e=>{const a=s(e.name);return{main:t(e.root),root:i(e.root),name:a}})}(c.subpackages).forEach(({name:e,root:s,main:t})=>{n[e]={url:"",md5:"",root:s,main:t,output:`${e}${h}`}});const o={packages:n};return Object.keys(n).forEach(a=>{const i=t.join(s,n[a].main);if(!e.existsSync(i)){const e=`开发者源代码包中的子包 ${a} 中的 main 文件 ${n[a].main} 不存在`;throw f.error(e),new Error(e)}}),e.existsSync(a)&&e.rmSync(a,{recursive:!0}),e.mkdirSync(a),e.writeFileSync(t.join(a,k),JSON.stringify(o,null,2)),f.info("基于源代码中的 game.json 生成 packageConfig.json 成功"),o}function T(e){const s=[];return Object.values(e).forEach(e=>{Object.values(e).forEach(e=>{s.push(...e)})}),s}function C({entry:s,root:a,exts:i=[".js",".ts",".jsx",".tsx"],excludeDirs:c=[]}){let n=[];if(!e.existsSync(s))return n;const o=e.readdirSync(s,{withFileTypes:!0});for(const e of o){const o=t.join(s,e.name);if(e.isDirectory()){const e=t.relative(a,o);if(c.find(s=>s===e)){console.log("collectPkgJsFiles skip dir",e);continue}n=n.concat(C({entry:o,root:a,exts:i,excludeDirs:c}))}else i.some(s=>e.name.endsWith(s))&&n.push(o)}return n}function R(s,a){const i=function(s){return e.readdirSync(s).filter(a=>{const i=t.join(s,a);return e.statSync(i).isDirectory()&&!["node_modules","dist","build","output"].includes(a)}).map(e=>e+"/")}(s),o=n.sync("**/*.js",{cwd:s,ignore:[]}),r=[];o.forEach(a=>{const n=t.join(s,a),o=e.readFileSync(n,"utf-8");let d;try{d=c.parse(o,{ecmaVersion:"latest",sourceType:"module"})}catch(e){return}L(d,a=>{if("CallExpression"===a.type&&"Identifier"===a.callee.type&&("require"===a.callee.name||"requireAsync"===a.callee.name)&&a.arguments.length>0&&"Literal"===a.arguments[0].type){const c=a.arguments[0].value;let o=null;if(c.startsWith("."))o=t.resolve(t.dirname(n),c);else if(c.startsWith("/"))o=t.resolve(s,"."+c);else for(const e of i)if(c.startsWith(e)){o=t.resolve(s,c);break}o&&!function(s){if(e.existsSync(s)&&e.statSync(s).isFile())return!0;const t=[".js",".ts",".jsx",".tsx"];for(const a of t)if(e.existsSync(s+a)&&e.statSync(s+a).isFile())return!0;return!1}(o)&&(o=null),r.push({file:t.relative(s,n),callee:a.callee.name,requirePath:c,resolvedPath:o?t.relative(s,o):""})}})});const d=r.map(e=>{let s="";Object.keys(a).forEach(t=>{e.resolvedPath&&e.resolvedPath.startsWith(a[t].root)&&(s=t)});let t="";return Object.keys(a).forEach(s=>{e.file.startsWith(a[s].root)&&(t=s)}),Object.assign(Object.assign({},e),{curModule:t,depModule:s})}),b=[".js",".ts",".jsx",".tsx"];return d.reduce((e,s)=>{return e[s.curModule]||(e[s.curModule]={}),s.curModule===s.depModule||s.resolvedPath&&(t=s.resolvedPath,b.some(e=>t.endsWith(e)))&&(e[s.curModule]=Object.assign(Object.assign({},e[s.curModule]),{[s.resolvedPath]:s.depModule})),e;var t},{})}function L(e,s){s(e);for(let t in e)if(e.hasOwnProperty(t)){const a=e[t];Array.isArray(a)?a.forEach(e=>e&&"string"==typeof e.type&&L(e,s)):a&&"string"==typeof a.type&&L(a,s)}}async function A({pkgRoot:s,destRoot:a,packedFiles:i,gameEntry:c,gameOutput:n,skipDirs:o=[],isRoot:r=!0}){if(!e.existsSync(s))return;const d=e.readdirSync(s,{withFileTypes:!0});for(const r of d){const d=t.join(s,r.name),b=t.join(a,r.name),m=t.join(s,r.name);if(r.isDirectory()){const e=t.relative(c,m);if(o.find(s=>s===e))continue}r.isDirectory()?A({pkgRoot:d,destRoot:b,gameOutput:n,packedFiles:i,gameEntry:c,skipDirs:o,isRoot:!1}):(u(t.dirname(b)),i.includes(t.relative(c,d))||e.copyFileSync(d,b))}}function G(e){const s=[];return e&&s.push(...Object.values(e).map(e=>e.root)),s}async function J({gameEntry:s,gameOutput:a,config:i}){let c=Date.now();!function(s){!function s(a){let i;try{i=e.readdirSync(a)}catch(e){return void f.warn(`无法访问目录: ${a}, ${e.message}`)}i.forEach(i=>{const c=t.join(a,i);try{const t=e.statSync(c);t.isDirectory()?q.includes(i)?(e.rmSync(c,{recursive:!0,force:!0}),f.info(`已删除无用目录: ${c}`)):s(c):t.isFile()&&$.includes(i)&&(e.rmSync(c,{force:!0}),f.info(`已删除无用文件: ${c}`))}catch(e){f.warn(`无法处理: ${c}, ${e.message}`)}})}(s)}(s),f.info(`pack start,startTime:${c}`);const{packages:n}=await N(s,a),d=R(s,n),b=function(e,s){f.info("开始基于游戏源代码收集分包中的 JS 文件");const a={},i=Object.values(s).filter(e=>e.root&&""!==e.root).map(e=>e.root).filter(Boolean),c=Array.from(new Set(i));for(const i in s){const n=s[i];let o,r;"__GAME__"===i?(r=C({entry:e,root:e,exts:[".js",".ts",".jsx",".tsx"],excludeDirs:c}),o=e):(o=t.join(e,n.root),r=C({entry:o,root:o,exts:[".js",".ts",".jsx",".tsx"],excludeDirs:[]}));const d=[];r.forEach(s=>{d.push(t.relative(e,s).replace(/\\/g,"/"))});const b=s[i].root;a[i]={[b?`${b}/game.pack.js`:"game.pack.js"]:d}}return f.info("基于游戏源代码收集分包中的 JS 文件完成"),a}(s,n);return await async function(e,s,t){const a=[],i=[];for(const c of s){const n=Promise.resolve().then(()=>t(c));if(a.push(n),e<=s.length){const s=n.then(()=>{i.splice(i.indexOf(s),1)});i.push(s),i.length>=e&&await Promise.race(i)}}return Promise.all(a)}(10,Object.keys(b),async c=>{f.info(`开始基于游戏源代码打包,包名:${c}`);let m=Date.now();const k=n[c],l=t.join(s,k.root),p=t.join(a,c,k.root);await function({gameEntry:s,pkgName:a,allDeps:i,allMaps:c,pkgConfig:n,destRoot:d,config:b}){var m,g,k,l;let p=Date.now();f.info(`pack start,startTime:${p}`);const _=c[a],h={main:n.main,deps:i[a]||{},map:c[a]};u(d),e.writeFileSync(t.join(d,"moduleConfig.json"),JSON.stringify(h));for(const i in _){const c=_[i];if(!Array.isArray(c)||0===c.length)continue;const n=new o.Bundle({separator:"\n"});for(const a of c){const c=t.join(s,a);if(e.existsSync(c)&&t.basename(c)!==i&&/\.js$/i.test(a))try{const{code:s}=r.transformSync(e.readFileSync(c,"utf-8"),{loader:"js",format:"cjs",target:"es2015"}),t=a.replace(/\\/g,"/"),i=`define("game:${t}", ["require", "requireAsync", "module", "exports", "sandboxGlobal"], function(require, requireAsync, module, exports, sandboxGlobal){\nwith(sandboxGlobal){\n`,d="\n}\n});\n",b=new o(s,{filename:t});b.prepend(i),b.append(d),n.addSource({content:b,filename:t})}catch(e){f.error(`pack error, relPath:${a},e:${e}`)}}const u=t.join(d,t.basename(i)),p=n.toString();if(null===(m=null==b?void 0:b.dev)||void 0===m?void 0:m.enableSourcemap){const s=n.generateMap({hires:!0,includeContent:!0,file:t.basename(i)});if(null===(g=null==b?void 0:b.dev)||void 0===g?void 0:g.inlineSourcemap){const t=p+`\n//# sourceMappingURL=data:application/json;base64,${Buffer.from(s.toString()).toString("base64")}`+`\n//# sourceURL=${i}`;e.writeFileSync(u,t,"utf-8")}else{const a=`http://${null===(k=null==b?void 0:b.dev)||void 0===k?void 0:k.host}:${null===(l=null==b?void 0:b.dev)||void 0===l?void 0:l.port}/game/files`,c=t.basename(i)+".map",n=t.join(d,c);e.writeFileSync(n,s.toString(),"utf-8");const o=p+`\n//# sourceMappingURL=${a}/${i}.map`+`\n//# sourceURL=${i}`;e.writeFileSync(u,o,"utf-8")}}else{const s=p+`\n//# sourceURL=${i}`;e.writeFileSync(u,s,"utf-8")}f.info(`pack end,packName:${a}`)}}({gameEntry:s,allDeps:d,allMaps:b,pkgName:c,pkgConfig:k,destRoot:p,config:i}),await A({gameEntry:s,gameOutput:a,pkgRoot:l,destRoot:p,packedFiles:T(b),skipDirs:G(n)});g(t.join(a,c)),f.info(`基于游戏源代码打包分包完成,包名:${c}`),f.info(`基于游戏源代码打包分包耗时,包名:${c},耗时:${Date.now()-m}ms`)}),f.info(`pack end,duration:${Date.now()-c}ms`),{}}exports.buildOdrPkgs=async function({tempDir:s}){e.existsSync(s)&&e.rmSync(s,{recursive:!0}),e.mkdirSync(s);const a=function(){const{MinisPackagingInfoMap:e}=M;return Object.keys(e).map(s=>({id:s,packages:JSON.parse(e[s].ODRPackages)}))}();f.info("odr_games"),await O(a,s);const i={};for(const e of a){const{id:a}=e,c=t.join(s,a);i[a]=S(c)}const c={};x.forEach(e=>{c[e]=[]});const n={};for(const o of a){const{id:a,packages:r}=o,d=i[a];if(d>v)continue;let b=null;for(const n of x){const o=t.join(s,n);e.existsSync(o)||e.mkdirSync(o);if(c[n].reduce((e,s)=>e+i[s],0)+d<=v){const i=t.join(s,a),r=t.join(o,a);e.cpSync(i,r,{recursive:!0}),c[n].push(a),b=n;break}}if(b){n[a]={tag:b,packages:{}};for(const e in r){const s=r[e];f.info(`TTMGODRConfig[gameId].packages, ${s}, ${e}`),n[a].packages[e]={root:s.root,main:s.main,asset_md5:s.asset_md5,asset_url:s.asset_url}}console.log(`finish to build ${a}`),e.rmSync(t.join(s,a),{recursive:!0})}}const o=t.join(s,"TTMG_ODR_CONFIG.json");e.writeFileSync(o,JSON.stringify(n,null,2),"utf-8"),f.info(`TTMG_ODR_CONFIG.json 已生成: ${o}`)},exports.buildPkgs=async function(s){const{entry:a,output:i,build:c}=s;let n=Date.now();f.init(i,!0),f.info("TTMG_PACK_VERSION: 0.0.33"),f.info(`pack start, startTime:${n}`),await P({entryDir:a,config:s}),await J({config:s,gameEntry:a,gameOutput:i}),(null==c?void 0:c.enableOdr)&&await async function(s){let a=Date.now();f.info(`makeOdrPkgs start, ${a}`);const i=t.join(s,k),c=JSON.parse(e.readFileSync(i,"utf-8")),n=c.packages;for(const a in n){const i=t.join(s,a),c=t.join(s,`${a}${y}`),o=t.join(s,`${a}${j}`);u(c),u(o),e.cpSync(i,c,{recursive:!0}),e.cpSync(i,o,{recursive:!0}),z(c),w(o),n[`${a}${y}`]=Object.assign(Object.assign({},n[a]),{code_url:"",code_md5:"",output:`${a}${y}${h}`}),n[`${a}${j}`]=Object.assign(Object.assign({},n[a]),{asset_url:"",asset_md5:"",output:`${a}${j}${h}`})}e.writeFileSync(i,JSON.stringify(c,null,2)),f.info(`makeOdrPkgs end, ${Date.now()-a}ms`)}(i),f.info(`pack end:${Date.now()-n}ms`)},exports.debugPkgs=async function(s){const{entry:a,output:i,dev:{enableLog:c}}=s;f.init(i,c);try{await P({entryDir:a,config:s}),await J({gameEntry:a,gameOutput:i,config:s}),await function(s){f.info("mergePkgs start");const a=t.join(s,k);let i;try{i=JSON.parse(e.readFileSync(a,"utf-8"))}catch(e){return void f.error(`读取配置文件失败: ${a}, err: ${e.message}`)}const c=i.packages;if(c&&"object"==typeof c){for(const a of Object.keys(c)){const i=t.join(s,a);if(!e.existsSync(i)){f.warn(`package 目录不存在: ${i}`);continue}const c=e.readdirSync(i);for(const a of c){const c=t.join(i,a),n=t.join(s,a);try{const s=e.statSync(c);if(s.isFile())e.copyFileSync(c,n),e.rmSync(c,{force:!0});else if(s.isDirectory())e.cpSync(c,n,{recursive:!0}),e.rmSync(c,{recursive:!0,force:!0});else if(s.isSymbolicLink()){console.log("symbolic link",c,n);const t=e.readlinkSync(c);e.existsSync(n)&&e.rmSync(n,{force:!0}),e.symlinkSync(t,n,s.isDirectory()?"dir":"file")}f.info(`合并文件/目录: ${c} -> ${n}`)}catch(e){f.error(`处理文件失败: ${c}, err: ${e.message}`)}}0===e.readdirSync(i).length&&(e.rmSync(i,{recursive:!0,force:!0}),f.info(`已删除 package 目录: ${i}`))}f.info("mergePkgs 完成")}else f.error("配置文件中 packages 字段无效, 请检查配置")}(i);const c=e.readFileSync(t.join(i,k),"utf-8");return{isSuccess:!0,packages:JSON.parse(c).packages}}catch(e){return{isSuccess:!1,errorMsg:e.message}}},exports.writeOdrConfig=async function(e){f.info("writeOdrPackagesConfig start");const s=m.join(e,k),t=JSON.parse(b.readFileSync(s,"utf-8")),a=t.packages,i=t.odr_packages||{};for(const e in a){if(e.endsWith(y)){const s=a[e],t=e.split(y)[0];i[t]||(i[t]={}),i[t]=Object.assign(Object.assign({},i[t]),{root:s.root,code_output:s.output,code_md5:s.md5,main:s.main}),delete a[e]}if(e.endsWith(j)){const s=e.split(j)[0],t=a[e];i[s]=Object.assign(Object.assign({},i[s]),{root:t.root,asset_output:t.output,asset_md5:t.md5}),delete a[e]}}t.odr_packages=i,b.writeFileSync(s,JSON.stringify(t,null,2)),f.info("writeOdrPackagesConfig end")};
|
|
9
|
+
"use strict";var e=require("fs"),s=require("winston"),t=require("path");require("crypto"),require("zlib");var a=require("fs/promises"),i=require("node-fetch"),c=require("acorn"),n=require("glob"),o=require("magic-string"),r=require("esbuild");function d(e){var s=Object.create(null);return e&&Object.keys(e).forEach(function(t){if("default"!==t){var a=Object.getOwnPropertyDescriptor(e,t);Object.defineProperty(s,t,a.get?a:{enumerable:!0,get:function(){return e[t]}})}}),s.default=e,Object.freeze(s)}var b=d(e),m=d(t);function u(s){e.existsSync(s)||e.mkdirSync(s,{recursive:!0})}const f=new class{constructor(){this.logger=null}init(e,a=!0){this.enableLog=a,this.logger=s.createLogger({level:"info",format:s.format.combine(s.format.timestamp(),s.format.json()),transports:[new s.transports.File({filename:t.join(e,"pack.log")}),new s.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 g(s){if(!e.existsSync(s))return!1;if(!e.statSync(s).isDirectory())return!1;let a=!0;const i=e.readdirSync(s);for(const c of i){const i=t.join(s,c);if(e.statSync(i).isDirectory()){g(i)||(a=!1)}else a=!1}return!!a&&(e.rmSync(s,{recursive:!0}),!0)}const k="packageConfig.json",l="project.config.json",p="game.json",_="__GAME__",h=".txt",y="_code",j="_asset",v=62914560,x=["ttmg_odr_high_channel","ttmg_odr_middle_channel","ttmg_odr_normal_channel"];function S(s,a=[]){let i=0;if(!e.existsSync(s))return 0;const c=e.readdirSync(s,{withFileTypes:!0});for(const n of c){const c=t.join(s,n.name);n.isDirectory()?a.includes(n.name)||(i+=S(c,a)):n.isFile()&&(i+=e.statSync(c).size)}return i}function w(s){const a=e.readdirSync(s,{withFileTypes:!0});for(const i of a){const a=t.join(s,i.name);if(i.isDirectory())w(a);else if(i.isFile()&&(a.endsWith(".js")||a.endsWith(".js.map")))try{e.unlinkSync(a)}catch(e){}}}function z(s){const a=e.readdirSync(s,{withFileTypes:!0});for(const i of a){const a=t.join(s,i.name);if(i.isDirectory())z(a);else if(i.isFile()&&!a.endsWith(".js")&&!a.endsWith(".js.map"))try{e.unlinkSync(a)}catch(e){}}}const q=["node_modules","__MACOSX"],$=[".DS_Store","Thumbs.db"];async function O(e,s,c=5){const n=[];for(const i of e){const{id:e,packages:c}=i,o=t.join(s,e);await a.mkdir(o,{recursive:!0});for(const e in c){const s=c[e],{code_url:a}=s,i=`${e}.txt`,r=t.join(o,i);n.push({code_url:a,filePath:r,packageName:e})}}await async function(e,s,t){const a=[],i=[];for(const c of s){const n=Promise.resolve().then(()=>t(c));if(a.push(n),e<=s.length){const s=n.then(()=>i.splice(i.indexOf(s),1));i.push(s),i.length>=e&&await Promise.race(i)}}return Promise.all(a)}(c,n,async e=>{const{code_url:s,filePath:t,packageName:c}=e;console.log(`start to download ${c}`);try{const e=await i(s);if(!e.ok)throw new Error(`Failed to fetch ${s}: ${e.statusText}`);const n=Buffer.from(await e.arrayBuffer());await a.writeFile(t,n),console.log(`finish to download ${c}`)}catch(e){console.error(`Error downloading ${c}:`,e)}})}var M={MinisPackagingInfoMap:{sbmg0cldak4bbz86dq:{ClientKey:"sbmg0cldak4bbz86dq",MinisStage:1,MinisURL:"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us",ODRPackages:'{"112":{"asset_md5":"9fe63b5fc31e9b4ccd1ee9fa9ed2913b","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/112_asset/9fe63b5fc31e9b4ccd1ee9fa9ed2913b.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/112_code/ebb1a6d341c0013a445f3dbf2600c9ae.txt","root":"subpackages/112","main":"subpackages/112/game.js"},"116":{"asset_md5":"690814b650564724a235a4f155cfe5e3","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/116_asset/690814b650564724a235a4f155cfe5e3.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/116_code/ddc0d442aef316e3e45cdbac9ccb88eb.txt","root":"subpackages/116","main":"subpackages/116/game.js"},"108":{"asset_md5":"b7b3aeafdd70aed9e77fb2f1554d6675","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/108_asset/b7b3aeafdd70aed9e77fb2f1554d6675.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/108_code/bdac7d266b4a15af9ca109d1673168a6.txt","root":"subpackages/108","main":"subpackages/108/game.js"},"game3":{"asset_md5":"bc4b09b1177a35fd8516b8fd5bb9b4e4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game3_asset/bc4b09b1177a35fd8516b8fd5bb9b4e4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game3_code/7f3c18114afd5b3d4fd8c39dd2481cf1.txt","root":"subpackages/game3","main":"subpackages/game3/game.js"},"game5":{"asset_md5":"6a75dd22dcf0d7c117cfcb03185696e6","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game5_asset/6a75dd22dcf0d7c117cfcb03185696e6.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game5_code/ed2c8ca8023418099d1232fdc6ac09ae.txt","root":"subpackages/game5","main":"subpackages/game5/game.js"},"main":{"asset_md5":"ecadb8f2a4138a73459e04a8a0c5f470","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/main_asset/ecadb8f2a4138a73459e04a8a0c5f470.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/main_code/3b7a7f0869e5392f1e244eef40ad8abe.txt","root":"subpackages/main","main":"subpackages/main/game.js"},"103":{"asset_md5":"069620ed43643956f6b6448c5d360471","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/103_asset/069620ed43643956f6b6448c5d360471.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/103_code/64a9caf580b6cd991888751e8ea51222.txt","root":"subpackages/103","main":"subpackages/103/game.js"},"other":{"asset_md5":"473ad574dee1813610988dba22fb0acc","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/other_asset/473ad574dee1813610988dba22fb0acc.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/other_code/003784b887ff99fb82d47899798e1ee0.txt","root":"subpackages/other","main":"subpackages/other/game.js"},"resSps":{"asset_md5":"4e72c930e3c8f77bda25be975e3c8019","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/resSps_asset/4e72c930e3c8f77bda25be975e3c8019.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/resSps_code/c6ac2ebde3c8c421ca75ff649280d90b.txt","root":"subpackages/resSps","main":"subpackages/resSps/game.js"},"111":{"asset_md5":"3452f073e30ecc6b3727f072a856068b","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/111_asset/3452f073e30ecc6b3727f072a856068b.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/111_code/3ca9e57876b82945faedce2ca0ab414e.txt","root":"subpackages/111","main":"subpackages/111/game.js"},"117":{"asset_md5":"60206e41273a6a1f6d21bbf634023962","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/117_asset/60206e41273a6a1f6d21bbf634023962.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/117_code/1a85a85048eca0d650ad4b8b198cefdc.txt","root":"subpackages/117","main":"subpackages/117/game.js"},"__GAME__":{"asset_md5":"dbee9c4f704800eb5bf482be01aecadc","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/__GAME___asset/dbee9c4f704800eb5bf482be01aecadc.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/__GAME___code/b42b2a68ce373eb274eedc22351aac02.txt","root":"","main":"game.js"},"jigsaw":{"asset_md5":"4782715dda5ea0d10897d866a84bca19","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/jigsaw_asset/4782715dda5ea0d10897d866a84bca19.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/jigsaw_code/799004f65797d4620549e0d4b27e3794.txt","root":"subpackages/jigsaw","main":"subpackages/jigsaw/game.js"},"newYear":{"asset_md5":"c2cb90fbaac0b4b86d1afa7caabd2f20","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/newYear_asset/c2cb90fbaac0b4b86d1afa7caabd2f20.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/newYear_code/a7d962b4f7a95e8114fcf00bbe27b8ac.txt","root":"subpackages/newYear","main":"subpackages/newYear/game.js"},"adapter":{"asset_md5":"a44f0cd93ad08da6db427c9dc3ab6fe4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/adapter_asset/a44f0cd93ad08da6db427c9dc3ab6fe4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/adapter_code/6c3090ba10c12ad637aefa993fe838fc.txt","root":"subpackages/adapter","main":"subpackages/adapter/game.js"},"101":{"asset_md5":"ad3f476d334e695c01f96eef67308f26","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/101_asset/ad3f476d334e695c01f96eef67308f26.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/101_code/dcb5f8547d5285c5e116a1726dc1ecdb.txt","root":"subpackages/101","main":"subpackages/101/game.js"},"102":{"asset_md5":"8b39712d9f0ae7549382da9041cc5f56","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/102_asset/8b39712d9f0ae7549382da9041cc5f56.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/102_code/b70fb4ea0d05b866a4a32ab4cae418c4.txt","root":"subpackages/102","main":"subpackages/102/game.js"},"110":{"asset_md5":"24da51736441ceb88b3d331775004551","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/110_asset/24da51736441ceb88b3d331775004551.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/110_code/f7c97cb3aed7acb4091feaf613c56dd3.txt","root":"subpackages/110","main":"subpackages/110/game.js"},"114":{"asset_md5":"c2f81aaf93311b9302995dba105d69d0","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/114_asset/c2f81aaf93311b9302995dba105d69d0.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/114_code/7574a218fb4053e2005497d84a56132c.txt","root":"subpackages/114","main":"subpackages/114/game.js"},"practice":{"asset_md5":"01d63b100e2cfb80d2d7568d11f346f8","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/practice_asset/01d63b100e2cfb80d2d7568d11f346f8.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/practice_code/2275ac8e93d1c5dd100427aa948cd23c.txt","root":"subpackages/practice","main":"subpackages/practice/game.js"},"zh_CN":{"asset_md5":"2b1c715a0c624c630acb29b4fdfcb240","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/zh_CN_asset/2b1c715a0c624c630acb29b4fdfcb240.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/zh_CN_code/bb44d46f30e57f29d20bf0678426c7c5.txt","root":"subpackages/zh_CN","main":"subpackages/zh_CN/game.js"},"107":{"asset_md5":"652c5385223642c4a19f823666c8a682","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/107_asset/652c5385223642c4a19f823666c8a682.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/107_code/9e717b17251bdc8eefb7b8135fcb6996.txt","root":"subpackages/107","main":"subpackages/107/game.js"},"113":{"asset_md5":"f53882a23326919614b0e9031d28ed39","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/113_asset/f53882a23326919614b0e9031d28ed39.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/113_code/666bb07aea74f0a408136c0ac16a04b8.txt","root":"subpackages/113","main":"subpackages/113/game.js"},"115":{"asset_md5":"6831f12abfeccc9ff720155d9ece2d53","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/115_asset/6831f12abfeccc9ff720155d9ece2d53.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/115_code/76b85e96a57c33c8ba9ff2f316290416.txt","root":"subpackages/115","main":"subpackages/115/game.js"},"118":{"asset_md5":"f787456ee45eccc4c13f20ee21c61858","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/118_asset/f787456ee45eccc4c13f20ee21c61858.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/118_code/e595f8ed244752fe2ddc72def8452cd5.txt","root":"subpackages/118","main":"subpackages/118/game.js"},"chapter":{"asset_md5":"9c4b6cb8dc0b39bfcb275efa0c6b8493","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/chapter_asset/9c4b6cb8dc0b39bfcb275efa0c6b8493.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/chapter_code/9ee175f09ca69c65b630cfee23c0e42c.txt","root":"subpackages/chapter","main":"subpackages/chapter/game.js"},"sound":{"asset_md5":"65c37817ffc86d942d1942443b5e3fb4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/sound_asset/65c37817ffc86d942d1942443b5e3fb4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/sound_code/2662512b5aaa6801b6742eb661c44828.txt","root":"subpackages/sound","main":"subpackages/sound/game.js"},"105":{"asset_md5":"82c8cde04fec4456ac6bab8adaef0028","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/105_asset/82c8cde04fec4456ac6bab8adaef0028.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/105_code/74e319194a7f5a204504fc3d83131112.txt","root":"subpackages/105","main":"subpackages/105/game.js"},"109":{"asset_md5":"926ebc459e3329a4d9cda7ddf9718bf4","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/109_asset/926ebc459e3329a4d9cda7ddf9718bf4.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/109_code/19cbdc3d88a72dbd0c5eb52e89e2635a.txt","root":"subpackages/109","main":"subpackages/109/game.js"},"game2":{"asset_md5":"9f1bb086f7587aada8312315f2dd94fc","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game2_asset/9f1bb086f7587aada8312315f2dd94fc.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game2_code/49d07e848cae8eb08428bfbf225a6f30.txt","root":"subpackages/game2","main":"subpackages/game2/game.js"},"game6":{"asset_md5":"b1407de78d19f60644b53436f1791ee6","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game6_asset/b1407de78d19f60644b53436f1791ee6.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game6_code/155651953a65a663f67b95475cd5c941.txt","root":"subpackages/game6","main":"subpackages/game6/game.js"},"prefab":{"asset_md5":"4c4ba8d442bae683575460e884ddba00","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/prefab_asset/4c4ba8d442bae683575460e884ddba00.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/prefab_code/7f037cd7f3b83e90765e749d28f32e3e.txt","root":"subpackages/prefab","main":"subpackages/prefab/game.js"},"104":{"asset_md5":"160c617e3e3adef19c94f57bc4520dba","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/104_asset/160c617e3e3adef19c94f57bc4520dba.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/104_code/ef86df3a24fca9ecffc55f7c33ff3747.txt","root":"subpackages/104","main":"subpackages/104/game.js"},"106":{"asset_md5":"b22052c8f569db2f00a2ae6fcb8872e6","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/106_asset/b22052c8f569db2f00a2ae6fcb8872e6.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/106_code/f13ed6653a2ebb49703d0bc882255cfd.txt","root":"subpackages/106","main":"subpackages/106/game.js"},"game4":{"asset_md5":"b98a8892ccb76e5999cbf7a8e81c4cb5","asset_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game4_asset/b98a8892ccb76e5999cbf7a8e81c4cb5.txt","code_url":"https://minis-sbmg0cldak4bbz86dq-vpkg-1.tiktok-minis.us/game4_code/ebbcf2b74998d340cb8cf85249748992.txt","root":"subpackages/game4","main":"subpackages/game4/game.js"}}',UpdatedTime:1756716749e3,AssetType:1}}};function D(s,a){f.info("start check project"),function(e,s){f.info("start check project size");const{build:{pkgSizeLimit:t},dev:a}=s,i=S(e,[]),c=Math.round(i/1048576),n=Math.round(t/1048576);if(i>t){const e=`Game size ${c}MB, must not exceed ${n}MB`;if(f.error(e),!a.enable)throw new Error(e);console.log("[1m[33m%s[0m",e)}else f.info(`Game size ${c}MB, check successfully!`),f.info("Game size check successfully!")}(s,a),function(s){f.info("start check config");const a=t.join(s,p);if(!e.existsSync(a)){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")}(s),function(s){f.info("start check project config");const a=t.join(s,l);if(!e.existsSync(a)){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 i;try{i=JSON.parse(e.readFileSync(a,"utf-8"))}catch(e){throw f.error("check project.config.json failed, your project.config.json is invalid json format"),e}if(!(null==i?void 0:i.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)}}(s),f.info("check project successfully")}function E(e,s){f.info("start check main package"),f.info("check game.js");const t=m.join(e,"game.js");if(!b.existsSync(t)){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:s}){f.info("start check main package size");const{build:t,dev:a}=s,i=S(e,["subpackages"]),c=Math.round(i/1048576),n=Math.round(t.pkgSizeLimit/1048576);if(i>t.pkgSizeLimit){const e=`Main package size ${c}MB, must not exceed ${n}MB`;if(!a.enable)throw f.error(e),new Error(e);console.log("[1m[33m%s[0m",e)}else f.info(`Main package size ${c}MB, check successfully!`)}({entryDir:e,config:s}),f.info("check main package successfully")}function F(s,a){const{build:i}=a||{},{subPkgSizeLimit:c}=i||{};f.info("start check subpackages in game.json");const n=t.join(s,p),o=JSON.parse(e.readFileSync(n,"utf-8")).subpackages||[];(null==o?void 0:o.length)?o.forEach(a=>{if(!a.name||!a.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=t.join(s,a.root);if(!e.existsSync(i)){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${a.name} entry dir, please check it`;throw f.error(e),new Error(e)}const n=a.root;if(!e.existsSync(t.join(s,n))){const e=`The subpackages configuration in game.json is invalid: can not find subpackage ${a.name} root, please check it`;throw f.error(e),new Error(e)}c&&function({entryDir:e,subPkgName:s,limit:a}){const i=t.join(e,"subpackages",s),c=S(i,[]),n=Math.round(c/1048576);if(n>a){const e=`${s} Sub package size ${n}MB, must not exceed ${a/1048576}MB`;throw f.error(e),new Error(e)}f.info(`${s} Sub package size ${n}MB, check successfully!`)}({entryDir:s,subPkgName:a.name,limit:c})}):f.info("subpackages is empty"),f.info("check subpackages in game.json successfully")}async function P({entryDir:e,config:s}){f.info("开始校验项目"),D(e,s),E(e,s),F(e,s),f.info("校验项目完成")}async function N(s,a){f.info("开始基于源代码中的 game.json 生成 packageConfig.json");const i=t.join(s,p),c=JSON.parse(e.readFileSync(i,"utf-8")),n={};n[_]={url:"",md5:"",root:"",main:"game.js",output:`${_}${h}`},function(e){if(!e||0===e.length)return[];function s(e){return String(e).replace(/^\/+|\/+$/g,"")}function t(e){return a(e)?`${e}game.js`:e}function a(e){return e.endsWith("/")||!/\.[a-zA-Z0-9]+$/.test(e)}function i(e){return e.replace(/^\/+|\/+$/g,"")}return e.map(e=>{const a=s(e.name);return{main:t(e.root),root:i(e.root),name:a}})}(c.subpackages).forEach(({name:e,root:s,main:t})=>{n[e]={url:"",md5:"",root:s,main:t,output:`${e}${h}`}});const o={packages:n};return Object.keys(n).forEach(a=>{const i=t.join(s,n[a].main);if(!e.existsSync(i)){const e=`开发者源代码包中的子包 ${a} 中的 main 文件 ${n[a].main} 不存在`;throw f.error(e),new Error(e)}}),e.existsSync(a)&&e.rmSync(a,{recursive:!0}),e.mkdirSync(a),e.writeFileSync(t.join(a,k),JSON.stringify(o,null,2)),f.info("基于源代码中的 game.json 生成 packageConfig.json 成功"),o}function T(e){const s=[];return Object.values(e).forEach(e=>{Object.values(e).forEach(e=>{s.push(...e)})}),s}function C({entry:s,root:a,exts:i=[".js",".ts",".jsx",".tsx"],excludeDirs:c=[]}){let n=[];if(!e.existsSync(s))return n;const o=e.readdirSync(s,{withFileTypes:!0});for(const e of o){const o=t.join(s,e.name);if(e.isDirectory()){const e=t.relative(a,o).replace(/\\/g,"/");if(c.find(s=>s===e))continue;n=n.concat(C({entry:o,root:a,exts:i,excludeDirs:c}))}else i.some(s=>e.name.endsWith(s))&&n.push(o)}return n}function R(s,a){const i=function(s){return e.readdirSync(s).filter(a=>{const i=t.join(s,a);return e.statSync(i).isDirectory()&&!["node_modules","dist","build","output"].includes(a)}).map(e=>e+"/")}(s),o=n.sync("**/*.js",{cwd:s,ignore:[]}),r=[];o.forEach(a=>{const n=t.join(s,a),o=e.readFileSync(n,"utf-8");let d;try{d=c.parse(o,{ecmaVersion:"latest",sourceType:"module"})}catch(e){return}L(d,a=>{if("CallExpression"===a.type&&"Identifier"===a.callee.type&&("require"===a.callee.name||"requireAsync"===a.callee.name)&&a.arguments.length>0&&"Literal"===a.arguments[0].type){const c=a.arguments[0].value;let o=null;if(c.startsWith("."))o=t.resolve(t.dirname(n),c);else if(c.startsWith("/"))o=t.resolve(s,"."+c);else for(const e of i)if(c.startsWith(e)){o=t.resolve(s,c);break}o&&!function(s){if(e.existsSync(s)&&e.statSync(s).isFile())return!0;const t=[".js",".ts",".jsx",".tsx"];for(const a of t)if(e.existsSync(s+a)&&e.statSync(s+a).isFile())return!0;return!1}(o)&&(o=null),r.push({file:t.relative(s,n),callee:a.callee.name,requirePath:c,resolvedPath:o?t.relative(s,o):""})}})});const d=r.map(e=>{let s="";Object.keys(a).forEach(t=>{e.resolvedPath&&e.resolvedPath.startsWith(a[t].root)&&(s=t)});let t="";return Object.keys(a).forEach(s=>{e.file.startsWith(a[s].root)&&(t=s)}),Object.assign(Object.assign({},e),{curModule:t,depModule:s})}),b=[".js",".ts",".jsx",".tsx"];return d.reduce((e,s)=>{return e[s.curModule]||(e[s.curModule]={}),s.curModule===s.depModule||s.resolvedPath&&(t=s.resolvedPath,b.some(e=>t.endsWith(e)))&&(e[s.curModule]=Object.assign(Object.assign({},e[s.curModule]),{[s.resolvedPath]:s.depModule})),e;var t},{})}function L(e,s){s(e);for(let t in e)if(e.hasOwnProperty(t)){const a=e[t];Array.isArray(a)?a.forEach(e=>e&&"string"==typeof e.type&&L(e,s)):a&&"string"==typeof a.type&&L(a,s)}}async function A({pkgRoot:s,destRoot:a,packedFiles:i,gameEntry:c,gameOutput:n,skipDirs:o=[],isRoot:r=!0}){if(!e.existsSync(s))return;const d=e.readdirSync(s,{withFileTypes:!0});for(const r of d){const d=t.join(s,r.name),b=t.join(a,r.name),m=t.join(s,r.name);if(r.isDirectory()){const e=t.relative(c,m).replace(/\\/g,"/");if(o.find(s=>s===e))continue}r.isDirectory()?A({pkgRoot:d,destRoot:b,gameOutput:n,packedFiles:i,gameEntry:c,skipDirs:o,isRoot:!1}):(u(t.dirname(b)),i.includes(t.relative(c,d).replace(/\\/g,"/"))||e.copyFileSync(d,b))}}function G(e){const s=[];return e&&s.push(...Object.values(e).map(e=>e.root)),s}async function J({gameEntry:s,gameOutput:a,config:i}){let c=Date.now();!function(s){!function s(a){let i;try{i=e.readdirSync(a)}catch(e){return void f.warn(`无法访问目录: ${a}, ${e.message}`)}i.forEach(i=>{const c=t.join(a,i);try{const t=e.statSync(c);t.isDirectory()?q.includes(i)?(e.rmSync(c,{recursive:!0,force:!0}),f.info(`已删除无用目录: ${c}`)):s(c):t.isFile()&&$.includes(i)&&(e.rmSync(c,{force:!0}),f.info(`已删除无用文件: ${c}`))}catch(e){f.warn(`无法处理: ${c}, ${e.message}`)}})}(s)}(s),f.info(`pack start,startTime:${c}`);const{packages:n}=await N(s,a),d=R(s,n),b=function(e,s){f.info("开始基于游戏源代码收集分包中的 JS 文件");const a={},i=Object.values(s).filter(e=>e.root&&""!==e.root).map(e=>e.root).filter(Boolean),c=Array.from(new Set(i));for(const i in s){const n=s[i];let o,r;"__GAME__"===i?(r=C({entry:e,root:e,exts:[".js",".ts",".jsx",".tsx"],excludeDirs:c}),o=e):(o=t.join(e,n.root),r=C({entry:o,root:o,exts:[".js",".ts",".jsx",".tsx"],excludeDirs:[]}));const d=[];r.forEach(s=>{d.push(t.relative(e,s).replace(/\\/g,"/"))});const b=s[i].root;a[i]={[b?`${b}/game.pack.js`:"game.pack.js"]:d}}return f.info("基于游戏源代码收集分包中的 JS 文件完成"),a}(s,n);return await async function(e,s,t){const a=[],i=[];for(const c of s){const n=Promise.resolve().then(()=>t(c));if(a.push(n),e<=s.length){const s=n.then(()=>{i.splice(i.indexOf(s),1)});i.push(s),i.length>=e&&await Promise.race(i)}}return Promise.all(a)}(10,Object.keys(b),async c=>{f.info(`开始基于游戏源代码打包,包名:${c}`);let m=Date.now();const k=n[c],l=t.join(s,k.root),p=t.join(a,c,k.root);await function({gameEntry:s,pkgName:a,allDeps:i,allMaps:c,pkgConfig:n,destRoot:d,config:b}){var m,g,k,l;let p=Date.now();f.info(`pack start,startTime:${p}`);const _=c[a],h={main:n.main,deps:i[a]||{},map:c[a]};u(d),e.writeFileSync(t.join(d,"moduleConfig.json"),JSON.stringify(h));for(const i in _){const c=_[i];if(!Array.isArray(c)||0===c.length)continue;const n=new o.Bundle({separator:"\n"});for(const a of c){const c=t.join(s,a);if(e.existsSync(c)&&t.basename(c)!==i&&/\.js$/i.test(a))try{const{code:s}=r.transformSync(e.readFileSync(c,"utf-8"),{loader:"js",format:"cjs",target:"es2015"}),t=a.replace(/\\/g,"/"),i=`define("game:${t}", ["require", "requireAsync", "module", "exports", "sandboxGlobal"], function(require, requireAsync, module, exports, sandboxGlobal){\nwith(sandboxGlobal){\n`,d="\n}\n});\n",b=new o(s,{filename:t});b.prepend(i),b.append(d),n.addSource({content:b,filename:t})}catch(e){f.error(`pack error, relPath:${a},e:${e}`)}}const u=t.join(d,t.basename(i)),p=n.toString();if(null===(m=null==b?void 0:b.dev)||void 0===m?void 0:m.enableSourcemap){const s=n.generateMap({hires:!0,includeContent:!0,file:t.basename(i)});if(null===(g=null==b?void 0:b.dev)||void 0===g?void 0:g.inlineSourcemap){const t=p+`\n//# sourceMappingURL=data:application/json;base64,${Buffer.from(s.toString()).toString("base64")}`+`\n//# sourceURL=${i}`;e.writeFileSync(u,t,"utf-8")}else{const a=`http://${null===(k=null==b?void 0:b.dev)||void 0===k?void 0:k.host}:${null===(l=null==b?void 0:b.dev)||void 0===l?void 0:l.port}/game/files`,c=t.basename(i)+".map",n=t.join(d,c);e.writeFileSync(n,s.toString(),"utf-8");const o=p+`\n//# sourceMappingURL=${a}/${i}.map`+`\n//# sourceURL=${i}`;e.writeFileSync(u,o,"utf-8")}}else{const s=p+`\n//# sourceURL=${i}`;e.writeFileSync(u,s,"utf-8")}f.info(`pack end,packName:${a}`)}}({gameEntry:s,allDeps:d,allMaps:b,pkgName:c,pkgConfig:k,destRoot:p,config:i}),await A({gameEntry:s,gameOutput:a,pkgRoot:l,destRoot:p,packedFiles:T(b),skipDirs:G(n)});g(t.join(a,c)),f.info(`基于游戏源代码打包分包完成,包名:${c}`),f.info(`基于游戏源代码打包分包耗时,包名:${c},耗时:${Date.now()-m}ms`)}),f.info(`pack end,duration:${Date.now()-c}ms`),{}}exports.buildOdrPkgs=async function({tempDir:s}){e.existsSync(s)&&e.rmSync(s,{recursive:!0}),e.mkdirSync(s);const a=function(){const{MinisPackagingInfoMap:e}=M;return Object.keys(e).map(s=>({id:s,packages:JSON.parse(e[s].ODRPackages)}))}();f.info("odr_games"),await O(a,s);const i={};for(const e of a){const{id:a}=e,c=t.join(s,a);i[a]=S(c)}const c={};x.forEach(e=>{c[e]=[]});const n={};for(const o of a){const{id:a,packages:r}=o,d=i[a];if(d>v)continue;let b=null;for(const n of x){const o=t.join(s,n);e.existsSync(o)||e.mkdirSync(o);if(c[n].reduce((e,s)=>e+i[s],0)+d<=v){const i=t.join(s,a),r=t.join(o,a);e.cpSync(i,r,{recursive:!0}),c[n].push(a),b=n;break}}if(b){n[a]={tag:b,packages:{}};for(const e in r){const s=r[e];f.info(`TTMGODRConfig[gameId].packages, ${s}, ${e}`),n[a].packages[e]={root:s.root,main:s.main,asset_md5:s.asset_md5,asset_url:s.asset_url}}console.log(`finish to build ${a}`),e.rmSync(t.join(s,a),{recursive:!0})}}const o=t.join(s,"TTMG_ODR_CONFIG.json");e.writeFileSync(o,JSON.stringify(n,null,2),"utf-8"),f.info(`TTMG_ODR_CONFIG.json 已生成: ${o}`)},exports.buildPkgs=async function(s){const{entry:a,output:i,build:c}=s;let n=Date.now();f.init(i,!0),f.info("TTMG_PACK_VERSION: 0.0.34"),f.info(`pack start, startTime:${n}`),await P({entryDir:a,config:s}),await J({config:s,gameEntry:a,gameOutput:i}),(null==c?void 0:c.enableOdr)&&await async function(s){let a=Date.now();f.info(`makeOdrPkgs start, ${a}`);const i=t.join(s,k),c=JSON.parse(e.readFileSync(i,"utf-8")),n=c.packages;for(const a in n){const i=t.join(s,a),c=t.join(s,`${a}${y}`),o=t.join(s,`${a}${j}`);u(c),u(o),e.cpSync(i,c,{recursive:!0}),e.cpSync(i,o,{recursive:!0}),z(c),w(o),n[`${a}${y}`]=Object.assign(Object.assign({},n[a]),{code_url:"",code_md5:"",output:`${a}${y}${h}`}),n[`${a}${j}`]=Object.assign(Object.assign({},n[a]),{asset_url:"",asset_md5:"",output:`${a}${j}${h}`})}e.writeFileSync(i,JSON.stringify(c,null,2)),f.info(`makeOdrPkgs end, ${Date.now()-a}ms`)}(i),f.info(`pack end:${Date.now()-n}ms`)},exports.debugPkgs=async function(s){const{entry:a,output:i,dev:{enableLog:c}}=s;f.init(i,c);try{await P({entryDir:a,config:s}),await J({gameEntry:a,gameOutput:i,config:s}),await async function(s){f.info("mergePkgs start");const a=t.join(s,k);let i;try{i=JSON.parse(e.readFileSync(a,"utf-8"))}catch(e){return void f.error(`读取配置文件失败: ${a}, err: ${e.message}`)}const c=i.packages;if(c&&"object"==typeof c){for(const a of Object.keys(c)){const i=t.join(s,a);if(!e.existsSync(i)){f.warn(`package 目录不存在: ${i}`);continue}const c=await e.promises.readdir(i);for(const a of c){const c=t.join(i,a),n=t.join(s,a);try{const s=await e.promises.stat(c);if(s.isFile())await e.promises.copyFile(c,n),await e.promises.rm(c,{recursive:!0,force:!0});else if(s.isDirectory())await e.promises.cp(c,n,{recursive:!0}),await e.promises.rm(c,{recursive:!0,force:!0});else if(s.isSymbolicLink()){const t=e.readlinkSync(c);e.existsSync(n)&&await e.promises.rm(n,{force:!0}),await e.promises.symlink(t,n,s.isDirectory()?"dir":"file")}f.info(`合并文件/目录: ${c} -> ${n}`)}catch(e){f.error(`处理文件失败: ${c}, err: ${e.message}`)}}0===e.readdirSync(i).length&&(e.rmSync(i,{recursive:!0,force:!0}),f.info(`已删除 package 目录: ${i}`))}f.info("mergePkgs 完成")}else f.error("配置文件中 packages 字段无效, 请检查配置")}(i);const c=e.readFileSync(t.join(i,k),"utf-8");return{isSuccess:!0,packages:JSON.parse(c).packages}}catch(e){return{isSuccess:!1,errorMsg:e.message}}},exports.writeOdrConfig=async function(e){f.info("writeOdrPackagesConfig start");const s=m.join(e,k),t=JSON.parse(b.readFileSync(s,"utf-8")),a=t.packages,i=t.odr_packages||{};for(const e in a){if(e.endsWith(y)){const s=a[e],t=e.split(y)[0];i[t]||(i[t]={}),i[t]=Object.assign(Object.assign({},i[t]),{root:s.root,code_output:s.output,code_md5:s.md5,main:s.main}),delete a[e]}if(e.endsWith(j)){const s=e.split(j)[0],t=a[e];i[s]=Object.assign(Object.assign({},i[s]),{root:t.root,asset_output:t.output,asset_md5:t.md5}),delete a[e]}}t.odr_packages=i,b.writeFileSync(s,JSON.stringify(t,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/utils.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/partition.ts","../src/libs/makePkgs/index.ts","../src/utils/asyncPool.ts","../src/libs/makePkgs/pack.ts","../src/libs/odrPkgs/buildOdrPkgs.ts","../src/libs/odrPkgs/getConfigs.ts","../src/libs/buildPkgs/index.ts","../src/libs/odrPkgs/makeOdrPkgs.ts","../src/libs/debugPkgs/index.ts","../src/libs/mergePkgs/index.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 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_high_channel',\n 'ttmg_odr_middle_channel',\n 'ttmg_odr_normal_channel',\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';\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","import { logger } from '@utils';\nimport { ODRPackages } from '@typings';\n\n// export 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// export async function fetchAllGames(): Promise<Array<{\n// id: string;\n// packages: ODRPackages;\n// }>> {\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\nimport fs from 'fs/promises';\nimport path from 'path';\nimport fetch from 'node-fetch';\n\nexport async function downloadAndSaveCodesAsync(\n games,\n tempDir,\n concurrency = 5, // 并发数\n) {\n // 1. 生成所有下载任务\n const tasks = [];\n\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameDir = path.join(tempDir, gameId);\n await fs.mkdir(gameDir, { recursive: true });\n\n for (const packageName in packages) {\n const packageInfo = packages[packageName];\n const { code_url } = packageInfo;\n const fileName = `${packageName}.txt`;\n const filePath = path.join(gameDir, fileName);\n\n tasks.push({\n code_url,\n filePath,\n packageName,\n });\n }\n }\n\n // 2. 用并发池执行所有任务\n await asyncPool(concurrency, tasks, async task => {\n const { code_url, filePath, packageName } = task;\n console.log(`start to download ${packageName}`);\n try {\n const res = await fetch(code_url);\n if (!res.ok)\n throw new Error(`Failed to fetch ${code_url}: ${res.statusText}`);\n const buffer = Buffer.from(await res.arrayBuffer());\n await fs.writeFile(filePath, buffer);\n console.log(`finish to download ${packageName}`);\n } catch (err) {\n console.error(`Error downloading ${packageName}:`, err);\n }\n });\n}\n\nasync function asyncPool(poolLimit, array, iteratorFn) {\n const ret = [];\n const executing = [];\n for (const item of array) {\n const p = Promise.resolve().then(() => iteratorFn(item));\n ret.push(p);\n if (poolLimit <= array.length) {\n const e = p.then(() => executing.splice(executing.indexOf(e), 1));\n executing.push(e);\n if (executing.length >= poolLimit) {\n await Promise.race(executing);\n }\n }\n }\n return Promise.all(ret);\n}\n","import path from 'path';\nimport fs from 'fs';\nimport { BuildConfig } 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: BuildConfig) {\n logger.info('start check project size');\n const {\n build: { pkgSizeLimit },\n dev,\n } = config;\n const gameSize = getDirSizeSync(entryDir, []);\n const gameSizeMB = Math.round(gameSize / (1024 * 1024));\n const limitMB = Math.round(pkgSizeLimit / (1024 * 1024));\n if (gameSize > pkgSizeLimit) {\n const errMsg = `Game size ${gameSizeMB}MB, must not exceed ${limitMB}MB`;\n logger.error(errMsg);\n if (!dev.enable) {\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: BuildConfig) {\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 { BuildConfig } 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: BuildConfig;\n}) {\n logger.info('start check main package size');\n const { build, dev } = config;\n const mainPkgSize = getDirSizeSync(entryDir, ['subpackages']);\n const mainPkgSizeMB = Math.round(mainPkgSize / (1024 * 1024));\n const limitMB = Math.round(build.pkgSizeLimit / (1024 * 1024));\n if (mainPkgSize > build.pkgSizeLimit) {\n const errMsg = `Main package size ${mainPkgSizeMB}MB, must not exceed ${limitMB}MB`;\n if (!dev.enable) {\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: BuildConfig) {\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 { BuildConfig } from '@typings';\nimport { GAME_ORIGIN_CONFIG_FILE_NAME } from '@constants';\nimport path from 'path';\nimport fs from 'fs';\n\nexport function checkSubpackages(entryDir: string, config?: BuildConfig) {\n const { build } = config || {};\n const { subPkgSizeLimit } = build || {};\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 (subPkgSizeLimit) {\n checkSubPackageSize({\n entryDir,\n subPkgName: sub.name,\n limit: subPkgSizeLimit,\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 { BuildConfig } 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: BuildConfig;\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 `${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 return root.replace(/^\\/+|\\/+$/g, '');\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// 递归收集 JS 文件,但可排除指定目录\nfunction collectPkgJsFiles({\n entry,\n root,\n exts = ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs = [],\n}: {\n entry: string;\n root: string;\n exts: string[];\n excludeDirs: string[];\n}) {\n let files = [];\n if (!fs.existsSync(entry)) return files;\n const entries = fs.readdirSync(entry, { withFileTypes: true });\n for (const item of entries) {\n const fullPath = path.join(entry, item.name);\n if (item.isDirectory()) {\n /**\n * 计算 item 与 root 的相对路径\n */\n const relativePath = path.relative(root, fullPath);\n if (excludeDirs.find(dir => dir === relativePath)) {\n console.log('collectPkgJsFiles skip dir', relativePath);\n continue; // 跳过排除目录\n }\n files = files.concat(\n collectPkgJsFiles({\n entry: fullPath,\n root: root,\n exts: exts,\n excludeDirs,\n }),\n ); // 只排除一级\n } else if (exts.some(ext => item.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) // 只提取一级目录名\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 = collectPkgJsFiles({\n entry: entryDir,\n root: entryDir,\n exts: ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs: uniqueSubRoots,\n });\n srcRoot = entryDir;\n } else {\n // 分包:只收集自己目录\n srcRoot = path.join(entryDir, pkg.root);\n jsFiles = collectPkgJsFiles({\n entry: srcRoot,\n root: srcRoot,\n exts: ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs: [],\n });\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 path from 'path';\nimport fs from 'fs';\nimport { ensureDirSync } from '@utils';\n\nexport async 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 const fullPath = path.join(pkgRoot, entry.name);\n // 只在第一层判断是否跳过分包目录\n if (entry.isDirectory()) {\n const relativePath = path.relative(gameEntry, fullPath);\n if (skipDirs.find(dir => dir === relativePath)) {\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 path from 'path';\nimport { BuildConfig, ProjectConfig } from '@typings';\nimport { logger, removeEmptyDir, removeSystemUseless, asyncPool } from '@utils';\nimport { setup } from './setup';\nimport { flattenMaps, collectMaps } from './collectMaps';\nimport { collectDeps } from './collectDeps';\nimport { pack } from './pack';\nimport { partition } from './partition';\n\nfunction getSkipDirs(packages: ProjectConfig['packages']\n) {\n const skipDirs = [];\n if (packages) {\n skipDirs.push(...Object.values(packages).map(item => item.root));\n }\n return skipDirs;\n}\n\nexport async function makePkgs({\n gameEntry,\n gameOutput,\n config,\n}: {\n gameEntry: string;\n gameOutput: string;\n config: BuildConfig;\n}) {\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 /**\n * 基于 asyncPool 并发打包\n */\n\n await asyncPool(10, Object.keys(allMaps), async pkgName => {\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 await pack({\n gameEntry,\n allDeps,\n allMaps,\n pkgName,\n pkgConfig,\n destRoot,\n config,\n });\n\n /**\n * 分包\n */\n await partition({\n gameEntry,\n gameOutput,\n pkgRoot,\n destRoot,\n packedFiles: flattenMaps(allMaps),\n // 这里写的不对,应该是 每一个分包都要跳过\n skipDirs: getSkipDirs(packages),\n });\n\n const pkgRootDir = path.join(gameOutput, pkgName);\n removeEmptyDir(pkgRootDir);\n logger.info(`基于游戏源代码打包分包完成,包名:${pkgName}`);\n logger.info(\n `基于游戏源代码打包分包耗时,包名:${pkgName},耗时:${Date.now() - startTime\n }ms`,\n );\n });\n\n\n logger.info(`pack end,duration:${Date.now() - startTime}ms`);\n\n return pkgOutput;\n}\n","export async function asyncPool<T, R>(\n poolLimit: number,\n array: T[],\n iteratorFn: (item: T) => Promise<R>,\n): Promise<R[]> {\n const ret: Promise<R>[] = [];\n const executing: Promise<void>[] = [];\n for (const item of array) {\n // 创建处理当前 item 的 Promise\n const p = Promise.resolve().then(() => iteratorFn(item));\n ret.push(p);\n\n if (poolLimit <= array.length) {\n // 包装 Promise,处理完成后从 executing 中移除\n const e = p.then(() => {\n executing.splice(executing.indexOf(e), 1);\n });\n executing.push(e);\n\n // 达到并发限制时,等待任意一个任务完成\n if (executing.length >= poolLimit) {\n await Promise.race(executing);\n }\n }\n }\n // 等待所有任务完成\n return Promise.all(ret);\n}\n","import fs from 'fs';\nimport path from 'path';\nimport MagicString, { Bundle } from 'magic-string';\nimport { transformSync } from 'esbuild'\nimport { GAME_MODULE_CONFIG_FILE_NAME } from '@constants';\nimport { BuildConfig } from '@typings';\nimport { logger, ensureDirSync } from '@utils';\n\nexport function pack({\n gameEntry,\n pkgName,\n allDeps,\n allMaps,\n pkgConfig,\n destRoot,\n config,\n}: {\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 config: BuildConfig;\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 if (/\\.js$/i.test(relPath)) {\n try {\n const { code } = transformSync(fs.readFileSync(absPath, 'utf-8'), {\n loader: 'js',\n format: 'cjs',\n target: 'es2015',\n })\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 const fileMagic = new MagicString(code, { filename: fileId });\n fileMagic.prepend(defineHeader);\n fileMagic.append(defineFooter);\n concat.addSource({ content: fileMagic, filename: fileId });\n } catch (e) {\n logger.error(`pack error, relPath:${relPath},e:${e}`);\n }\n }\n }\n const outFile = path.join(destRoot, path.basename(packFileName));\n const bundledCode = concat.toString();\n\n if (config?.dev?.enableSourcemap) {\n const map = concat.generateMap({\n hires: true,\n includeContent: true,\n file: path.basename(packFileName), // 推荐用最终输出文件名\n });\n /**\n * 如果是走内敛\n */\n if (config?.dev?.inlineSourcemap) {\n // 生成 base 64\n const base64 = Buffer.from(map.toString()).toString('base64');\n const sourcemapContent = `data:application/json;base64,${base64}`;\n // 4. 写入 sourcemap 内容\n const fileContent =\n bundledCode +\n `\\n//# sourceMappingURL=${sourcemapContent}` + // 只写文件名即可,和 JS 文件同目录\n `\\n//# sourceURL=${packFileName}`;\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n } else {\n const sourcemapBaseUrl = `http://${config?.dev?.host}:${config?.dev?.port}/game/files`;\n // 1. 生成 sourcemap 文件路径\n const mapFileName = path.basename(packFileName) + '.map';\n const mapFilePath = path.join(destRoot, mapFileName);\n // 2. 写入 sourcemap 文件\n fs.writeFileSync(mapFilePath, map.toString(), 'utf-8');\n // 3. JS 文件末尾加外链注释\n const fileContent =\n bundledCode +\n `\\n//# sourceMappingURL=${sourcemapBaseUrl}/${packFileName}.map` + // 只写文件名即可,和 JS 文件同目录\n `\\n//# sourceURL=${packFileName}`;\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n }\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","/**\n * 合并请求拆包\n */\nimport fs from 'fs';\nimport path from 'path';\nimport { getDirSizeSync, logger } from '@utils';\nimport { downloadAndSaveCodesAsync } from './utils';\nimport { AVAILABLE_ODR_TAG_LIST, ONE_ODR_PKG_LIMIT } from '@constants';\nimport { getOdrConfigs } from './getConfigs';\n\nexport async function buildOdrPkgs({ tempDir }: { tempDir: string }) {\n if (fs.existsSync(tempDir)) {\n fs.rmSync(tempDir, { recursive: true });\n }\n fs.mkdirSync(tempDir);\n const games = getOdrConfigs();\n logger.info('odr_games');\n await downloadAndSaveCodesAsync(games, tempDir);\n const gamesContentSize = {};\n for (const game of games) {\n const { id: gameId } = game;\n const gameDir = path.join(tempDir, 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(tempDir, 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(tempDir, 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 console.log(`finish to build ${gameId}`);\n fs.rmSync(path.join(tempDir, gameId), { recursive: true });\n }\n // 写入 JSON 文件\n const configPath = path.join(tempDir, '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","import data from './config.json';\n\nexport function getOdrConfigs() {\n const { MinisPackagingInfoMap } = data;\n return Object.keys(MinisPackagingInfoMap).map(clientKey => {\n return {\n id: clientKey,\n packages: JSON.parse(MinisPackagingInfoMap[clientKey].ODRPackages),\n };\n });\n}\n","import { logger } from '@utils';\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, build } = 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 });\n /**\n * 打包\n */\n await makePkgs({\n config,\n gameEntry: entryDir,\n gameOutput: outputDir,\n });\n\n if (build?.enableOdr) {\n /**\n * 等待文件全部读写完成后\n */\n await makeOdrPkgs(outputDir);\n /**\n * 分拆 odr 包\n */\n }\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';\nimport { logger } from '@utils';\nimport { GAME_PACK_CONFIG_FILE_NAME } 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 dev: { enableLog },\n } = config;\n logger.init(outputDir, enableLog);\n try {\n /**\n * 校验\n */\n await checkPkgs({\n entryDir,\n config,\n });\n /**\n * 打包\n */\n await makePkgs({\n gameEntry: entryDir,\n gameOutput: outputDir,\n config,\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 // 复制好了之后就删除\n fs.rmSync(srcPath, { force: true });\n } else if (stat.isDirectory()) {\n fs.cpSync(srcPath, destPath, { recursive: true });\n // 复制好了之后就删除\n fs.rmSync(srcPath, { recursive: true, force: true });\n } else if (stat.isSymbolicLink()) {\n console.log('symbolic link', srcPath, destPath);\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 * 如果最后 pkgDir 为空,就删除 \n */\n if (fs.readdirSync(pkgDir).length === 0) {\n fs.rmSync(pkgDir, { recursive: true, force: true });\n logger.info(`已删除 package 目录: ${pkgDir}`);\n }\n }\n\n logger.info('mergePkgs 完成');\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","getDirSizeSync","filterDirs","totalSize","entries","withFileTypes","entry","name","includes","isFile","size","deleteJsFilesSync","endsWith","unlinkSync","err","deleteNoJsFilesSync","uselessDirs","uselessFiles","async","downloadAndSaveCodesAsync","games","tempDir","concurrency","tasks","game","id","gameId","packages","gameDir","mkdir","packageName","packageInfo","code_url","fileName","filePath","push","poolLimit","array","iteratorFn","ret","executing","item","p","Promise","resolve","then","length","e","splice","indexOf","race","all","asyncPool","task","console","res","fetch","ok","Error","statusText","buffer","Buffer","from","arrayBuffer","writeFile","checkProject","entryDir","config","build","pkgSizeLimit","dev","gameSize","gameSizeMB","Math","round","limitMB","errMsg","enable","checkProjectSize","gameJsonPath","checkGameJson","gameJsonConfigPath","gameJsonConfig","JSON","parse","readFileSync","appid","checkProjectConfig","checkMainPackage","gameJsPath","mainPkgSize","mainPkgSizeMB","checkMainPackageSize","checkSubpackages","subPkgSizeLimit","subpackages","forEach","sub","root","subpackageEntryDir","subpackageRoot","subPkgName","limit","subPkgDir","subPkgSize","sizeMB","checkSubPackageSize","checkPkgs","setup","outputDir","gameJson","url","md5","main","configs","normalizeName","String","replace","getMainFromRoot","str","test","normalizeRoot","map","transformSubPackages","result","Object","keys","pkg","pkgPath","writeFileSync","stringify","flattenMaps","allMaps","list","values","arr","collectPkgJsFiles","exts","excludeDirs","relativePath","relative","find","concat","some","ext","collectDeps","gameEntry","rootPrefixes","filter","getRootPrefixes","glob","sync","cwd","ignore","requireCalls","code","ast","acorn","ecmaVersion","sourceType","walkAst","node","type","callee","arguments","requirePath","value","resolvedPath","startsWith","dirname","prefix","fileExistsWithExtensions","newRequireCalls","i","depModule","curModule","assign","reduce","acc","cur","f","cb","key","hasOwnProperty","child","Array","isArray","c","partition","pkgRoot","destRoot","packedFiles","gameOutput","skipDirs","isRoot","srcPath","destPath","copyFileSync","getSkipDirs","makePkgs","startTime","Date","now","removeUselessDirs","message","stat","force","removeSystemUseless","allDeps","subRoots","Boolean","uniqueSubRoots","Set","pkgName","srcRoot","jsFiles","absPath","packNameRoot","collectMaps","pkgConfig","packMap","moduleConfig","deps","packFileName","relPaths","Bundle","separator","relPath","basename","transformSync","loader","target","fileId","defineHeader","defineFooter","fileMagic","MagicString","prepend","append","addSource","content","outFile","bundledCode","toString","_a","enableSourcemap","generateMap","hires","includeContent","_b","inlineSourcemap","fileContent","sourcemapBaseUrl","_c","host","_d","port","mapFileName","mapFilePath","pack","MinisPackagingInfoMap","data","clientKey","ODRPackages","getOdrConfigs","gamesContentSize","odrTagGameMap","tag","TTMGODRConfig","assignedTag","odrTag","odrTagDir","sum","gid","srcDir","destDir","cpSync","asset_md5","asset_url","configPath","enableOdr","outputPath","pkgOutput","codeOutput","assetOutput","code_md5","makeOdrPkgs","gamePackConfigPath","gamePackConfig","pkgDir","isSymbolicLink","linkTarget","readlinkSync","symlinkSync","mergePkgs","packageConfig","isSuccess","errorMsg","odrPackages","odr_packages","relatedPkgName","split","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/utils.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/partition.ts","../src/libs/makePkgs/index.ts","../src/utils/asyncPool.ts","../src/libs/makePkgs/pack.ts","../src/libs/odrPkgs/buildOdrPkgs.ts","../src/libs/odrPkgs/getConfigs.ts","../src/libs/buildPkgs/index.ts","../src/libs/odrPkgs/makeOdrPkgs.ts","../src/libs/debugPkgs/index.ts","../src/libs/mergePkgs/index.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 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_high_channel',\n 'ttmg_odr_middle_channel',\n 'ttmg_odr_normal_channel',\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';\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","import { logger } from '@utils';\nimport { ODRPackages } from '@typings';\n\n// export 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// export async function fetchAllGames(): Promise<Array<{\n// id: string;\n// packages: ODRPackages;\n// }>> {\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\nimport fs from 'fs/promises';\nimport path from 'path';\nimport fetch from 'node-fetch';\n\nexport async function downloadAndSaveCodesAsync(\n games,\n tempDir,\n concurrency = 5, // 并发数\n) {\n // 1. 生成所有下载任务\n const tasks = [];\n\n for (const game of games) {\n const { id: gameId, packages } = game;\n const gameDir = path.join(tempDir, gameId);\n await fs.mkdir(gameDir, { recursive: true });\n\n for (const packageName in packages) {\n const packageInfo = packages[packageName];\n const { code_url } = packageInfo;\n const fileName = `${packageName}.txt`;\n const filePath = path.join(gameDir, fileName);\n\n tasks.push({\n code_url,\n filePath,\n packageName,\n });\n }\n }\n\n // 2. 用并发池执行所有任务\n await asyncPool(concurrency, tasks, async task => {\n const { code_url, filePath, packageName } = task;\n console.log(`start to download ${packageName}`);\n try {\n const res = await fetch(code_url);\n if (!res.ok)\n throw new Error(`Failed to fetch ${code_url}: ${res.statusText}`);\n const buffer = Buffer.from(await res.arrayBuffer());\n await fs.writeFile(filePath, buffer);\n console.log(`finish to download ${packageName}`);\n } catch (err) {\n console.error(`Error downloading ${packageName}:`, err);\n }\n });\n}\n\nasync function asyncPool(poolLimit, array, iteratorFn) {\n const ret = [];\n const executing = [];\n for (const item of array) {\n const p = Promise.resolve().then(() => iteratorFn(item));\n ret.push(p);\n if (poolLimit <= array.length) {\n const e = p.then(() => executing.splice(executing.indexOf(e), 1));\n executing.push(e);\n if (executing.length >= poolLimit) {\n await Promise.race(executing);\n }\n }\n }\n return Promise.all(ret);\n}\n","import path from 'path';\nimport fs from 'fs';\nimport { BuildConfig } 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: BuildConfig) {\n logger.info('start check project size');\n const {\n build: { pkgSizeLimit },\n dev,\n } = config;\n const gameSize = getDirSizeSync(entryDir, []);\n const gameSizeMB = Math.round(gameSize / (1024 * 1024));\n const limitMB = Math.round(pkgSizeLimit / (1024 * 1024));\n if (gameSize > pkgSizeLimit) {\n const errMsg = `Game size ${gameSizeMB}MB, must not exceed ${limitMB}MB`;\n logger.error(errMsg);\n if (!dev.enable) {\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: BuildConfig) {\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 { BuildConfig } 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: BuildConfig;\n}) {\n logger.info('start check main package size');\n const { build, dev } = config;\n const mainPkgSize = getDirSizeSync(entryDir, ['subpackages']);\n const mainPkgSizeMB = Math.round(mainPkgSize / (1024 * 1024));\n const limitMB = Math.round(build.pkgSizeLimit / (1024 * 1024));\n if (mainPkgSize > build.pkgSizeLimit) {\n const errMsg = `Main package size ${mainPkgSizeMB}MB, must not exceed ${limitMB}MB`;\n if (!dev.enable) {\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: BuildConfig) {\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 { BuildConfig } from '@typings';\nimport { GAME_ORIGIN_CONFIG_FILE_NAME } from '@constants';\nimport path from 'path';\nimport fs from 'fs';\n\nexport function checkSubpackages(entryDir: string, config?: BuildConfig) {\n const { build } = config || {};\n const { subPkgSizeLimit } = build || {};\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 (subPkgSizeLimit) {\n checkSubPackageSize({\n entryDir,\n subPkgName: sub.name,\n limit: subPkgSizeLimit,\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 { BuildConfig } 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: BuildConfig;\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 `${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 return root.replace(/^\\/+|\\/+$/g, '');\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// 递归收集 JS 文件,但可排除指定目录\nfunction collectPkgJsFiles({\n entry,\n root,\n exts = ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs = [],\n}: {\n entry: string;\n root: string;\n exts: string[];\n excludeDirs: string[];\n}) {\n let files = [];\n if (!fs.existsSync(entry)) return files;\n const entries = fs.readdirSync(entry, { withFileTypes: true });\n for (const item of entries) {\n const fullPath = path.join(entry, item.name);\n if (item.isDirectory()) {\n /**\n * 计算 item 与 root 的相对路径\n */\n /**\n * 讲 \\ 替换为 /\n */\n const relativePath = path.relative(root, fullPath).replace(/\\\\/g, '/');\n if (excludeDirs.find(dir => dir === relativePath)) {\n continue; // 跳过排除目录\n }\n files = files.concat(\n collectPkgJsFiles({\n entry: fullPath,\n root: root,\n exts: exts,\n excludeDirs,\n }),\n ); // 只排除一级\n } else if (exts.some(ext => item.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) // 只提取一级目录名\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 = collectPkgJsFiles({\n entry: entryDir,\n root: entryDir,\n exts: ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs: uniqueSubRoots,\n });\n srcRoot = entryDir;\n } else {\n // 分包:只收集自己目录\n srcRoot = path.join(entryDir, pkg.root);\n jsFiles = collectPkgJsFiles({\n entry: srcRoot,\n root: srcRoot,\n exts: ['.js', '.ts', '.jsx', '.tsx'],\n excludeDirs: [],\n });\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 path from 'path';\nimport fs from 'fs';\nimport { ensureDirSync } from '@utils';\n\nexport async 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 const fullPath = path.join(pkgRoot, entry.name);\n // 只在第一层判断是否跳过分包目录\n if (entry.isDirectory()) {\n const relativePath = path.relative(gameEntry, fullPath).replace(/\\\\/g, '/');\n if (skipDirs.find(dir => dir === relativePath)) {\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).replace(/\\\\/g, '/'))) {\n fs.copyFileSync(srcPath, destPath);\n }\n }\n }\n}\n","import path from 'path';\nimport { BuildConfig, ProjectConfig } from '@typings';\nimport { logger, removeEmptyDir, removeSystemUseless, asyncPool } from '@utils';\nimport { setup } from './setup';\nimport { flattenMaps, collectMaps } from './collectMaps';\nimport { collectDeps } from './collectDeps';\nimport { pack } from './pack';\nimport { partition } from './partition';\n\nfunction getSkipDirs(packages: ProjectConfig['packages']\n) {\n const skipDirs = [];\n if (packages) {\n skipDirs.push(...Object.values(packages).map(item => item.root));\n }\n return skipDirs;\n}\n\nexport async function makePkgs({\n gameEntry,\n gameOutput,\n config,\n}: {\n gameEntry: string;\n gameOutput: string;\n config: BuildConfig;\n}) {\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 /**\n * 基于 asyncPool 并发打包\n */\n\n await asyncPool(10, Object.keys(allMaps), async pkgName => {\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 await pack({\n gameEntry,\n allDeps,\n allMaps,\n pkgName,\n pkgConfig,\n destRoot,\n config,\n });\n\n /**\n * 分包\n */\n await partition({\n gameEntry,\n gameOutput,\n pkgRoot,\n destRoot,\n packedFiles: flattenMaps(allMaps),\n // 这里写的不对,应该是 每一个分包都要跳过\n skipDirs: getSkipDirs(packages),\n });\n\n const pkgRootDir = path.join(gameOutput, pkgName);\n removeEmptyDir(pkgRootDir);\n logger.info(`基于游戏源代码打包分包完成,包名:${pkgName}`);\n logger.info(\n `基于游戏源代码打包分包耗时,包名:${pkgName},耗时:${Date.now() - startTime\n }ms`,\n );\n });\n\n\n logger.info(`pack end,duration:${Date.now() - startTime}ms`);\n\n return pkgOutput;\n}\n","export async function asyncPool<T, R>(\n poolLimit: number,\n array: T[],\n iteratorFn: (item: T) => Promise<R>,\n): Promise<R[]> {\n const ret: Promise<R>[] = [];\n const executing: Promise<void>[] = [];\n for (const item of array) {\n // 创建处理当前 item 的 Promise\n const p = Promise.resolve().then(() => iteratorFn(item));\n ret.push(p);\n\n if (poolLimit <= array.length) {\n // 包装 Promise,处理完成后从 executing 中移除\n const e = p.then(() => {\n executing.splice(executing.indexOf(e), 1);\n });\n executing.push(e);\n\n // 达到并发限制时,等待任意一个任务完成\n if (executing.length >= poolLimit) {\n await Promise.race(executing);\n }\n }\n }\n // 等待所有任务完成\n return Promise.all(ret);\n}\n","import fs from 'fs';\nimport path from 'path';\nimport MagicString, { Bundle } from 'magic-string';\nimport { transformSync } from 'esbuild'\nimport { GAME_MODULE_CONFIG_FILE_NAME } from '@constants';\nimport { BuildConfig } from '@typings';\nimport { logger, ensureDirSync } from '@utils';\n\nexport function pack({\n gameEntry,\n pkgName,\n allDeps,\n allMaps,\n pkgConfig,\n destRoot,\n config,\n}: {\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 config: BuildConfig;\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 if (/\\.js$/i.test(relPath)) {\n try {\n const { code } = transformSync(fs.readFileSync(absPath, 'utf-8'), {\n loader: 'js',\n format: 'cjs',\n target: 'es2015',\n })\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 const fileMagic = new MagicString(code, { filename: fileId });\n fileMagic.prepend(defineHeader);\n fileMagic.append(defineFooter);\n concat.addSource({ content: fileMagic, filename: fileId });\n } catch (e) {\n logger.error(`pack error, relPath:${relPath},e:${e}`);\n }\n }\n }\n const outFile = path.join(destRoot, path.basename(packFileName));\n const bundledCode = concat.toString();\n\n if (config?.dev?.enableSourcemap) {\n const map = concat.generateMap({\n hires: true,\n includeContent: true,\n file: path.basename(packFileName), // 推荐用最终输出文件名\n });\n /**\n * 如果是走内敛\n */\n if (config?.dev?.inlineSourcemap) {\n // 生成 base 64\n const base64 = Buffer.from(map.toString()).toString('base64');\n const sourcemapContent = `data:application/json;base64,${base64}`;\n // 4. 写入 sourcemap 内容\n const fileContent =\n bundledCode +\n `\\n//# sourceMappingURL=${sourcemapContent}` + // 只写文件名即可,和 JS 文件同目录\n `\\n//# sourceURL=${packFileName}`;\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n } else {\n const sourcemapBaseUrl = `http://${config?.dev?.host}:${config?.dev?.port}/game/files`;\n // 1. 生成 sourcemap 文件路径\n const mapFileName = path.basename(packFileName) + '.map';\n const mapFilePath = path.join(destRoot, mapFileName);\n // 2. 写入 sourcemap 文件\n fs.writeFileSync(mapFilePath, map.toString(), 'utf-8');\n // 3. JS 文件末尾加外链注释\n const fileContent =\n bundledCode +\n `\\n//# sourceMappingURL=${sourcemapBaseUrl}/${packFileName}.map` + // 只写文件名即可,和 JS 文件同目录\n `\\n//# sourceURL=${packFileName}`;\n fs.writeFileSync(outFile, fileContent, 'utf-8');\n }\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","/**\n * 合并请求拆包\n */\nimport fs from 'fs';\nimport path from 'path';\nimport { getDirSizeSync, logger } from '@utils';\nimport { downloadAndSaveCodesAsync } from './utils';\nimport { AVAILABLE_ODR_TAG_LIST, ONE_ODR_PKG_LIMIT } from '@constants';\nimport { getOdrConfigs } from './getConfigs';\n\nexport async function buildOdrPkgs({ tempDir }: { tempDir: string }) {\n if (fs.existsSync(tempDir)) {\n fs.rmSync(tempDir, { recursive: true });\n }\n fs.mkdirSync(tempDir);\n const games = getOdrConfigs();\n logger.info('odr_games');\n await downloadAndSaveCodesAsync(games, tempDir);\n const gamesContentSize = {};\n for (const game of games) {\n const { id: gameId } = game;\n const gameDir = path.join(tempDir, 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(tempDir, 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(tempDir, 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 console.log(`finish to build ${gameId}`);\n fs.rmSync(path.join(tempDir, gameId), { recursive: true });\n }\n // 写入 JSON 文件\n const configPath = path.join(tempDir, '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","import data from './config.json';\n\nexport function getOdrConfigs() {\n const { MinisPackagingInfoMap } = data;\n return Object.keys(MinisPackagingInfoMap).map(clientKey => {\n return {\n id: clientKey,\n packages: JSON.parse(MinisPackagingInfoMap[clientKey].ODRPackages),\n };\n });\n}\n","import { logger } from '@utils';\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, build } = 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 });\n /**\n * 打包\n */\n await makePkgs({\n config,\n gameEntry: entryDir,\n gameOutput: outputDir,\n });\n\n if (build?.enableOdr) {\n /**\n * 等待文件全部读写完成后\n */\n await makeOdrPkgs(outputDir);\n /**\n * 分拆 odr 包\n */\n }\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';\nimport { logger } from '@utils';\nimport { GAME_PACK_CONFIG_FILE_NAME } 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 dev: { enableLog },\n } = config;\n logger.init(outputDir, enableLog);\n try {\n /**\n * 校验\n */\n await checkPkgs({\n entryDir,\n config,\n });\n /**\n * 打包\n */\n await makePkgs({\n gameEntry: entryDir,\n gameOutput: outputDir,\n config,\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 async 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 = await fs.promises.readdir(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 = await fs.promises.stat(srcPath);\n if (stat.isFile()) {\n await fs.promises.copyFile(srcPath, destPath);\n // 复制好了之后就删除\n await fs.promises.rm(srcPath, {\n recursive: true,\n force: true,\n });\n } else if (stat.isDirectory()) {\n await fs.promises.cp(srcPath, destPath, { recursive: true });\n // 复制好了之后就删除\n await fs.promises.rm(srcPath, { recursive: true, force: true }); \n } else if (stat.isSymbolicLink()) {\n const linkTarget = fs.readlinkSync(srcPath);\n // 判断目标是否已存在符号链接\n if (fs.existsSync(destPath)) {\n await fs.promises.rm(destPath, { force: true });\n }\n await fs.promises.symlink(\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 * 如果最后 pkgDir 为空,就删除 \n */\n if (fs.readdirSync(pkgDir).length === 0) {\n fs.rmSync(pkgDir, { recursive: true, force: true });\n logger.info(`已删除 package 目录: ${pkgDir}`);\n }\n }\n\n logger.info('mergePkgs 完成');\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","getDirSizeSync","filterDirs","totalSize","entries","withFileTypes","entry","name","includes","isFile","size","deleteJsFilesSync","endsWith","unlinkSync","err","deleteNoJsFilesSync","uselessDirs","uselessFiles","async","downloadAndSaveCodesAsync","games","tempDir","concurrency","tasks","game","id","gameId","packages","gameDir","mkdir","packageName","packageInfo","code_url","fileName","filePath","push","poolLimit","array","iteratorFn","ret","executing","item","p","Promise","resolve","then","length","e","splice","indexOf","race","all","asyncPool","task","console","res","fetch","ok","Error","statusText","buffer","Buffer","from","arrayBuffer","writeFile","checkProject","entryDir","config","build","pkgSizeLimit","dev","gameSize","gameSizeMB","Math","round","limitMB","errMsg","enable","checkProjectSize","gameJsonPath","checkGameJson","gameJsonConfigPath","gameJsonConfig","JSON","parse","readFileSync","appid","checkProjectConfig","checkMainPackage","gameJsPath","mainPkgSize","mainPkgSizeMB","checkMainPackageSize","checkSubpackages","subPkgSizeLimit","subpackages","forEach","sub","root","subpackageEntryDir","subpackageRoot","subPkgName","limit","subPkgDir","subPkgSize","sizeMB","checkSubPackageSize","checkPkgs","setup","outputDir","gameJson","url","md5","main","configs","normalizeName","String","replace","getMainFromRoot","str","test","normalizeRoot","map","transformSubPackages","result","Object","keys","pkg","pkgPath","writeFileSync","stringify","flattenMaps","allMaps","list","values","arr","collectPkgJsFiles","exts","excludeDirs","relativePath","relative","find","concat","some","ext","collectDeps","gameEntry","rootPrefixes","filter","getRootPrefixes","glob","sync","cwd","ignore","requireCalls","code","ast","acorn","ecmaVersion","sourceType","walkAst","node","type","callee","arguments","requirePath","value","resolvedPath","startsWith","dirname","prefix","fileExistsWithExtensions","newRequireCalls","i","depModule","curModule","assign","reduce","acc","cur","f","cb","key","hasOwnProperty","child","Array","isArray","c","partition","pkgRoot","destRoot","packedFiles","gameOutput","skipDirs","isRoot","srcPath","destPath","copyFileSync","getSkipDirs","makePkgs","startTime","Date","now","removeUselessDirs","message","stat","force","removeSystemUseless","allDeps","subRoots","Boolean","uniqueSubRoots","Set","pkgName","srcRoot","jsFiles","absPath","packNameRoot","collectMaps","pkgConfig","packMap","moduleConfig","deps","packFileName","relPaths","Bundle","separator","relPath","basename","transformSync","loader","target","fileId","defineHeader","defineFooter","fileMagic","MagicString","prepend","append","addSource","content","outFile","bundledCode","toString","_a","enableSourcemap","generateMap","hires","includeContent","_b","inlineSourcemap","fileContent","sourcemapBaseUrl","_c","host","_d","port","mapFileName","mapFilePath","pack","MinisPackagingInfoMap","data","clientKey","ODRPackages","getOdrConfigs","gamesContentSize","odrTagGameMap","tag","TTMGODRConfig","assignedTag","odrTag","odrTagDir","sum","gid","srcDir","destDir","cpSync","asset_md5","asset_url","configPath","enableOdr","outputPath","pkgOutput","codeOutput","assetOutput","code_md5","makeOdrPkgs","gamePackConfigPath","gamePackConfig","pkgDir","promises","readdir","copyFile","rm","cp","isSymbolicLink","linkTarget","readlinkSync","symlink","mergePkgs","packageConfig","isSuccess","errorMsg","odrPackages","odr_packages","relatedPkgName","split","code_output","asset_output"],"mappings":"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"}
|