diff --git a/.github/workflows/test.yaml b/.github/workflows/test.yaml index 9ccec7d..366abbe 100644 --- a/.github/workflows/test.yaml +++ b/.github/workflows/test.yaml @@ -5,5 +5,7 @@ jobs: Test-GitHub-Actions: runs-on: ubuntu-latest steps: + - uses: actions/checkout@v3 + - uses: ./ # with: diff --git a/.gitignore b/.gitignore index dd59e30..fbc2b7d 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,3 @@ -build node_modules .env diff --git a/action.yaml b/action.yaml index 86b8c90..fa4007b 100644 --- a/action.yaml +++ b/action.yaml @@ -2,4 +2,4 @@ name: "profile-stats" description: "retrieve github profile stats and save locally to repo for further processing" runs: using: "node20" - main: "build/index.js" + main: "./build/index.js" diff --git a/build/Types.js b/build/Types.js new file mode 100644 index 0000000..dbb2d86 --- /dev/null +++ b/build/Types.js @@ -0,0 +1,11 @@ +export const NOT_LANGUAGES = [ + 'html', + 'markdown', + 'dockerfile', + 'roff', + 'rich text format', + 'powershell', + 'css', + 'php', +]; +//# sourceMappingURL=Types.js.map \ No newline at end of file diff --git a/build/Types.js.map b/build/Types.js.map new file mode 100644 index 0000000..b0f00a2 --- /dev/null +++ b/build/Types.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Types.js","sourceRoot":"","sources":["../src/Types.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,aAAa,GAAG;IAC5B,MAAM;IACN,UAAU;IACV,YAAY;IACZ,MAAM;IACN,kBAAkB;IAClB,YAAY;IACZ,KAAK;IACL,KAAK;CACL,CAAC"} \ No newline at end of file diff --git a/build/index.js b/build/index.js new file mode 100644 index 0000000..4047c82 --- /dev/null +++ b/build/index.js @@ -0,0 +1,114 @@ +import core from "@actions/core"; +import { Octokit } from "octokit"; +import { config } from "dotenv"; +import { getContributionCollection, getGraphQLData, getReposContributorsStats, getReposViewCount, getTotalCommits, } from "./octokit"; +import { writeFileSync } from "fs"; +config(); +export const NOT_LANGUAGES = [ + "html", + "markdown", + "dockerfile", + "roff", + "rich text format", + "powershell", + "css", + "php", +]; +const NOT_LANGUAGES_OBJ = Object.fromEntries(NOT_LANGUAGES.map((l) => [l, true])); +try { + const token = process.env["GITHUB_TOKEN"]; + if (!token) + throw new Error("GITHUB_TOKEN is not present"); + const octokit = new Octokit({ auth: token }); + const fetchedAt = Date.now(); + const userDetails = await octokit.rest.users.getAuthenticated(); + const username = userDetails.data.login; + const accountCreationDate = userDetails.data.created_at; + const [graphQLData, totalCommits, contributionsCollection] = await Promise.all([ + getGraphQLData(octokit, username), + getTotalCommits(octokit, username), + getContributionCollection(octokit, accountCreationDate), + ]); + console.log(userDetails); + console.log(graphQLData); + console.log(totalCommits); + console.log(contributionsCollection); + let starCount = 0; + let forkCount = 0; + for (const repo of graphQLData.user.repositories.nodes) { + starCount += repo.stargazers.totalCount; + forkCount += repo.forkCount; + } + const contributorStatsPromises = []; + const viewCountPromises = []; + for (const repo of graphQLData.user.repositories.nodes) { + contributorStatsPromises.push(getReposContributorsStats(octokit, username, repo.name)); + viewCountPromises.push(getReposViewCount(octokit, username, repo.name)); + } + const contributorStats = (await Promise.all(contributorStatsPromises)) + .filter((entry) => entry !== null || entry !== undefined) + .map((entry) => { + return (Array.isArray(entry.data) ? entry.data : [entry.data]) + .filter((contributor) => contributor.author?.login === userDetails.data.login) + .map((contributor) => contributor.weeks); + }); + let linesOfCodeChanged = 0; + for (const repo of contributorStats) { + for (const week of repo) { + for (const day of week) { + linesOfCodeChanged += (day.a || 0) + (day.d || 0) + (day.c || 0); + } + } + } + const viewCounts = await Promise.all(viewCountPromises); + let repoViews = 0; + for (const viewCount of viewCounts) { + repoViews += viewCount.data.count; + } + const topLanguages = []; + let codeByteTotal = 0; + for (const node of graphQLData.user.repositories.nodes) { + for (const edge of node.languages.edges) { + if (NOT_LANGUAGES_OBJ[edge.node.name.toLowerCase()]) { + continue; + } + const existingLanguage = topLanguages.find((l) => l.languageName === edge.node.name); + if (existingLanguage) { + existingLanguage.value += edge.size; + codeByteTotal += edge.size; + } + else { + topLanguages.push({ + languageName: edge.node.name, + color: edge.node.color, + value: edge.size, + }); + codeByteTotal += edge.size; + } + } + } + const allDays = contributionsCollection.contributionCalendar.weeks + .map((w) => w.contributionDays) + .flat(1); + writeFileSync("github-user-stats.json", JSON.stringify({ + name: userDetails.data.name || "", + username, + repoViews, + linesOfCodeChanged, + totalCommits: totalCommits.data.total_count, + totalPullRequests: graphQLData.user.pullRequests.totalCount, + codeByteTotal, + topLanguages, + forkCount, + starCount, + totalContributions: contributionsCollection.contributionCalendar.totalContributions, + closedIssues: graphQLData.viewer.closedIssues.totalCount, + openIssues: graphQLData.viewer.openIssues.totalCount, + fetchedAt, + contributionData: allDays, + }, null, 4)); +} +catch (error) { + core.setFailed(error); +} +//# sourceMappingURL=index.js.map \ No newline at end of file diff --git a/build/index.js.map b/build/index.js.map new file mode 100644 index 0000000..17cdb2f --- /dev/null +++ b/build/index.js.map @@ -0,0 +1 @@ +{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"} \ No newline at end of file diff --git a/build/octokit.js b/build/octokit.js new file mode 100644 index 0000000..19d0536 --- /dev/null +++ b/build/octokit.js @@ -0,0 +1,195 @@ +export async function getGraphQLData(octokit, username) { + return octokit.graphql.paginate(`query userInfo($login: String!, $cursor: String) { + user(login: $login) { + name + login + repositories( + orderBy: {field: STARGAZERS, direction: DESC} + ownerAffiliations: OWNER + isFork: false + first: 100 + after: $cursor + ) { + totalCount + nodes { + stargazers { + totalCount + } + forkCount + name + languages(first: 10, orderBy: {field: SIZE, direction: DESC}) { + edges { + size + node { + + color + name + } + } + } + } + pageInfo { + endCursor + hasNextPage + } + } + pullRequests(first: 1) { + totalCount + } + } + viewer { + openIssues: issues(states: OPEN) { + totalCount + } + closedIssues: issues(states: CLOSED) { + totalCount + } + } + rateLimit { + limit + remaining + used + resetAt + } + }`, { + login: username, + }); +} +export async function getContributionCollection(octokit, year) { + const yearCreated = new Date(year); + const currentYear = new Date(); + const promises = []; + for (let i = yearCreated.getFullYear(); i <= currentYear.getFullYear(); i++) { + let startYear = `${i}-01-01T00:00:00.000Z`; + if (i === yearCreated.getFullYear()) + startYear = year; + let endYear = `${i + 1}-01-01T00:00:00.000Z`; + if (i === currentYear.getFullYear()) + endYear = currentYear.toISOString(); + promises.push(octokit + .graphql(`query { + rateLimit { + limit + remaining + used + resetAt + } + viewer { + contributionsCollection(from: "${startYear}", to: "${endYear}") { + totalCommitContributions + restrictedContributionsCount + totalIssueContributions + totalCommitContributions + totalRepositoryContributions + totalPullRequestContributions + totalPullRequestReviewContributions + popularPullRequestContribution { + pullRequest { + id + title + repository { + name + owner { + login + } + } + } + } + contributionCalendar { + totalContributions + weeks { + contributionDays { + contributionCount + date + } + } + } + commitContributionsByRepository { + contributions { + totalCount + } + repository { + name + owner { + login + } + languages(first: 5, orderBy: { field: SIZE, direction: DESC }) { + edges { + size + node { + color + name + id + } + } + } + } + } + } + } + } + `) + .catch((error) => { + throw new Error(`Failed to fetch data for year ${i}: ${error.message}`); + })); + } + console.log(promises); + const years = (await Promise.all(promises)).filter(Boolean); + console.debug(years); + if (years.length === 0) { + throw new Error('Failed to fetch data for all years'); + } + const { contributionsCollection } = years[0].viewer; + for (const year of years.slice(1)) { + contributionsCollection.commitContributionsByRepository = [ + ...contributionsCollection.commitContributionsByRepository, + ...year.viewer.contributionsCollection.commitContributionsByRepository, + ]; + contributionsCollection.contributionCalendar.totalContributions += + year.viewer.contributionsCollection.contributionCalendar.totalContributions; + contributionsCollection.contributionCalendar.weeks = [ + ...contributionsCollection.contributionCalendar.weeks, + ...year.viewer.contributionsCollection.contributionCalendar.weeks, + ]; + } + return contributionsCollection; +} +export async function getTotalCommits(octokit, username) { + return octokit.rest.search.commits({ + q: `author:${username}`, + }); +} +export async function getUsersStars(octokit, username) { + return octokit.rest.activity.listReposStarredByUser({ + username, + }); +} +export async function getReposContributorsStats(octokit, username, repo) { + return octokit.rest.repos + .getContributorsStats({ + owner: username, + repo, + }) + .then((res) => { + if (res.status === 202) { + setTimeout(() => { + return octokit.rest.repos.getContributorsStats({ + owner: username, + repo, + }); + }, 2000); + } + return res; + }) + .catch((error) => { + throw new Error(`Failed to fetch data for repo ${repo}: ${error.message}`); + }); +} +export async function getReposViewCount(octokit, username, repo) { + return octokit.rest.repos.getViews({ + per: 'week', + owner: username, + repo, + }); +} +//# sourceMappingURL=octokit.js.map \ No newline at end of file diff --git a/build/octokit.js.map b/build/octokit.js.map new file mode 100644 index 0000000..76c543a --- /dev/null +++ b/build/octokit.js.map @@ -0,0 +1 @@ +{"version":3,"file":"octokit.js","sourceRoot":"","sources":["../src/octokit.ts"],"names":[],"mappings":"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"} \ No newline at end of file