UNPKG

27.1 kBJavaScriptView Raw
1"use strict";
2var __importStar = (this && this.__importStar) || function (mod) {
3 if (mod && mod.__esModule) return mod;
4 var result = {};
5 if (mod != null) for (var k in mod) if (Object.hasOwnProperty.call(mod, k)) result[k] = mod[k];
6 result["default"] = mod;
7 return result;
8};
9Object.defineProperty(exports, "__esModule", { value: true });
10const net_1 = require("@xmcl/net");
11const parser = __importStar(require("fast-html-parser"));
12const index_1 = require("./index");
13var ForgeWebPage;
14(function (ForgeWebPage) {
15 /**
16 * Parse the html string of forge webpage
17 */
18 function parse(content) {
19 const dom = parser.parse(content);
20 const selected = dom.querySelector(".elem-active");
21 const mcversion = selected.text;
22 return {
23 timestamp: "",
24 mcversion,
25 versions: dom.querySelector(".download-list").querySelector("tbody").querySelectorAll("tr")
26 .map((e) => {
27 const links = e.querySelector(".download-links").childNodes
28 .filter((elem) => elem.tagName === "li")
29 .map((elem) => {
30 elem = elem.removeWhitespace();
31 /*
32 * <div class="info-tooltip">
33 * <strong>MD5:</strong> 31742b6c996f53af96f606b7a0c46e2a<br>
34 * <strong>SHA1:</strong> 8d6a23554839d6f6014fbdb7991e3cd8af7eca80
35 * </div>
36 */
37 const tooltipInfo = elem.querySelector(".info-tooltip");
38 const url = tooltipInfo.querySelector("a") || elem.querySelector("a");
39 // href is like /maven/net/minecraftforge/forge/1.14.4-28.1.70/forge-1.14.4-28.1.70-changelog.txt
40 const href = url.attributes.href.trim();
41 const matched = /forge-.+-.+-(\w+)\.\w+/.exec(href);
42 let name = "", sha1 = "", md5 = "";
43 if (matched) {
44 name = matched[1];
45 }
46 if (!name) {
47 throw new Error(`Cannot determine name for forge url "${href}". Maybe the forge webisite changed?`);
48 }
49 try {
50 md5 = tooltipInfo.childNodes[1].text.trim();
51 sha1 = tooltipInfo.childNodes[4].text.trim();
52 }
53 catch (_a) {
54 console.warn(`Error during fetching the sha1 and md5 for the forge "${href}". The result might be wrong.`);
55 }
56 const isSha1 = /\b([a-f0-9]{40})\b/i;
57 const isMd5 = /\b[a-f0-9]{32}\b/i;
58 if (!isMd5.test(md5.trim())) {
59 console.warn(`Illegal Md5 for "${href}": ${md5}`);
60 md5 = "";
61 }
62 if (!isSha1.test(sha1.trim())) {
63 console.warn(`Illegal Sha1 for "${href}": ${sha1}`);
64 sha1 = "";
65 }
66 return {
67 name,
68 md5,
69 sha1,
70 path: href,
71 };
72 });
73 const downloadVersionElem = e.querySelector(".download-version");
74 let version;
75 let type = "common";
76 const icon = downloadVersionElem.querySelector("i");
77 if (icon) {
78 if (icon.classNames.indexOf("promo-recommended") !== -1) {
79 type = "recommended";
80 }
81 else if (icon.classNames.indexOf("promo-latest") !== -1) {
82 type = "latest";
83 }
84 else if (icon.classNames.indexOf("fa-bug") !== -1) {
85 type = "buggy";
86 }
87 version = downloadVersionElem.firstChild.text.trim();
88 }
89 else {
90 version = downloadVersionElem.text.trim();
91 }
92 const installer = links.find((l) => l.name === "installer");
93 const universal = links.find((l) => l.name === "universal");
94 const changelog = links.find((l) => l.name === "changelog");
95 const installerWin = links.find((l) => l.name === "installer-win");
96 const source = links.find((l) => l.name === "source");
97 const launcher = links.find((l) => l.name === "launcher");
98 const mdk = links.find((l) => l.name === "mdk");
99 if (installer === undefined || universal === undefined) {
100 throw new Error("Cannot parse forge web since it missing installer and universal jar info.");
101 }
102 const result = {
103 version,
104 "date": e.querySelector(".download-time").text.trim(),
105 changelog,
106 installer,
107 mdk,
108 universal,
109 source,
110 launcher,
111 "installer-win": installerWin,
112 "mcversion": mcversion,
113 type,
114 };
115 return result;
116 }),
117 };
118 }
119 ForgeWebPage.parse = parse;
120 /**
121 * Query the webpage content from files.minecraftforge.net.
122 *
123 * You can put the last query result to the fallback option. It will check if your old result is up-to-date.
124 * It will request a new page only when the fallback option is outdated.
125 *
126 * @param option The option can control querying minecraft version, and page caching.
127 */
128 async function getWebPage(option = {}) {
129 const mcversion = option.mcversion || "";
130 const url = mcversion === "" ? "http://files.minecraftforge.net/maven/net/minecraftforge/forge/index.html" : `http://files.minecraftforge.net/maven/net/minecraftforge/forge/index_${mcversion}.html`;
131 const page = await net_1.getIfUpdate(url, parse, option.fallback);
132 return page;
133 }
134 ForgeWebPage.getWebPage = getWebPage;
135 let Version;
136 (function (Version) {
137 function to(webPageVersion) {
138 return {
139 universal: webPageVersion.universal,
140 installer: webPageVersion.installer,
141 mcversion: webPageVersion.mcversion,
142 version: webPageVersion.version,
143 };
144 }
145 Version.to = to;
146 })(Version = ForgeWebPage.Version || (ForgeWebPage.Version = {}));
147})(ForgeWebPage = exports.ForgeWebPage || (exports.ForgeWebPage = {}));
148index_1.ForgeInstaller.VersionMeta = index_1.ForgeInstaller.VersionMeta || {};
149index_1.ForgeInstaller.VersionMeta.from = ForgeWebPage.Version.to;
150//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"forgeweb.js","sourceRoot":"","sources":["forgeweb.ts"],"names":[],"mappings":"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","sourcesContent":["import { getIfUpdate, UpdatedObject } from \"@xmcl/net\";\nimport * as parser from \"fast-html-parser\";\nimport { ForgeInstaller } from \"./index\";\n\nexport namespace ForgeWebPage {\n\n    export interface Download {\n        md5: string;\n        sha1: string;\n        path: string;\n    }\n\n    /**\n     * Parse the html string of forge webpage\n     */\n    export function parse(content: string): ForgeWebPage {\n        const dom = parser.parse(content);\n        const selected = dom.querySelector(\".elem-active\");\n        const mcversion = selected.text;\n        return {\n            timestamp: \"\",\n            mcversion,\n            versions: dom.querySelector(\".download-list\").querySelector(\"tbody\").querySelectorAll(\"tr\")\n                .map((e) => {\n                    const links = e.querySelector(\".download-links\").childNodes\n                        .filter((elem) => elem.tagName === \"li\")\n                        .map((elem) => {\n                            elem = elem.removeWhitespace();\n                            /*\n                             * <div class=\"info-tooltip\">\n                             *   <strong>MD5:</strong> 31742b6c996f53af96f606b7a0c46e2a<br>\n                             *   <strong>SHA1:</strong> 8d6a23554839d6f6014fbdb7991e3cd8af7eca80\n                             * </div>\n                             */\n                            const tooltipInfo = elem.querySelector(\".info-tooltip\");\n                            const url = tooltipInfo.querySelector(\"a\") || elem.querySelector(\"a\");\n                            // href is like /maven/net/minecraftforge/forge/1.14.4-28.1.70/forge-1.14.4-28.1.70-changelog.txt\n                            const href = url.attributes.href.trim();\n                            const matched = /forge-.+-.+-(\\w+)\\.\\w+/.exec(href);\n                            let name = \"\", sha1 = \"\", md5 = \"\";\n                            if (matched) { name = matched[1]; }\n                            if (!name) {\n                                throw new Error(`Cannot determine name for forge url \"${href}\". Maybe the forge webisite changed?`);\n                            }\n                            try {\n                                md5 = tooltipInfo.childNodes[1].text.trim();\n                                sha1 = tooltipInfo.childNodes[4].text.trim();\n                            } catch {\n                                console.warn(`Error during fetching the sha1 and md5 for the forge \"${href}\". The result might be wrong.`);\n                            }\n                            const isSha1 = /\\b([a-f0-9]{40})\\b/i\n                            const isMd5 = /\\b[a-f0-9]{32}\\b/i;\n                            if (!isMd5.test(md5.trim())) {\n                                console.warn(`Illegal Md5 for \"${href}\": ${md5}`)\n                                md5 = \"\";\n                            }\n                            if (!isSha1.test(sha1.trim())) {\n                                console.warn(`Illegal Sha1 for \"${href}\": ${sha1}`)\n                                sha1 = \"\";\n                            }\n                            return {\n                                name,\n                                md5,\n                                sha1,\n                                path: href,\n                            };\n                        });\n                    const downloadVersionElem = e.querySelector(\".download-version\");\n                    let version;\n                    let type: ForgeWebPage.Version[\"type\"] = \"common\";\n                    const icon = downloadVersionElem.querySelector(\"i\");\n                    if (icon) {\n                        if (icon.classNames.indexOf(\"promo-recommended\") !== -1) {\n                            type = \"recommended\";\n                        } else if (icon.classNames.indexOf(\"promo-latest\") !== -1) {\n                            type = \"latest\";\n                        } else if (icon.classNames.indexOf(\"fa-bug\") !== -1) {\n                            type = \"buggy\";\n                        }\n                        version = downloadVersionElem.firstChild.text.trim();\n                    } else {\n                        version = downloadVersionElem.text.trim();\n                    }\n                    const installer = links.find((l) => l.name === \"installer\");\n                    const universal = links.find((l) => l.name === \"universal\");\n                    const changelog = links.find((l) => l.name === \"changelog\");\n                    const installerWin = links.find((l) => l.name === \"installer-win\");\n                    const source = links.find((l) => l.name === \"source\");\n                    const launcher = links.find((l) => l.name === \"launcher\");\n\n                    const mdk = links.find((l) => l.name === \"mdk\");\n\n                    if (installer === undefined || universal === undefined) {\n                        throw new Error(\"Cannot parse forge web since it missing installer and universal jar info.\");\n                    }\n                    const result = {\n                        version,\n                        \"date\": e.querySelector(\".download-time\").text.trim(),\n                        changelog,\n                        installer,\n                        mdk,\n                        universal,\n                        source,\n                        launcher,\n                        \"installer-win\": installerWin,\n                        \"mcversion\": mcversion,\n                        type,\n                    };\n\n                    return result;\n                }),\n        };\n    }\n\n    /**\n     * Query the webpage content from files.minecraftforge.net.\n     *\n     * You can put the last query result to the fallback option. It will check if your old result is up-to-date.\n     * It will request a new page only when the fallback option is outdated.\n     *\n     * @param option The option can control querying minecraft version, and page caching.\n     */\n    export function getWebPage(): Promise<ForgeWebPage | undefined>;\n    /**\n     * Query the webpage content from files.minecraftforge.net.\n     *\n     * You can put the last query result to the fallback option. It will check if your old result is up-to-date.\n     * It will request a new page only when the fallback option is outdated.\n     *\n     * @param option The option can control querying minecraft version, and page caching.\n     */\n    export function getWebPage(option?: {\n        mcversion?: string;\n    }): Promise<ForgeWebPage | undefined>;\n    /**\n     * Query the webpage content from files.minecraftforge.net.\n     *\n     * You can put the last query result to the fallback option. It will check if your old result is up-to-date.\n     * It will request a new page only when the fallback option is outdated.\n     *\n     * @param option The option can control querying minecraft version, and page caching.\n     */\n    export function getWebPage(option?: {\n        mcversion?: string;\n        fallback?: ForgeWebPage;\n    }): Promise<ForgeWebPage | undefined>;\n    /**\n     * Query the webpage content from files.minecraftforge.net.\n     *\n     * You can put the last query result to the fallback option. It will check if your old result is up-to-date.\n     * It will request a new page only when the fallback option is outdated.\n     *\n     * @param option The option can control querying minecraft version, and page caching.\n     */\n    export function getWebPage(option?: {\n        mcversion?: string;\n        fallback: ForgeWebPage;\n    }): Promise<ForgeWebPage>;\n\n    /**\n     * Query the webpage content from files.minecraftforge.net.\n     *\n     * You can put the last query result to the fallback option. It will check if your old result is up-to-date.\n     * It will request a new page only when the fallback option is outdated.\n     *\n     * @param option The option can control querying minecraft version, and page caching.\n     */\n    export async function getWebPage(option: {\n        mcversion?: string,\n        fallback?: ForgeWebPage,\n    } = {}): Promise<ForgeWebPage | undefined> {\n        const mcversion = option.mcversion || \"\";\n        const url = mcversion === \"\" ? \"http://files.minecraftforge.net/maven/net/minecraftforge/forge/index.html\" : `http://files.minecraftforge.net/maven/net/minecraftforge/forge/index_${mcversion}.html`;\n        const page = await getIfUpdate(url, parse, option.fallback);\n        return page;\n    }\n\n    /**\n     * A richer version info than forge installer required\n     */\n    export interface Version extends ForgeInstaller.VersionMeta {\n        /**\n         * The minecraft version\n         */\n        mcversion: string;\n        /**\n         * The version of forge\n         */\n        version: string;\n        date: string;\n        /**\n         * The changelog info\n         */\n        changelog?: ForgeWebPage.Download;\n        installer: ForgeWebPage.Download;\n        mdk?: ForgeWebPage.Download;\n        universal: ForgeWebPage.Download;\n        source?: ForgeWebPage.Download;\n        launcher?: ForgeWebPage.Download;\n\n        /**\n         * The type of the forge release. The `common` means the normal release.\n         */\n        type: \"buggy\" | \"recommended\" | \"common\" | \"latest\";\n    }\n\n    export namespace Version {\n        export function to(webPageVersion: ForgeWebPage.Version): ForgeInstaller.VersionMeta {\n            return {\n                universal: webPageVersion.universal,\n                installer: webPageVersion.installer,\n                mcversion: webPageVersion.mcversion,\n                version: webPageVersion.version,\n            };\n        }\n    }\n}\n\ndeclare module \"./index\" {\n    export namespace VersionMeta {\n        export function from(webPageVersion: ForgeWebPage.Version): ForgeInstaller.VersionMeta;\n    }\n}\n\n(ForgeInstaller as any).VersionMeta = (ForgeInstaller as any).VersionMeta || {};\n(ForgeInstaller as any).VersionMeta.from = ForgeWebPage.Version.to;\n\nexport interface ForgeWebPage extends UpdatedObject {\n    versions: ForgeWebPage.Version[];\n    mcversion: string;\n}\n\n\n\n"]}
\No newline at end of file