Greasy Fork is available in English.

长毛象实例测速

打开 instances.social Legacy列表后,点击右上角图标开始测试。为了结果的准确性,请至少进行三轮测试。

// ==UserScript==
// @name        长毛象实例测速
// @namespace   https://blog.bgme.me
// @match       https://instances.social/list/old
// @grant       GM.xmlHttpRequest
// @connect     *
// @run-at      document-end
// @version     1.0.3
// @author      bgme
// @description 打开 instances.social Legacy列表后,点击右上角图标开始测试。为了结果的准确性,请至少进行三轮测试。
// @supportURL  https://github.com/yingziwu/Greasemonkey/issues
// @icon        https://raw.githubusercontent.com/tootsuite/mastodon/master/public/favicon.ico
// @license     AGPL-3.0-or-later
// ==/UserScript==

window.addEventListener("load", function () {
  addButton();
});

function addButton() {
  let img = document.createElement("img");
  img.src =
    "";
  img.style.cssText = "height: 50px;	width: 43px;";

  let button = document.createElement("button");
  button.className = "icon_pc";
  button.style.cssText = `position: fixed;
                          top: 15%;
                          right: 5%;
                          z-index: 99;
                          border-style: none;
                          text-align:center;
                          vertical-align:baseline;
                          background-color: #fafafa;
                          border-radius: 50%;`;
  button.onclick = function () {
    addTh();
    startTest();
  };
  button.appendChild(img);
  document.body.appendChild(button);
}

let fetchCount = 0;
const Maxthread = 25;
const Timeout = 10000;

function startTest() {
  let nodes = document.querySelectorAll('td[scope="row"]');

  let j = 0;
  let timerId = setInterval(function () {
    for (; j < nodes.length; j++) {
      if (fetchCount <= Maxthread) {
        domainTest(nodes[j]);
        if (j === nodes.length - 1) {
          clearInterval(timerId);
          alert("测试完成!");
        }
        fetchCount++;
      } else {
        break;
      }
    }
  }, 1000);
}

function addTh() {
  if (document.querySelector("thead > tr").childElementCount === 13) {
    return;
  }

  // 添加表头
  let th = document.createElement("th");
  th.setAttribute("class", "nosort");
  th.setAttribute("data-sortcolumn", "3");
  th.setAttribute("data-sortkey", "3-0");
  th.innerText = "Connecting Time";
  document.querySelector('th[data-sortcolumn="2"]').after(th);

  // 修改相应数值
  let next = th.nextElementSibling;
  for (let i = 4; next; i++) {
    next.setAttribute("data-sortcolumn", `${i}`);
    next.setAttribute("data-sortkey", `${i}-0`);
    next = next.nextElementSibling;
  }
}

function domainTest(node) {
  let domain = node.getAttribute("data-value");
  let url = `https://${domain}/api/v1/instance?timestamp=${Date.now()}`;

  // 初始化
  let message;
  if (node.parentElement.childElementCount !== 13) {
    message = document.createElement("td");
    message.setAttribute("data-value", "");
    node.after(message);
  } else {
    message = node.nextElementSibling;
    message.setAttribute("data-value", "");
    message.innerText = "";
  }

  // 开始测试
  let startTime;
  let endTime;
  startTime = new Date().getTime();
  GM.xmlHttpRequest({
    method: "GET",
    url: url,
    onload(response) {
      endTime = new Date().getTime();
      if (response.status === 200) {
        let duration = endTime - startTime;
        message.setAttribute("data-value", duration);
        message.innerText = `${duration}`;
      } else {
        message.setAttribute("data-value", 9999999);
        message.innerHTML = '<span title="response.status != 200">❌</span>';
      }
      fetchCount--;
    },
    onerror(response) {
      message.setAttribute("data-value", 9999999);
      message.innerHTML = '<span title="error">❌</span>';
      fetchCount--;
    },
    ontimeout(response) {
      message.setAttribute("data-value", 9999999);
      message.innerHTML = '<span title="timeout">❌</span>';
      fetchCount--;
    },
    timeout: Timeout,
    synchronous: false,
  });
}