MooMoo.io Anti-Kick

Anti-kick for MooMoo.io

此脚本不应直接安装,它是一个供其他脚本使用的外部库。如果您需要使用该库,请在脚本元属性加入:// @require https://update.greasyfork.org/scripts/474034/1271161/MooMooio%20Anti-Kick.js

// ==UserScript==
// @name         MooMoo.io Anti-Kick
// @namespace    https://greasyfork.org/users/1064285-vcrazy-gaming
// @version      1
// @description  Prevent Getting Kicked From MooMoo.io
// @author       _VcrazY_
// @match        *://*/*
// @icon         
// @grant        none
// @license      MIT
// ==/UserScript==

// Check the current URL
if (window.location.hostname.includes("moomoo.io") ||
    window.location.hostname.includes("sandbox.moomoo.io") ||
    window.location.hostname.includes("dev.moomoo.io")) {
  // The code will only run on the specified domains

  (() => {
    "use strict";

    const PACKET_LIMITS = {
      PER_MINUTE: 1000,
      PER_SECOND: 80,
    };

    const IGNORED_PACKET_TYPES = new Set(["pp", "rmd"]);
    const IGNORED_QUEUE_PACKETS = new Set(["5", "c", "33", "2", "7", "13c"]);

    /**
     * Class to manage anti-kick functionality
     */
    class AntiKick {
      constructor() {
        this.resetRateLimit();
      }

      /**
       * Reset packet rate limits
       */
      resetRateLimit() {
        // Packet history
        this.packetHistory = new Map();
        // Packet queue
        this.packetQueue = [];
        // Last packet sent time
        this.lastSent = Date.now();
      }

      /**
       * Check if the packet is rate-limited
       * @param {ArrayBuffer} data - Binary packet data
       * @returns {boolean} True if rate-limited, false otherwise
       */
      isRateLimited(data) {
        const binaryData = new Uint8Array(data);

        if (Date.now() - this.lastSent > PACKET_LIMITS.PER_MINUTE) {
          this.resetRateLimit();
        }

        const packetType = binaryData[0];

        if (!IGNORED_PACKET_TYPES.has(packetType)) {
          if (this.packetHistory.has(packetType) &&
              (("2" === packetType || "33" === packetType) && this.packetHistory.get(packetType)[0] === binaryData[1])) {
            return true;
          }

          if (this.packetQueue.length > PACKET_LIMITS.PER_SECOND) {
            return IGNORED_QUEUE_PACKETS.has(packetType) || this.packetQueue.push(data);
          }

          this.packetQueue.push({ type: packetType, data: binaryData.slice(1) });
          this.packetHistory.set(packetType, binaryData.slice(1));
        }

        return false;
      }
    }

    const antiKick = new AntiKick();

    // Override the send method of WebSocket
    WebSocket.prototype.send = new Proxy(WebSocket.prototype.send, {
      /**
       * Apply the send method override
       * @param {function} target - WebSocket method to override
       * @param {object} thisArg - The WebSocket object
       * @param {ArrayBuffer} binary - Binary data to send
       * @returns {*} Result of the original send method
       */
      apply(target, thisArg, binary) {
        if (!thisArg.messageListenerSet) {
          // Listen for messages to process the packet queue
          thisArg.addEventListener("message", (event) => {
            if (antiKick.packetQueue.length) {
              const binaryData = new Uint8Array(event.data);
              if (binaryData[0] === 51) {
                thisArg.send(antiKick.packetQueue[0]);
                antiKick.packetQueue.shift();
              }
            }
          });
          thisArg.messageListenerSet = true;
        }

        // Send the packet only if it's not rate-limited
        if (!antiKick.isRateLimited(binary)) {
          return Reflect.apply(target, thisArg, binary);
        }
      },
    });
  })();
}