rancher log collect [rancher日志收集]

用于收集rancher中pod的日志

// ==UserScript==
// @name         rancher log collect [rancher日志收集]
// @namespace    http://tampermonkey.net/
// @version      0.0.1
// @description  用于收集rancher中pod的日志
// @author       maple.
// @match        https://rancher.indata.cc/dashboard/*
// @icon         https://www.rancher.com/assets/img/favicon.png
// @grant        none
// ==/UserScript==


(function () {
    'use strict';

    // 需要hook的ws包含url
    const TARGET_URL = 'exec';
    const originalWebSocket = window.WebSocket;

    let wsIdIncr = 0;

    /**
     * rancher exec ws中,发送消息前缀是0; 接受消息前缀是1
     */
    class WebSocketProxy extends originalWebSocket {

        wsId = 0;
        url = '';

        constructor(...args) {
            super(...args);

            // 检查是否需要hook这个WebSocket连接
            if (!this.shouldHook(args[0])) {
                return this;
            }

            this.wsId = ++wsIdIncr;
            this.shellExt = new ShellExt()
            this.hookMain()
        }

        shouldHook(url) {
            this.url = url;
            return typeof TARGET_URL === 'string'
                ? url.includes(TARGET_URL)
                : TARGET_URL.test(url);
        }

        hookMain() {
            console.log(`Hook-WS [${this.wsId}]:`, this.url);
            this.hookOnMessage(this.onMessage.bind(this));
            this.hookSend(this.onSend.bind(this));
        }

        onMessage(event) {
            // console.log('接收:', event.data);
            if (!this.shellExt.openCollectLog()) {
                return;
            }

            // 过滤回车
            if (event.data === '1DQo=') {
                return;
            }

            let base64Result = atob(event.data.substring(1));

            // 过滤:]
            if (base64Result.startsWith("]0;") && base64Result.includes("")) {
                console.log("过滤数据", base64Result)
                return;
            }
            // 替换颜色
            let result1 = base64Result.replaceAll(/\[\d+;*\d+m/g, "");
            // 查找字符串前缀颜色
            let result2 = result1.replaceAll(/\[K/g, "");
            let result3 = result2.replaceAll(/\[m/g, "");
            // 解码中文base64结果
            let result = this.decodeBase64(result3)
            // console.log("解析数据\n", result)
            this.shellExt.collectResult(result)
        }

        decodeBase64(binaryString) {
            const bytes = new Uint8Array(binaryString.length);
            for (let i = 0; i < binaryString.length; i++) {
                bytes[i] = binaryString.charCodeAt(i);
            }
            // 将二进制数据解码为文本
            return new TextDecoder("utf-8").decode(bytes);
        }

        onSend(data) {
            // console.log('发送:', data);
            // 发送回车,从等待采集变更为采集中
            if (this.shellExt.status === 1 && data === "0DQ==") {
                this.shellExt.changeBtnStatus()
            }
        }

        hookOnMessage(onMessage) {
            // 保存原始的onmessage
            const originalOnMessage = this.onmessage;
            // Hook onmessage
            Object.defineProperty(this, 'onmessage', {
                set: function (fn) {
                    return this.addEventListener('message', function (event) {
                        onMessage(event);
                        // 执行原始的消息处理
                        fn.call(this, event);
                    });
                },
                get: function () {
                    return originalOnMessage;
                }
            });
        }

        hookSend(onSend) {
            // Hook send
            const originalSend = this.send;
            this.send = function (data) {
                onSend(data)
                // 执行原始的发送操作
                return originalSend.call(this, data);
            };
        }
    }

    class ShellExt {

        collectLogBtn = null
        status = 0;
        result = "";

        constructor() {
            this.renderCollectBtn()
        }


        openCollectLog() {
            return this.status === 2
        }

        renderCollectBtn() {
            let btnCssText = "display: inline-block;border-radius: 4px;background: #e5f8f8;color: #00a6a7; text-decoration: none;padding: 6px 12px;cursor: pointer";
            let collectLogBtn = DOMApi.createTag("div", "采集日志", btnCssText);
            collectLogBtn.classList.add("collect-log-btn");
            this.collectLogBtn = collectLogBtn
            DOMApi.eventListener(collectLogBtn, "click", () => {
                this.collectLogHandler()
            })

            setTimeout(() => {
                let bottomTitleList = document.querySelectorAll(".title.clearfix");
                bottomTitleList.forEach(bottomTitle => {
                    let collectBtn = bottomTitle.querySelector(".collect-log-btn");
                    if (collectBtn) {
                        return;
                    }
                    bottomTitle.append(this.collectLogBtn)
                })
            }, 500)
        }

        collectLogHandler() {
            if (this.status === 2) {
                this.copyToClipboard(this.result)
                this.result = ''
            }
            this.changeBtnStatus()
        }

        collectResult(segmentStr) {
            if (!this.openCollectLog()) {
                return;
            }

            this.result += segmentStr;
        }

        changeBtnStatus() {
            // 未开启
            if (this.status === 0) {
                this.collectLogBtn.innerHTML = "等待回车开始采集"
                this.collectLogBtn.style.backgroundColor = "rgb(251,224,224)";
                this.collectLogBtn.style.color = "rgb(191,110,75)";
                this.status = 1
                // 等待采集
            } else if (this.status === 1) {
                this.collectLogBtn.innerHTML = "采集中"
                this.collectLogBtn.style.backgroundColor = "rgb(184,53,53)";
                this.collectLogBtn.style.color = "rgb(57,40,40)";
                this.status = 2
                // 采集中
            } else if (this.status === 2) {
                this.collectLogBtn.innerHTML = "采集日志"
                this.collectLogBtn.style.backgroundColor = "rgb(215,254,195)";
                this.collectLogBtn.style.color = "rgb(2,180,6)";
                this.status = 0
            }
        }

        copyToClipboard(text) {
            if (navigator.clipboard) {
                navigator.clipboard.writeText(text).then(() => {
                    console.log('Text copied to clipboard');
                    this.showMessage("日志内容已经复制到剪切板", 2000)
                }).catch(err => {
                    console.error('Failed to copy text to clipboard', err);
                });
            } else {
                console.log('Clipboard API not available');
            }
        }

        showMessage(text, duration) {
            // 创建一个div元素作为消息提示
            let message = document.createElement('div');
            message.style.position = 'fixed';
            message.style.top = '15%';
            message.style.left = '50%';
            message.style.transform = 'translate(-50%, -50%)';
            message.style.padding = '10px';
            message.style.border = '1px solid #ccc';
            message.style.backgroundColor = '#e1dcdc';
            message.style.zIndex = '1000';
            message.style.boxShadow = '0 2px 4px rgba(0,0,0,0.2)';
            message.style.borderRadius = '5px';
            message.style.fontSize = '16px';
            message.style.fontWeight = 'bold';
            message.style.color = '#ba6868';
            message.style.textAlign = 'center';
            message.style.maxWidth = '300px';
            message.style.boxSizing = 'border-box';
            message.style.visibility = 'hidden'; // 初始不可见

            // 设置消息文本
            message.textContent = text;

            // 将消息元素添加到body中
            document.body.appendChild(message);

            // 显示消息
            message.style.visibility = 'visible';

            // 指定时间后消失
            setTimeout(function () {
                message.style.visibility = 'hidden';
                setTimeout(function () {
                    document.body.removeChild(message);
                }, 300); // 等待300毫秒后移除元素,以确保过渡效果
            }, duration);
        }
    }

    class DOMApi {

        static createTag(tag, name, style) {
            let htmlTag = document.createElement(tag);
            if (name) {
                htmlTag.innerHTML = name;
            }
            if (style) {
                htmlTag.style.cssText = style;
            }
            return htmlTag;
        }

        static eventListener(tag, eventType, func) {
            tag.addEventListener(eventType, func)
        }
    }

    window.WebSocket = WebSocketProxy;
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址