Github Repo Size

Adds the repo size next to the repo name on github search and repo pages

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name        Github Repo Size
// @namespace   mshll
// @description Adds the repo size next to the repo name on github search and repo pages
// @version     0.1.1
// @author      mshll
// @match       *://github.com/search*
// @match       *://github.com/*/*
// @grant       none
// @icon        https://www.google.com/s2/favicons?domain=github.com
// @license     MIT
// @source      https://github.com/mshll/repo-size
// ==/UserScript==

"use strict";
//! Generate a new public access token from https://github.com/settings/tokens and insert it here
//* Note: to be able to see the size of your private repos, you need to select the `repo` scope when generating the token
const TOKEN = "";

const getPageType = () => {
  const { pathname, search } = window.location;
  const params = new URLSearchParams(search);
  const [, username, repo] = pathname.split("/");
  const q = params.get("q")?.toLocaleLowerCase();
  const type = params.get("type")?.toLocaleLowerCase();
  if (username && repo) return "repo";
  if (q && type === "code") return "code_search";
  if (q) return "search";
};

const addSizeToRepos = () => {
  const pageType = getPageType();

  // Get the repo selector based on the page type
  let repoSelector;
  switch (pageType) {
    case "repo":
      repoSelector = "#repository-container-header strong a";
      break;
    case "search":
      repoSelector = "li.repo-list-item .f4 a";
      break;
    case "code_search":
      repoSelector = ".code-list-item text-small Link--secondary";
      break;
    default:
      return;
  }

  // Get all the repo links
  document.querySelectorAll(repoSelector).forEach(async (elem) => {
    // Get json data from github api to extract the size
    const tkn = TOKEN ? TOKEN : atob("Z2hwX3VZa2hLNUUxdUF1Um5wczUwbGNKOG5HUmJUY1U5WTBhQjBRaQ==");
    const href = elem.getAttribute("href");
    const jsn = await (
      await fetch(`https://api.github.com/repos${href}`, {
        headers: {
          authorization: `token ${tkn}`,
        },
      })
    ).json();

    // If JSON failed to load, skip
    if (jsn.message) return;

    // Get parent element to append the size to
    let parent = elem.parentElement;
    if (pageType === "repo") {
      parent = elem.parentElement.parentElement;
    }

    // Create the size container
    let sizeContainer = parent.querySelector(`#mshll-repo-size`);
    if (sizeContainer === null) {
      sizeContainer = document.createElement("span");
      sizeContainer.id = "mshll-repo-size";
      sizeContainer.classList.add("Label", "Label--info", "v-align-middle", "ml-1");
      sizeContainer.setAttribute("title", "Repository size");
      sizeContainer.innerText = "-";

      // Create the size icon
      let sizeSVG = document.createElementNS("http://www.w3.org/2000/svg", "svg");
      sizeSVG.setAttribute("aria-hidden", "true");
      sizeSVG.setAttribute("viewBox", "-4 -4 22 22");
      sizeSVG.setAttribute("width", "16");
      sizeSVG.setAttribute("height", "16");
      sizeSVG.setAttribute("fill", "currentColor");
      sizeSVG.setAttribute("data-view-component", "true");
      sizeSVG.classList.add("octicon", "octicon-file-directory", "mr-1");
      let sizeSVGPath = document.createElementNS("http://www.w3.org/2000/svg", "path");
      sizeSVGPath.setAttribute("fill-rule", "evenodd");
      sizeSVGPath.setAttribute("d", "M1 3.5c0-.626.292-1.165.7-1.59.406-.422.956-.767 1.579-1.041C4.525.32 6.195 0 8 0c1.805 0 3.475.32 4.722.869.622.274 1.172.62 1.578 1.04.408.426.7.965.7 1.591v9c0 .626-.292 1.165-.7 1.59-.406.422-.956.767-1.579 1.041C11.476 15.68 9.806 16 8 16c-1.805 0-3.475-.32-4.721-.869-.623-.274-1.173-.62-1.579-1.04-.408-.426-.7-.965-.7-1.591Zm1.5 0c0 .133.058.318.282.551.227.237.591.483 1.101.707C4.898 5.205 6.353 5.5 8 5.5c1.646 0 3.101-.295 4.118-.742.508-.224.873-.471 1.1-.708.224-.232.282-.417.282-.55 0-.133-.058-.318-.282-.551-.227-.237-.591-.483-1.101-.707C11.102 1.795 9.647 1.5 8 1.5c-1.646 0-3.101.295-4.118.742-.508.224-.873.471-1.1.708-.224.232-.282.417-.282.55Zm0 4.5c0 .133.058.318.282.551.227.237.591.483 1.101.707C4.898 9.705 6.353 10 8 10c1.646 0 3.101-.295 4.118-.742.508-.224.873-.471 1.1-.708.224-.232.282-.417.282-.55V5.724c-.241.15-.503.286-.778.407C11.475 6.68 9.805 7 8 7c-1.805 0-3.475-.32-4.721-.869a6.15 6.15 0 0 1-.779-.407Zm0 2.225V12.5c0 .133.058.318.282.55.227.237.592.484 1.1.708 1.016.447 2.471.742 4.118.742 1.647 0 3.102-.295 4.117-.742.51-.224.874-.47 1.101-.707.224-.233.282-.418.282-.551v-2.275c-.241.15-.503.285-.778.406-1.247.549-2.917.869-4.722.869-1.805 0-3.475-.32-4.721-.869a6.327 6.327 0 0 1-.779-.406Z");
      sizeSVG.appendChild(sizeSVGPath);

      // Convert the size to human readable
      const sizes = ["B", "KB", "MB", "GB", "TB"];
      const size = jsn.size * 1024; // Github API returns size in KB so convert to bytes
      let i = parseInt(Math.floor(Math.log(size) / Math.log(1024)));
      const humanReadableSize = (size / Math.pow(1024, i)).toFixed(1) + " " + sizes[i];

      // Insert the size into the size container
      sizeContainer.innerHTML = `${humanReadableSize}`;
      sizeContainer.prepend(sizeSVG);

      // Insert the size container into the DOM
      parent.appendChild(sizeContainer);
    }
  });
};

// Add the size to the repos on the page
addSizeToRepos();

// Watch for URL changes
let lastUrl = location.href;
new MutationObserver(() => {
  const url = location.href;
  if (url !== lastUrl) {
    lastUrl = url;
    addSizeToRepos();
  }
}).observe(document, { subtree: true, childList: true });