This commit is contained in:
dragsbruh 2026-06-13 14:06:38 +05:30
commit 1745c1cfa7
14 changed files with 17598 additions and 0 deletions

12
.editorconfig Normal file
View File

@ -0,0 +1,12 @@
# http://editorconfig.org
root = true
[*]
indent_style = tab
end_of_line = lf
charset = utf-8
trim_trailing_whitespace = true
insert_final_newline = true
[*.yml]
indent_style = space

167
.gitignore vendored Normal file
View File

@ -0,0 +1,167 @@
# Logs
logs
_.log
npm-debug.log_
yarn-debug.log*
yarn-error.log*
lerna-debug.log*
.pnpm-debug.log*
# Diagnostic reports (https://nodejs.org/api/report.html)
report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
# Runtime data
pids
_.pid
_.seed
\*.pid.lock
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
\*.lcov
# nyc test coverage
.nyc_output
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# Bower dependency directory (https://bower.io/)
bower_components
# node-waf configuration
.lock-wscript
# Compiled binary addons (https://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules/
jspm_packages/
# Snowpack dependency directory (https://snowpack.dev/)
web_modules/
# TypeScript cache
\*.tsbuildinfo
# Optional npm cache directory
.npm
# Optional eslint cache
.eslintcache
# Optional stylelint cache
.stylelintcache
# Microbundle cache
.rpt2_cache/
.rts2_cache_cjs/
.rts2_cache_es/
.rts2_cache_umd/
# Optional REPL history
.node_repl_history
# Output of 'npm pack'
\*.tgz
# Yarn Integrity file
.yarn-integrity
# parcel-bundler cache (https://parceljs.org/)
.cache
.parcel-cache
# Next.js build output
.next
out
# Nuxt.js build / generate output
.nuxt
dist
# Gatsby files
.cache/
# Comment in the public line in if your project uses Gatsby and not Next.js
# https://nextjs.org/blog/next-9-1#public-directory-support
# public
# vuepress build output
.vuepress/dist
# vuepress v2.x temp and cache directory
.temp
.cache
# Docusaurus cache and generated files
.docusaurus
# Serverless directories
.serverless/
# FuseBox cache
.fusebox/
# DynamoDB Local files
.dynamodb/
# TernJS port file
.tern-port
# Stores VSCode versions used for testing VSCode extensions
.vscode-test
# yarn v2
.yarn/cache
.yarn/unplugged
.yarn/build-state.yml
.yarn/install-state.gz
.pnp.\*
# wrangler project
.dev.vars*
!.dev.vars.example
.env*
!.env.example
.wrangler/

5
.ignore Normal file
View File

@ -0,0 +1,5 @@
pnpm-lock.yaml
pnpm-workspace.yaml
tsconfig.json
worker-configuration.d.ts
wrangler.jsonc

6
.prettierrc Normal file
View File

@ -0,0 +1,6 @@
{
"printWidth": 140,
"singleQuote": true,
"semi": true,
"useTabs": true
}

21
package.json Normal file
View File

@ -0,0 +1,21 @@
{
"name": "contribapi",
"version": "0.0.0",
"private": true,
"scripts": {
"deploy": "wrangler deploy",
"dev": "wrangler dev",
"start": "wrangler dev",
"test": "vitest",
"cf-typegen": "wrangler types"
},
"devDependencies": {
"@cloudflare/vitest-pool-workers": "^0.12.21",
"@cloudflare/workers-types": "^4.20260530.1",
"typescript": "^5.9.3",
"wrangler": "^3.114.17"
},
"dependencies": {
"zod": "^4.4.3"
}
}

2739
pnpm-lock.yaml Normal file

File diff suppressed because it is too large Load Diff

4
pnpm-workspace.yaml Normal file
View File

@ -0,0 +1,4 @@
allowBuilds:
esbuild: true
sharp: true
workerd: true

16
src/codeberg.ts Normal file
View File

@ -0,0 +1,16 @@
import { z } from "zod";
import { Day, daySchema, USER_AGENT } from "./common";
export const codebergResponse = z.array(daySchema);
export async function queryCodeberg(env: Env): Promise<{ codeberg: Day[] }> {
const response = await fetch(
`https://soxy.dragsbruh.workers.dev/?u=https://codeberg.org/api/v1/users/${env.CODEBERG_USERNAME}/heatmap`,
{
headers: {
"User-Agent": USER_AGENT,
},
},
);
return { codeberg: codebergResponse.parse(await response.json()) };
}

10
src/common.ts Normal file
View File

@ -0,0 +1,10 @@
import { z } from "zod";
export const USER_AGENT = "contribapi/0.1.0-dev (https://codeberg.org/dragsbruh/contribapi)";
export const daySchema = z.strictObject({
timestamp: z.int().positive(),
contributions: z.int().positive(),
});
export type Day = z.infer<typeof daySchema>;

66
src/github.ts Normal file
View File

@ -0,0 +1,66 @@
import { z } from "zod";
import { Day, USER_AGENT } from "./common";
const graphqlQuery = `query($username: String!) {
user(login: $username) {
contributionsCollection {
contributionCalendar {
weeks {
contributionDays {
date
contributionCount
color
}
}
}
}
}
}`;
export const githubResponse = z
.object({
data: z.object({
user: z.object({
contributionsCollection: z.object({
contributionCalendar: z.object({
weeks: z.array(
z.object({
contributionDays: z.array(
z
.object({
date: z.coerce.date(),
contributionCount: z.number(),
})
.transform((o) => ({
timestamp: Math.floor(o.date.getTime() / 1000),
contributions: o.contributionCount,
})),
),
}),
),
}),
}),
}),
}),
})
.transform((o) => ({
github: o.data.user.contributionsCollection.contributionCalendar.weeks
.map((d) => d.contributionDays)
.flat(),
}));
export async function queryGithub(env: Env): Promise<{ github: Day[] }> {
const response = await fetch("https://api.github.com/graphql", {
method: "POST",
headers: {
"User-Agent": USER_AGENT,
"Content-Type": "application/json",
Authorization: `Bearer ${env.GITHUB_TOKEN}`,
},
body: JSON.stringify({
query: graphqlQuery,
variables: { username: env.GITHUB_USERNAME },
}),
});
return githubResponse.parse(await response.json());
}

30
src/index.ts Normal file
View File

@ -0,0 +1,30 @@
import { queryCodeberg } from "./codeberg";
import { queryGithub } from "./github";
const corsHeaders = {
"Access-Control-Allow-Origin": "*",
"Access-Control-Allow-Methods": "GET, OPTIONS",
"Access-Control-Allow-Headers": "Content-Type",
};
export default {
async fetch(request, env): Promise<Response> {
if (request.method === "OPTIONS") {
return new Response(null, {
headers: corsHeaders,
});
} else if (request.method != "GET") {
return new Response("OwO wutz that");
}
const responses = await Promise.all([queryGithub(env), queryCodeberg(env)]);
const merged = Object.assign({}, ...responses);
return Response.json(merged, {
headers: {
...corsHeaders,
"Cache-Control": "public, max-age=3600",
},
});
},
} satisfies ExportedHandler<Env>;

30
tsconfig.json Normal file
View File

@ -0,0 +1,30 @@
{
"compilerOptions": {
"target": "es2022",
"lib": [
"es2022"
],
"jsx": "react-jsx",
"module": "es2022",
"moduleResolution": "Bundler",
"resolveJsonModule": true,
"allowJs": true,
"checkJs": false,
"noEmit": true,
"isolatedModules": true,
"allowSyntheticDefaultImports": true,
"forceConsistentCasingInFileNames": true,
"strict": true,
"skipLibCheck": true,
"types": [
"@cloudflare/workers-types"
]
},
"exclude": [
"test"
],
"include": [
"worker-configuration.d.ts",
"src/**/*.ts"
]
}

14484
worker-configuration.d.ts vendored Normal file

File diff suppressed because it is too large Load Diff

8
wrangler.jsonc Normal file
View File

@ -0,0 +1,8 @@
{
"name": "contrib",
"main": "src/index.ts",
"compatibility_date": "2026-05-31",
"observability": {
"enabled": true
}
}