Github Repo Size

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

Du musst eine Erweiterung wie Tampermonkey, Greasemonkey oder Violentmonkey installieren, um dieses Skript zu installieren.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

Sie müssten eine Skript Manager Erweiterung installieren damit sie dieses Skript installieren können

(Ich habe schon ein Skript Manager, Lass mich es installieren!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==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 });