Github Repo Size

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

Você precisará instalar uma extensão como Tampermonkey, Greasemonkey ou Violentmonkey para instalar este script.

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

Você precisará instalar uma extensão como Tampermonkey ou Violentmonkey para instalar este script.

Você precisará instalar uma extensão como Tampermonkey ou Userscripts para instalar este script.

Você precisará instalar uma extensão como o Tampermonkey para instalar este script.

Você precisará instalar um gerenciador de scripts de usuário para instalar este script.

(Eu já tenho um gerenciador de scripts de usuário, me deixe instalá-lo!)

Você precisará instalar uma extensão como o Stylus para instalar este estilo.

Você precisará instalar uma extensão como o Stylus para instalar este estilo.

Você precisará instalar uma extensão como o Stylus para instalar este estilo.

Você precisará instalar um gerenciador de estilos de usuário para instalar este estilo.

Você precisará instalar um gerenciador de estilos de usuário para instalar este estilo.

Você precisará instalar um gerenciador de estilos de usuário para instalar este estilo.

(Eu já possuo um gerenciador de estilos de usuário, me deixar fazer a instalação!)

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