User:Eejit43/scripts/script-updater.js

This is an old revision of this page, as edited by Eejit43 (talk | contribs) at 03:53, 26 January 2024 (Syncing script from GitHub (via script)). The present address (URL) is a permanent link to this revision, which may differ significantly from the current revision.
Note: After saving, you have to bypass your browser's cache to see the changes. Google Chrome, Firefox, Microsoft Edge and Safari: Hold down the ⇧ Shift key and click the Reload toolbar button. For details and instructions about other browsers, see Wikipedia:Bypass your cache.
// <nowiki>
// Note: This script was compiled from TypeScript. For a more readable version, see https://github.com/Eejit43/wikipedia-scripts/blob/main/scripts/script-updater.ts

"use strict";mw.loader.using(["mediawiki.util"],()=>{if(mw.config.get("wgUserName")!=="Eejit43"||mw.config.get("wgPageName")!=="User:Eejit43")return;const s="Eejit43",n="wikipedia-scripts";mw.util.addPortletLink(mw.config.get("skin")==="minerva"?"p-tb":"p-cactions","#","Sync user scripts from GitHub","sync-scripts").addEventListener("click",async y=>{y.preventDefault();const m=await fetch(`https://api.github.com/repos/${s}/${n}/commits`);if(!m.ok)return mw.notify(`Failed to fetch latest commit hash from GitHub: ${m.statusText} (${m.status})`,{type:"error",tag:"sync-scripts-notification"});const l=(await m.json())[0].sha,p=await fetch(`https://raw.githubusercontent.com/${s}/${n}/${l}/scripts.json`);if(!p.ok)return mw.notify(`Failed to fetch script data from GitHub: ${p.statusText} (${p.status})`,{type:"error",tag:"sync-scripts-notification"});const a=await p.json();mw.notify("Syncing scripts...",{autoHide:!1,tag:"sync-scripts-notification"}),await Promise.all(a.map(async t=>{const e=`User:Eejit43/scripts/${t.name}`,o=`User talk:Eejit43/scripts/${t.name}`,u=`${e}.js`,f=`${e}.css`,$=["{{User:Eejit43/script-documentation",t["use-instead"]?`| use-instead       = [[User:Eejit43/scripts/${t["use-instead"]}|${t["use-instead"]}]]`:null,t["image-size"]?`| image-size        = ${t["image-size"]}`:null,t["image-caption"]?`| image-caption     = ${t["image-caption"]}`:null,t["other-authors"]?`| other-authors     = ${t["other-authors"].map(i=>`[[User:${i}|${i}]]`).join(", ")}`:null,`| description-short = ${t["short-description"]}`,`| description       = ${t.description}`,t.usage?`| usage             = ${t.usage}`:null,t.changelog?`| changelog         = ${Object.entries(t.changelog).map(([i,h])=>`'''{{start date and age|${i}}}:''' ${h}`).join(`
`)}`:null,`| skin-support      = {{User:Eejit43/skin-support|${Object.entries(t["skin-support"]).map(([i,h])=>`${i}=${h}`).join("|")}}}`,`| released          = {{start date and age|${t.released}}}`,`| updated           = {{start date and age|${t.updated}}}`,"}}"].filter(Boolean);let c=null;const g=await fetch(`https://raw.githubusercontent.com/${s}/${n}/${l}/dist/scripts/${t.name}.js`);if(g.ok)c=await g.text();else return mw.notify(`Failed to fetch "${t.name}.js" from GitHub: ${g.statusText} (${g.status})`,{type:"error",tag:"sync-scripts-notification"});let w=null;if(t.css){const i=await fetch(`https://raw.githubusercontent.com/${s}/${n}/${l}/dist/styles/${t.name}.css`);i.ok?w=await i.text():mw.notify(`Failed to fetch "${t.name}.css" from GitHub: ${i.statusText} (${i.status})`,{type:"error",tag:"sync-scripts-notification"})}t.personal||(await r(e,$.join(`
`),"Syncing script documentation from GitHub"),await r(o,"#REDIRECT [[User talk:Eejit43]]","Redirecting script documentation talk page to main user talk page")),c&&await r(u,`// <nowiki>
// Note: This script was compiled from TypeScript. For a more readable version, see https://github.com/${s}/${n}/blob/main/scripts/${t.name}.ts

${c}
// </nowiki>`,"Syncing script from GitHub"),t.css&&w&&await r(f,`/* <nowiki> */
/* Note: This script was compiled from modern CSS. For a more readable version, see https://github.com/${s}/${n}/blob/main/styles/${t.name}.css */

${w}
/* </nowiki> */`,"Syncing styles from GitHub")})),await r("User:Eejit43/scripts-info",[d(a.filter(t=>!t.personal&&!t.fork)),"","=== Forks ===",d(a.filter(t=>t.fork)),"","=== Personal scripts ===",d(a.filter(t=>t.personal))].join(`
`),"Syncing script list from GitHub"),mw.notify(`Synced ${a.length} scripts from GitHub!`,{type:"success",tag:"sync-scripts-notification"});function d(t){return t.map(e=>`* [[User:Eejit43/scripts/${e.name}${e.personal?".js":""}|${e.name}]] - ${e["short-description"]||e.description}${e["use-instead"]?' (<span style="color: #bd2828">deprecated</span>)':""}`).join(`
`)}async function r(t,e,o){o+=" (via [[User:Eejit43/scripts/script-updater.js|script]])",await new mw.Api().edit(t,()=>({text:e,summary:o,watchlist:"watch"})).catch(async(u,f)=>{if(u==="nocreate-missing")await new mw.Api().create(t,{summary:o,watchlist:"watch"},e).catch(($,c)=>{mw.notify(`Error creating ${t}: ${c?.error.info??"Unknown error"} (${$})`,{type:"error"})});else{mw.notify(`Error editing or creating ${t}: ${f?.error.info??"Unknown error"} (${u})`,{type:"error"});return}})}})});
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../scripts/script-updater.ts"],
  "sourcesContent": ["import { MediaWikiDataError } from '../global-types';\n\ninterface Script {\n    name: string;\n    'use-instead'?: string;\n    'image-size'?: string;\n    'image-caption'?: string;\n    'short-description': string;\n    description: string;\n    usage?: string;\n    changelog?: Record<string, string>;\n    'other-authors'?: string[];\n    fork?: true;\n    personal?: true;\n    'skin-support': Record<string, boolean>;\n    released: string;\n    updated: string;\n    css?: true;\n}\n\nmw.loader.using(['mediawiki.util'], () => {\n    if (mw.config.get('wgUserName') !== 'Eejit43' || mw.config.get('wgPageName') !== 'User:Eejit43') return;\n\n    const repoOwner = 'Eejit43';\n    const repoName = 'wikipedia-scripts';\n\n    const link = mw.util.addPortletLink(mw.config.get('skin') === 'minerva' ? 'p-tb' : 'p-cactions', '#', 'Sync user scripts from GitHub', 'sync-scripts')!;\n\n    link.addEventListener('click', async (event) => {\n        event.preventDefault();\n\n        const latestCommitHashResponse = await fetch(`https://api.github.com/repos/${repoOwner}/${repoName}/commits`);\n        if (!latestCommitHashResponse.ok)\n            return mw.notify(`Failed to fetch latest commit hash from GitHub: ${latestCommitHashResponse.statusText} (${latestCommitHashResponse.status})`, {\n                type: 'error',\n                tag: 'sync-scripts-notification',\n            });\n\n        const latestCommitHash = ((await latestCommitHashResponse.json()) as { sha: string }[])[0].sha;\n\n        const scriptDataResponse = await fetch(`https://raw.githubusercontent.com/${repoOwner}/${repoName}/${latestCommitHash}/scripts.json`);\n        if (!scriptDataResponse.ok)\n            return mw.notify(`Failed to fetch script data from GitHub: ${scriptDataResponse.statusText} (${scriptDataResponse.status})`, { type: 'error', tag: 'sync-scripts-notification' });\n\n        const scriptData = (await scriptDataResponse.json()) as Script[];\n\n        mw.notify('Syncing scripts...', { autoHide: false, tag: 'sync-scripts-notification' });\n\n        await Promise.all(\n            scriptData.map(async (script) => {\n                const subpageName = `User:Eejit43/scripts/${script.name}`;\n                const subpageTalkName = `User talk:Eejit43/scripts/${script.name}`;\n                const scriptName = `${subpageName}.js`;\n                const styleName = `${subpageName}.css`;\n\n                const fullSubpageInfo = [\n                    '{{User:Eejit43/script-documentation', //\n                    script['use-instead'] ? `| use-instead       = [[User:Eejit43/scripts/${script['use-instead']}|${script['use-instead']}]]` : null,\n                    script['image-size'] ? `| image-size        = ${script['image-size']}` : null,\n                    script['image-caption'] ? `| image-caption     = ${script['image-caption']}` : null,\n                    script['other-authors'] ? `| other-authors     = ${script['other-authors'].map((author) => `[[User:${author}|${author}]]`).join(', ')}` : null,\n                    `| description-short = ${script['short-description']}`,\n                    `| description       = ${script.description}`,\n                    script.usage ? `| usage             = ${script.usage}` : null,\n                    script.changelog\n                        ? `| changelog         = ${Object.entries(script.changelog)\n                              .map(([date, description]) => `'''{{start date and age|${date}}}:''' ${description}`)\n                              .join('\\n')}`\n                        : null,\n                    `| skin-support      = {{User:Eejit43/skin-support|${Object.entries(script['skin-support'])\n                        .map(([skin, status]) => `${skin}=${status}`)\n                        .join('|')}}}`,\n                    `| released          = {{start date and age|${script.released}}}`,\n                    `| updated           = {{start date and age|${script.updated}}}`,\n                    '}}',\n                ].filter(Boolean);\n\n                let scriptContent = null;\n\n                const scriptContentResponse = await fetch(`https://raw.githubusercontent.com/${repoOwner}/${repoName}/${latestCommitHash}/dist/scripts/${script.name}.js`);\n                if (scriptContentResponse.ok) scriptContent = await scriptContentResponse.text();\n                else\n                    return mw.notify(`Failed to fetch \"${script.name}.js\" from GitHub: ${scriptContentResponse.statusText} (${scriptContentResponse.status})`, {\n                        type: 'error',\n                        tag: 'sync-scripts-notification',\n                    });\n\n                let styleContent = null;\n                if (script.css) {\n                    const styleContentResponse = await fetch(`https://raw.githubusercontent.com/${repoOwner}/${repoName}/${latestCommitHash}/dist/styles/${script.name}.css`);\n\n                    if (styleContentResponse.ok) styleContent = await styleContentResponse.text();\n                    else\n                        mw.notify(`Failed to fetch \"${script.name}.css\" from GitHub: ${styleContentResponse.statusText} (${styleContentResponse.status})`, {\n                            type: 'error',\n                            tag: 'sync-scripts-notification',\n                        });\n                }\n\n                if (!script.personal) {\n                    await editOrCreate(subpageName, fullSubpageInfo.join('\\n'), 'Syncing script documentation from GitHub');\n                    await editOrCreate(subpageTalkName, '#REDIRECT [[User talk:Eejit43]]', 'Redirecting script documentation talk page to main user talk page');\n                }\n\n                if (scriptContent)\n                    await editOrCreate(\n                        scriptName,\n                        `// <nowiki>\\n// Note: This script was compiled from TypeScript. For a more readable version, see https://github.com/${repoOwner}/${repoName}/blob/main/scripts/${script.name}.ts\\n\\n${scriptContent}\\n// </nowiki>`,\n                        'Syncing script from GitHub',\n                    );\n\n                if (script.css && styleContent)\n                    await editOrCreate(\n                        styleName,\n                        `/* <nowiki> */\\n/* Note: This script was compiled from modern CSS. For a more readable version, see https://github.com/${repoOwner}/${repoName}/blob/main/styles/${script.name}.css */\\n\\n${styleContent}\\n/* </nowiki> */`,\n                        'Syncing styles from GitHub',\n                    );\n            }),\n        );\n\n        await editOrCreate(\n            'User:Eejit43/scripts-info',\n            [\n                mapScripts(scriptData.filter((script) => !script.personal && !script.fork)), //\n                '',\n                '=== Forks ===',\n                mapScripts(scriptData.filter((script) => script.fork)),\n                '',\n                '=== Personal scripts ===',\n                mapScripts(scriptData.filter((script) => script.personal)),\n            ].join('\\n'),\n            'Syncing script list from GitHub',\n        );\n\n        mw.notify(`Synced ${scriptData.length} scripts from GitHub!`, { type: 'success', tag: 'sync-scripts-notification' });\n\n        /**\n         * Maps scripts to a bulleted list.\n         * @param scripts The scripts to map.\n         * @returns The mapped scripts.\n         */\n        function mapScripts(scripts: Script[]) {\n            return scripts\n                .map(\n                    (script) =>\n                        `* [[User:Eejit43/scripts/${script.name}${script.personal ? '.js' : ''}|${script.name}]] - ${script['short-description'] || script.description}${\n                            script['use-instead'] ? ' (<span style=\"color: #bd2828\">deprecated</span>)' : ''\n                        }`,\n                )\n                .join('\\n');\n        }\n\n        /**\n         * Edits a page, or creates it if it doesn't exist.\n         * @param title The title of the page to edit.\n         * @param text The page content to set.\n         * @param summary The edit summary (will append script notice).\n         */\n        async function editOrCreate(title: string, text: string, summary: string): Promise<void> {\n            summary += ' (via [[User:Eejit43/scripts/script-updater.js|script]])';\n            await new mw.Api()\n                .edit(title, () => ({ text, summary, watchlist: 'watch' }))\n                .catch(async (errorCode: string, errorInfo: MediaWikiDataError) => {\n                    if (errorCode === 'nocreate-missing')\n                        await new mw.Api().create(title, { summary, watchlist: 'watch' }, text).catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                            mw.notify(`Error creating ${title}: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                            return;\n                        });\n                    else {\n                        mw.notify(`Error editing or creating ${title}: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, { type: 'error' });\n                        return;\n                    }\n                });\n        }\n    });\n});\n"],
  "mappings": "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",
  "names": ["repoOwner", "repoName", "event", "latestCommitHashResponse", "latestCommitHash", "scriptDataResponse", "scriptData", "script", "subpageName", "subpageTalkName", "scriptName", "styleName", "fullSubpageInfo", "author", "date", "description", "skin", "status", "scriptContent", "scriptContentResponse", "styleContent", "styleContentResponse", "editOrCreate", "mapScripts", "scripts", "title", "text", "summary", "errorCode", "errorInfo"]
}


// </nowiki>