自动识别验证码

图片验证码自动识别

// ==UserScript==
// @name         自动识别验证码
// @description  图片验证码自动识别
// @version      1.1.1
// @author       LiK4vin
// @icon         
// @license      MIT
// @match        *://*/*
// @grant        GM_xmlhttpRequest
// @namespace http://tampermonkey.net/
// ==/UserScript==

(function () {
    'use strict';

    // 访问地址
    // 使用 TrWebOCR 实现
    let origin = 'http://127.0.0.1:8089';

    console.log("【自动识别验证码】正在运行...");
    setTimeout(() => {
        findCode();
        let observer = new MutationObserver(findCode);
        observer.observe(document.body, {attributes: false, childList: true, subtree: true})
    }, 1000);

    // 寻找验证码
    function findCode() {
        for (let element of document.querySelectorAll('img')) {
            if (isCode(element)) {
                let src = element.src;
                if (src.includes('data:image')) {
                    resolve(src);
                } else {
                    let image = new Image();
                    image.src = src;
                    image.onload = () => {
                        let canvas = document.createElement('canvas');
                        canvas.width = element.width;
                        canvas.height = element.height;
                        canvas.getContext('2d').drawImage(element, 0, 0);
                        resolve(canvas.toDataURL());
                    };
                }
            }
        }
    }

    // 判断是否为验证码
    function isCode(element) {
        let attrs = ['id', 'className', 'title', 'src', 'alt'];
        let texts = ['code', 'captcha', '验证码', '看不清', '换一张'];
        for (let attr of attrs) {
            for (let text of texts) {
                if (element[attr].includes(text)) {
                    return true;
                }
            }
        }
        return element.src.includes('data:image');
    }

    // 解析验证码
    function resolve(base64) {
        let data = new FormData();
        data.append('img', base64.replace(/^(.*)base64,/, ''));

        GM_xmlhttpRequest({
            url: `${origin}/api/tr-run/`,
            method: 'post',
            data: data,
            responseType: 'json',
            onload: (response) => {
                if (response.status == 200) {
                    let result = '';
                    for (let raw of response.response.data.raw_out) {
                        result = result + raw[1];
                    }
                    result = result.replace(/\s+/g, '');
                    result = result.replace(/[^a-zA-Z0-9_]+/g, '');

                    if (result.length >= 4 && result.length <= 8) {
                        console.log("识别结果:" + result);
                        writeIn(result);
                    }
                } else {
                    console.log("识别失败");
                }
            }
        });
    }

    // 写入验证码输入框
    function writeIn(text) {
        for (let element of document.querySelectorAll('input')) {
            if (isInput(element)) {
                element.value = text;
                if (typeof (InputEvent) !== "undefined") {
                    element.value = text;
                    element.dispatchEvent(new InputEvent('input'));
                    let events = ['input', 'change', 'focus', 'keypress', 'keyup', 'keydown', 'select'];
                    for (let event of events) {
                        let e = document.createEvent('HTMLEvents');
                        e.initEvent(event, true, true);
                        element.dispatchEvent(e);
                    }
                    element.value = text;
                } else if (KeyboardEvent) {
                    element.dispatchEvent(new KeyboardEvent('input'));
                }
            }
        }
    }

    // 判断是否为验证码输入框
    function isInput(element) {
        let attrs = ['id', 'className', 'title', 'placeholder', 'alt'];
        let texts = ['code', 'captcha', '验证码', '看不清', '换一张'];
        for (let attr of attrs) {
            for (let text of texts) {
                if (element[attr].includes(text)) {
                    return true;
                }
            }
        }

        // 查找父级
        element = element.parentNode;
        for (let text of texts) {
            if (element.textContent.includes(text)) {
                return true;
            }
        }
        return false;
    }
})();