4chan Session ID Unbreaker

Tries to detect and un-break Session IDs posted on 4chan

Verze ze dne 13. 05. 2024. Zobrazit nejnovější verzi.

K instalaci tototo skriptu si budete muset nainstalovat rozšíření jako Tampermonkey, Greasemonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Userscripts.

You will need to install an extension such as Tampermonkey to install this script.

K instalaci tohoto skriptu si budete muset nainstalovat manažer uživatelských skriptů.

(Už mám manažer uživatelských skriptů, nechte mě ho nainstalovat!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(Už mám manažer uživatelských stylů, nechte mě ho nainstalovat!)

// ==UserScript==
// @name         4chan Session ID Unbreaker
// @license      GPLv3
// @namespace    https://boards.4chan.org/
// @version      1.9
// @description  Tries to detect and un-break Session IDs posted on 4chan
// @author       ceodoe
// @match        https://boards.4chan.org/*/thread/*
// @match        https://boards.4chan.org/*/res/*
// @match        https://archived.moe/*/thread/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=4chan.org
// @grant        GM_getValue
// @grant        GM_setValue
// ==/UserScript==

function parsePosts() {
    let posts = document.querySelectorAll("blockquote.postMessage");

    if(location.href.startsWith("https://archived.moe/")) {
        posts = document.querySelectorAll("article > div.text, div.post_wrapper > div.text");
    }

    for(let i = 0; i < posts.length; i++) {
        if(posts[i].getAttribute("data-4sidu-parsed") !== "1") {
            posts[i].setAttribute("data-4sidu-parsed", "1");

            // Strip all backlinks as they are likely to contain the magic number 05 that all Session IDs start with
            let postText = posts[i].innerText.replace(/\>\>\b[0-9]+\b/g, "");
            let idStartIndex = postText.indexOf("05");

            if(idStartIndex > -1) {
                let id = "";

                // "Smart" detection mechanism removes all non-alphanumeric chars, then ignores words with
                // non-hexadecimal chars in them, and tries to build a string exactly 66 chars long
                let words = postText.substring(idStartIndex).split(/\s/);
                for(let j = 0; j < words.length && id.length < 66; j++) {
                    let word = words[j].replace(/[^A-Za-z0-9]/g, "");

                    if(!word.match(/[^A-Fa-f0-9]/g)) {
                        id += word;
                    }
                }

                if(id.length == 66) { // All IDs are 66 chars; if we didn't get exactly 66, ID is invalid
                    let opPost = "";
                    if(posts[i].parentNode.classList.contains("op")) {
                        opPost = "overflow: auto;";
                    }

                    let archivePost = "border-top: 1px solid; width: fit-content;";
                    if(posts[i].parentNode.tagName.toLowerCase() == "article") {
                        archivePost = "";
                    }

                    let html = `
                        <div style="margin-top: 1em; padding: 0.5em; ${archivePost} ${opPost}">
                            <span style="color: #66cc33; font-weight: bold;">Session ID:</span> ${id}
                            <input type="button" class="4sidu-copy-btn" id="4sidu-copy-btn-${i}" data-4sidu-session-id="${id}" style="margin-left: 0.5em;" value="Copy">
                        </div>
                    `;

                    posts[i].insertAdjacentHTML("beforeend", html);

                    document.getElementById(`4sidu-copy-btn-${i}`).addEventListener("click", async function() {
                        let tempInput = document.createElement("input");
                        tempInput.value = this.getAttribute("data-4sidu-session-id");
                        tempInput.select();
                        tempInput.setSelectionRange(0,66);

                        try {
                            await navigator.clipboard.writeText(tempInput.value);
                            this.value = "✓";

                            window.setTimeout(function() {
                                document.getElementById(`4sidu-copy-btn-${i}`).value = "Copy";
                            }, 3000);
                        } catch (err) {
                            alert("Failed to copy to clipboard: " + err);
                        }
                    });
                }
            }
        }
    }
}

new MutationObserver(function(event) { parsePosts(); }).observe(document.querySelector("div.thread"), {childList: true});
parsePosts();