User:Eejit43/scripts/script-updater.js
Appearance
< User:Eejit43 | scripts
Code that you insert on this page could contain malicious content capable of compromising your account. If you import a script from another page with "importScript", "mw.loader.load", "iusc", or "lusc", take note that this causes you to dynamically load a remote script, which could be changed by others. Editors are responsible for all edits and actions they perform, including by scripts. User scripts are not centrally supported and may malfunction or become inoperable due to software changes. A guide to help you find broken scripts is available. If you are unsure whether code you are adding to this page is safe, you can ask at the appropriate village pump. This code will be executed when previewing this page. |
Documentation for this user script can be added at User:Eejit43/scripts/script-updater. |
// <nowiki>
// Note: This script was compiled and minified 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","oojs-ui-core","oojs-ui-widgets","oojs-ui-windows"],()=>{if(mw.config.get("wgUserName")!=="Eejit43"||mw.config.get("wgPageName")!=="User:Eejit43")return;let m=document.querySelector(".mw-editsection").cloneNode(!0),r=document.createElement("a");r.href="#",r.style.fontWeight="bold",r.textContent="Sync from GitHub",r.addEventListener("click",h=>{h.preventDefault();let e=new OO.ui.WindowManager;document.body.append(e.$element[0]);let t=new a;e.addWindows([t]),t.open()}),m.querySelector("a").replaceWith(r),document.querySelector("h2#My_user_scripts + .mw-editsection").after(m);class a extends OO.ui.ProcessDialog{repoOwner="Eejit43";repoName="wikipedia-scripts";content;checkboxElements=[];actionsMultiselect;latestCommitHash;scripts;constructor(){super({size:"medium"}),a.static.name="ScriptUpdaterDialog",a.static.title="What scripts do you want to update?",a.static.actions=[{action:"cancel",label:"Close",flags:["safe","close"]},{action:"save",label:"Run",flags:["primary","progressive"]}]}getSetupProcess=()=>a.super.prototype.getSetupProcess.call(this).next(()=>this.wrapAsyncMethod(this.loadScriptData).then(e=>{if(e){let s=new OO.ui.MessageWidget({type:"error",label:e});this.content=new OO.ui.PanelLayout({padded:!0,expanded:!1}),this.content.$element.append(s.$element),this.$body.append(this.content.$element),this.getActions().setAbilities({save:!1});return}this.content=new OO.ui.PanelLayout({padded:!0,expanded:!1});for(let s of this.scripts){let n=new OO.ui.CheckboxInputWidget;this.checkboxElements.push([s.name,n]);let o=new OO.ui.FieldLayout(n,{align:"inline",label:s.name});this.content.$element.append(o.$element)}this.actionsMultiselect=new OO.ui.CheckboxMultiselectWidget({items:[{id:"documentation",name:"Update script documentation"},{id:"script",name:"Update script code"},{id:"style",name:"Update stylesheet"},{id:"talk",name:"Create talk redirect",selectedDefault:!1}].map(({id:s,name:n,selectedDefault:o})=>new OO.ui.CheckboxMultioptionWidget({data:s,label:n,selected:o??!0}))});let t=new OO.ui.FieldLayout(this.actionsMultiselect,{label:new OO.ui.HtmlSnippet("<b>Actions to take (if applicable):</b>"),align:"inline"});this.content.$element.append(t.$element),this.$body.append(this.content.$element)}));getActionProcess=e=>e==="cancel"?new OO.ui.Process(()=>{this.getManager().closeWindow(this)}):e==="save"?new OO.ui.Process(()=>{let t=[];for(let[s,n]of this.checkboxElements)n.isSelected()&&t.push(this.scripts.find(o=>o.name===s));this.getManager().closeWindow(this),(async()=>(mw.notify("Syncing scripts...",{tag:"sync-scripts-notification"}),await Promise.all(t.map(s=>this.handleScript(s))),await this.editOrCreate("User:Eejit43/scripts-info",[this.mapScripts(this.scripts.filter(s=>!s.personal&&!s.fork)),"","=== Forks ===",this.mapScripts(this.scripts.filter(s=>s.fork)),"","=== Personal scripts ===",this.mapScripts(this.scripts.filter(s=>s.personal))].join(`
`),"Syncing script list from GitHub"),mw.notify(`Synced ${t.length} script${t.length===1?"":"s"} from GitHub!`,{type:"success",tag:"sync-scripts-notification"})))()}):a.super.prototype.getActionProcess.call(this,e);getTeardownProcess=()=>a.super.prototype.getTeardownProcess.call(this).next(()=>{this.$body.empty()});wrapAsyncMethod(e){let t=$.Deferred();return e().then(s=>t.resolve(s)),t.promise()}loadScriptData=async()=>{let e=await fetch(`https://api.github.com/repos/${this.repoOwner}/${this.repoName}/commits`);if(!e.ok)return`Failed to fetch latest commit hash from GitHub: ${e.statusText} (${e.status})`;this.latestCommitHash=(await e.json())[0].sha;let t=await fetch(`https://raw.githubusercontent.com/${this.repoOwner}/${this.repoName}/${this.latestCommitHash}/scripts.json`);if(!t.ok)return`Failed to fetch script data from GitHub: ${t.statusText} (${t.status})`;this.scripts=await t.json()};async handleScript(e){let t=this.actionsMultiselect.findSelectedItemsData(),s=`User:Eejit43/scripts/${e.name}`,n=`User talk:Eejit43/scripts/${e.name}`,o=`${s}.js`,l=`${s}.css`,u=["{{User:Eejit43/script-documentation",e.image===!1?`| image = ${e.image}`:null,e["in-development"]?`| in-development = ${e["in-development"]}`:null,e["use-instead"]?`| use-instead = ${e["use-instead"].includes("User:")?e["use-instead"]:`[[User:Eejit43/scripts/${e["use-instead"]}|${e["use-instead"]}]]`}`:null,e["image-caption"]?`| image-caption = ${e["image-caption"]}`:null,e["other-authors"]?`| other-authors = ${e["other-authors"].map(i=>`[[User:${i}|${i}]]`).join(", ")}`:null,`| description-short = ${e["short-description"]}`,`| description = ${e.description}`,e.usage?`| usage = ${e.usage}`:null,e.configuration?`| configuration = ${e.configuration}`:null,e.changelog?`| changelog =
${Object.entries(e.changelog).map(([i,c])=>`* '''{{start date and age|${i}}}:'''${Array.isArray(c)?`
${c.map(g=>`** ${g}`).join(`
`)}`:` ${c}`}`).join(`
`)}`:null,`| skin-support = {{User:Eejit43/script-documentation/skin-support|${Object.entries(e["skin-support"]).map(([i,c])=>`${i}=${c}`).join("|")}}}`,`| released = {{start date and age|${e.released}}}`,`| updated = {{start date and age|${e.updated}}}`,"}}"].filter(Boolean),p=null;if(t.includes("script")){let i=await fetch(`https://raw.githubusercontent.com/${this.repoOwner}/${this.repoName}/${this.latestCommitHash}/dist/scripts/${e.name}.js`);if(i.ok)p=await i.text();else return mw.notify(`Failed to fetch "${e.name}.js" from GitHub: ${i.statusText} (${i.status})`,{type:"error",tag:"sync-scripts-notification"})}let d=null;if(t.includes("style")&&e.css){let i=await fetch(`https://raw.githubusercontent.com/${this.repoOwner}/${this.repoName}/${this.latestCommitHash}/dist/styles/${e.name}.css`);i.ok?d=await i.text():mw.notify(`Failed to fetch "${e.name}.css" from GitHub: ${i.statusText} (${i.status})`,{type:"error",tag:"sync-scripts-notification"})}e.personal||(t.includes("documentation")&&await this.editOrCreate(s,u.join(`
`),"Syncing script documentation from GitHub"),t.includes("talk")&&await this.editOrCreate(n,"#REDIRECT [[User talk:Eejit43]]","Redirecting script documentation talk page to main user talk page")),p&&await this.editOrCreate(o,p,"Syncing script from GitHub"),e.css&&d&&await this.editOrCreate(l,d,"Syncing styles from GitHub")}mapScripts(e){return e.map(t=>`* [[User:Eejit43/scripts/${t.name}${t.personal?".js":""}|${t.name}]] - ${t["short-description"]||t.description}${t["in-development"]?' (<span style="color: #bd2828">in development</span>)':""}${t["use-instead"]?' (<span style="color: #bd2828">deprecated</span>)':""}`).join(`
`)}async editOrCreate(e,t,s){s+=" (via [[User:Eejit43/scripts/script-updater.js|script]])",await new mw.Api().edit(e,()=>({text:t,summary:s,watchlist:"watch"})).catch(async(n,o)=>{if(n==="nocreate-missing")await new mw.Api().create(e,{summary:s,watchlist:"watch"},t).catch((l,u)=>{mw.notify(`Error creating ${e}: ${u?.error.info??"Unknown error"} (${l})`,{type:"error"})});else{mw.notify(`Error editing or creating ${e}: ${o?.error.info??"Unknown error"} (${n})`,{type:"error"});return}})}}Object.assign(a.prototype,OO.ui.ProcessDialog.prototype)});})();
// </nowiki>
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../scripts/script-updater.ts"],
  "sourcesContent": ["import type { MediaWikiDataError } from '../global-types';\n\ninterface Script {\n    'name': string;\n    'in-development'?: boolean;\n    'use-instead'?: string;\n    'image'?: false;\n    'image-caption'?: string;\n    'short-description': string;\n    'description': string;\n    'usage'?: string;\n    'configuration'?: string;\n    'changelog'?: Record<string, 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', 'oojs-ui-core', 'oojs-ui-widgets', 'oojs-ui-windows'], () => {\n    if (mw.config.get('wgUserName') !== 'Eejit43' || mw.config.get('wgPageName') !== 'User:Eejit43') return;\n\n    const fullLinkElement = document.querySelector('.mw-editsection')!.cloneNode(true) as HTMLSpanElement;\n\n    const link = document.createElement('a');\n    link.href = '#';\n    link.style.fontWeight = 'bold';\n    link.textContent = 'Sync from GitHub';\n    link.addEventListener('click', (event) => {\n        event.preventDefault();\n\n        const windowManager = new OO.ui.WindowManager();\n        document.body.append(windowManager.$element[0]);\n\n        const scriptUpdaterDialog = new ScriptUpdaterDialog();\n        windowManager.addWindows([scriptUpdaterDialog]);\n\n        scriptUpdaterDialog.open();\n    });\n\n    fullLinkElement.querySelector('a')!.replaceWith(link);\n\n    document.querySelector('h2#My_user_scripts + .mw-editsection')!.after(fullLinkElement);\n\n    /**\n     * An instance of this class is a dialog that manages updating scripts.\n     */\n    class ScriptUpdaterDialog extends OO.ui.ProcessDialog {\n        private repoOwner = 'Eejit43';\n        private repoName = 'wikipedia-scripts';\n\n        private content!: OO.ui.PanelLayout;\n        private checkboxElements: [string, OO.ui.CheckboxInputWidget][] = [];\n        private actionsMultiselect!: OO.ui.CheckboxMultiselectWidget;\n\n        private latestCommitHash!: string;\n        private scripts!: Script[];\n\n        constructor() {\n            super({ size: 'medium' });\n\n            ScriptUpdaterDialog.static.name = 'ScriptUpdaterDialog';\n            ScriptUpdaterDialog.static.title = 'What scripts do you want to update?';\n            ScriptUpdaterDialog.static.actions = [\n                { action: 'cancel', label: 'Close', flags: ['safe', 'close'] },\n                { action: 'save', label: 'Run', flags: ['primary', 'progressive'] },\n            ];\n        }\n\n        getSetupProcess = () => {\n            return ScriptUpdaterDialog.super.prototype.getSetupProcess.call(this).next(() => {\n                return this.wrapAsyncMethod(this.loadScriptData).then((error?: string) => {\n                    if (error) {\n                        const messageElement = new OO.ui.MessageWidget({ type: 'error', label: error });\n\n                        this.content = new OO.ui.PanelLayout({ padded: true, expanded: false });\n                        this.content.$element.append(messageElement.$element);\n\n                        (this as unknown as { $body: JQuery }).$body.append(this.content.$element);\n\n                        this.getActions().setAbilities({ save: false });\n\n                        return;\n                    }\n\n                    this.content = new OO.ui.PanelLayout({ padded: true, expanded: false });\n\n                    for (const script of this.scripts) {\n                        const checkbox = new OO.ui.CheckboxInputWidget();\n\n                        this.checkboxElements.push([script.name, checkbox]);\n\n                        const layout = new OO.ui.FieldLayout(checkbox, { align: 'inline', label: script.name });\n\n                        this.content.$element.append(layout.$element);\n                    }\n\n                    this.actionsMultiselect = new OO.ui.CheckboxMultiselectWidget({\n                        items: [\n                            { id: 'documentation', name: 'Update script documentation' },\n                            { id: 'script', name: 'Update script code' },\n                            { id: 'style', name: 'Update stylesheet' },\n                            { id: 'talk', name: 'Create talk redirect', selectedDefault: false },\n                        ].map(\n                            ({ id, name, selectedDefault }) =>\n                                new OO.ui.CheckboxMultioptionWidget({ data: id, label: name, selected: selectedDefault ?? true }),\n                        ),\n                    });\n\n                    const actionsMultiselectLayout = new OO.ui.FieldLayout(this.actionsMultiselect, {\n                        label: new OO.ui.HtmlSnippet('<b>Actions to take (if applicable):</b>'),\n                        align: 'inline',\n                    });\n\n                    this.content.$element.append(actionsMultiselectLayout.$element);\n\n                    (this as unknown as { $body: JQuery }).$body.append(this.content.$element);\n                });\n            });\n        };\n\n        getActionProcess = (action: string) => {\n            if (action === 'cancel')\n                return new OO.ui.Process(() => {\n                    this.getManager().closeWindow(this);\n                });\n            else if (action === 'save')\n                return new OO.ui.Process(() => {\n                    const selectedScripts = [];\n                    for (const [scriptName, checkbox] of this.checkboxElements)\n                        if (checkbox.isSelected()) selectedScripts.push(this.scripts.find((script) => script.name === scriptName)!);\n\n                    this.getManager().closeWindow(this);\n\n                    (async () => {\n                        mw.notify('Syncing scripts...', { tag: 'sync-scripts-notification' });\n\n                        await Promise.all(selectedScripts.map((script) => this.handleScript(script)));\n\n                        await this.editOrCreate(\n                            'User:Eejit43/scripts-info',\n                            [\n                                this.mapScripts(this.scripts.filter((script) => !script.personal && !script.fork)),\n                                '',\n                                '=== Forks ===',\n                                this.mapScripts(this.scripts.filter((script) => script.fork)),\n                                '',\n                                '=== Personal scripts ===',\n                                this.mapScripts(this.scripts.filter((script) => script.personal)),\n                            ].join('\\n'),\n                            'Syncing script list from GitHub',\n                        );\n\n                        mw.notify(`Synced ${selectedScripts.length} script${selectedScripts.length === 1 ? '' : 's'} from GitHub!`, {\n                            type: 'success',\n                            tag: 'sync-scripts-notification',\n                        });\n                    })();\n                });\n            else return ScriptUpdaterDialog.super.prototype.getActionProcess.call(this, action);\n        };\n\n        getTeardownProcess = () => {\n            return ScriptUpdaterDialog.super.prototype.getTeardownProcess.call(this).next(() => {\n                (this as unknown as { $body: JQuery }).$body.empty();\n            });\n        };\n\n        /**\n         * Wraps an async method into a jQuery Deferred object.\n         * @param method The method to wrap.\n         */\n        private wrapAsyncMethod(method: () => Promise<unknown>) {\n            const deferred = $.Deferred();\n\n            method().then((result) => deferred.resolve(result));\n\n            return deferred.promise();\n        }\n\n        /**\n         * Loads data for all scripts.\n         */\n        private loadScriptData = async () => {\n            const latestCommitHashResponse = await fetch(`https://api.github.com/repos/${this.repoOwner}/${this.repoName}/commits`);\n            if (!latestCommitHashResponse.ok)\n                return `Failed to fetch latest commit hash from GitHub: ${latestCommitHashResponse.statusText} (${latestCommitHashResponse.status})`;\n\n            this.latestCommitHash = ((await latestCommitHashResponse.json()) as { sha: string }[])[0].sha;\n\n            const scriptDataResponse = await fetch(\n                `https://raw.githubusercontent.com/${this.repoOwner}/${this.repoName}/${this.latestCommitHash}/scripts.json`,\n            );\n            if (!scriptDataResponse.ok)\n                return `Failed to fetch script data from GitHub: ${scriptDataResponse.statusText} (${scriptDataResponse.status})`;\n\n            this.scripts = (await scriptDataResponse.json()) as Script[];\n        };\n\n        /**\n         * Handles and edits/creates pages for a given script.\n         * @param script The script to handle.\n         */\n        private async handleScript(script: Script) {\n            const actionsToTake = this.actionsMultiselect.findSelectedItemsData() as ('documentation' | 'script' | 'style' | 'talk')[];\n\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.image === false ? `| image             = ${script.image}` : null,\n                script['in-development'] ? `| in-development    = ${script['in-development']}` : null,\n                script['use-instead']\n                    ? `| use-instead       = ${script['use-instead'].includes('User:') ? script['use-instead'] : `[[User:Eejit43/scripts/${script['use-instead']}|${script['use-instead']}]]`}`\n                    : null,\n                script['image-caption'] ? `| image-caption     = ${script['image-caption']}` : null,\n                script['other-authors']\n                    ? `| other-authors     = ${script['other-authors'].map((author) => `[[User:${author}|${author}]]`).join(', ')}`\n                    : null,\n                `| description-short = ${script['short-description']}`,\n                `| description       = ${script.description}`,\n                script.usage ? `| usage             = ${script.usage}` : null,\n                script.configuration ? `| configuration     = ${script.configuration}` : null,\n                script.changelog\n                    ? `| changelog         = \\n${Object.entries(script.changelog)\n                          .map(\n                              ([date, description]) =>\n                                  `* '''{{start date and age|${date}}}:'''${Array.isArray(description) ? `\\n${description.map((line) => `** ${line}`).join('\\n')}` : ` ${description}`}`,\n                          )\n                          .join('\\n')}`\n                    : null,\n                `| skin-support      = {{User:Eejit43/script-documentation/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            if (actionsToTake.includes('script')) {\n                const scriptContentResponse = await fetch(\n                    `https://raw.githubusercontent.com/${this.repoOwner}/${this.repoName}/${this.latestCommitHash}/dist/scripts/${script.name}.js`,\n                );\n                if (scriptContentResponse.ok) scriptContent = await scriptContentResponse.text();\n                else\n                    return mw.notify(\n                        `Failed to fetch \"${script.name}.js\" from GitHub: ${scriptContentResponse.statusText} (${scriptContentResponse.status})`,\n                        {\n                            type: 'error',\n                            tag: 'sync-scripts-notification',\n                        },\n                    );\n            }\n\n            let styleContent = null;\n            if (actionsToTake.includes('style') && script.css) {\n                const styleContentResponse = await fetch(\n                    `https://raw.githubusercontent.com/${this.repoOwner}/${this.repoName}/${this.latestCommitHash}/dist/styles/${script.name}.css`,\n                );\n\n                if (styleContentResponse.ok) styleContent = await styleContentResponse.text();\n                else\n                    mw.notify(\n                        `Failed to fetch \"${script.name}.css\" from GitHub: ${styleContentResponse.statusText} (${styleContentResponse.status})`,\n                        {\n                            type: 'error',\n                            tag: 'sync-scripts-notification',\n                        },\n                    );\n            }\n\n            if (!script.personal) {\n                if (actionsToTake.includes('documentation'))\n                    await this.editOrCreate(subpageName, fullSubpageInfo.join('\\n'), 'Syncing script documentation from GitHub');\n\n                if (actionsToTake.includes('talk'))\n                    await this.editOrCreate(\n                        subpageTalkName,\n                        '#REDIRECT [[User talk:Eejit43]]',\n                        'Redirecting script documentation talk page to main user talk page',\n                    );\n            }\n\n            if (scriptContent) await this.editOrCreate(scriptName, scriptContent, 'Syncing script from GitHub');\n\n            if (script.css && styleContent) await this.editOrCreate(styleName, styleContent, 'Syncing styles from GitHub');\n        }\n\n        /**\n         * Maps scripts to a bulleted list.\n         * @param scripts The scripts to map.\n         * @returns The mapped scripts.\n         */\n        private 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['in-development'] ? ' (<span style=\"color: #bd2828\">in development</span>)' : ''\n                        }${script['use-instead'] ? ' (<span style=\"color: #bd2828\">deprecated</span>)' : ''}`,\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        private async editOrCreate(title: string, text: string, summary: string) {\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()\n                            .create(title, { summary, watchlist: 'watch' }, text)\n                            .catch((errorCode: string, errorInfo: MediaWikiDataError) => {\n                                mw.notify(`Error creating ${title}: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, {\n                                    type: 'error',\n                                });\n                                return;\n                            });\n                    else {\n                        mw.notify(`Error editing or creating ${title}: ${errorInfo?.error.info ?? 'Unknown error'} (${errorCode})`, {\n                            type: 'error',\n                        });\n                        return;\n                    }\n                });\n        }\n    }\n\n    Object.assign(ScriptUpdaterDialog.prototype, OO.ui.ProcessDialog.prototype);\n});\n"],
  "mappings": "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",
  "names": ["fullLinkElement", "link", "event", "windowManager", "scriptUpdaterDialog", "ScriptUpdaterDialog", "error", "messageElement", "script", "checkbox", "layout", "id", "name", "selectedDefault", "actionsMultiselectLayout", "action", "selectedScripts", "scriptName", "method", "deferred", "result", "latestCommitHashResponse", "scriptDataResponse", "actionsToTake", "subpageName", "subpageTalkName", "styleName", "fullSubpageInfo", "author", "date", "description", "line", "skin", "status", "scriptContent", "scriptContentResponse", "styleContent", "styleContentResponse", "scripts", "title", "text", "summary", "errorCode", "errorInfo"]
}
