1
mirror of https://github.com/jakejarvis/jarv.is.git synced 2025-07-03 18:06:38 -04:00

use @octokit/graphql instead of plain graphql for /api/projects

This commit is contained in:
2021-07-14 07:26:07 -04:00
parent 38ba41ca0b
commit 9c26cd7bff
9 changed files with 270 additions and 227 deletions

View File

@ -1,19 +1,17 @@
/// <reference types="./types/hits" />
import * as Sentry from "@sentry/node";
import * as Tracing from "@sentry/tracing"; // eslint-disable-line @typescript-eslint/no-unused-vars
import { VercelRequest, VercelResponse } from "@vercel/node";
import { Client, query as q } from "faunadb";
import fetch from "node-fetch";
import parser from "fast-xml-parser";
import { decode } from "html-entities";
import type { PageStats, OverallStats } from "./types/hits";
const baseUrl = "https://jarv.is/";
Sentry.init({
dsn: process.env.SENTRY_DSN || "",
environment: process.env.NODE_ENV || process.env.VERCEL_ENV || process.env.SENTRY_ENVIRONMENT || "",
tracesSampleRate: 1.0,
});
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types

View File

@ -1,19 +1,13 @@
/// <reference types="./types/projects" />
import * as Sentry from "@sentry/node";
import * as Tracing from "@sentry/tracing"; // eslint-disable-line @typescript-eslint/no-unused-vars
import { VercelRequest, VercelResponse } from "@vercel/node";
import { graphql, GraphQlQueryResponseData } from "@octokit/graphql";
import { encode } from "html-entities";
import { GraphQLClient } from "graphql-request";
import { gql } from "graphql-tag";
const username = "jakejarvis";
const endpoint = "https://api.github.com/graphql";
import type { Repository } from "./types/projects";
Sentry.init({
dsn: process.env.SENTRY_DSN || "",
environment: process.env.NODE_ENV || process.env.VERCEL_ENV || process.env.SENTRY_ENVIRONMENT || "",
tracesSampleRate: 1.0,
});
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types
@ -27,19 +21,21 @@ export default async (req: VercelRequest, res: VercelResponse) => {
throw new Error("GitHub API credentials aren't set.");
}
// default to latest repos
let sortBy = "PUSHED_AT";
// get most popular repos (/projects/?top)
if (typeof req.query.top !== "undefined") sortBy = "STARGAZERS";
const repos = await fetchRepos(sortBy, 16);
let result;
if (typeof req.query.top !== "undefined") {
// get most popular repos (/projects/?top)
result = await fetchRepos("STARGAZERS");
} else {
// default to latest repos
result = await fetchRepos("PUSHED_AT");
}
// let Vercel edge and browser cache results for 15 mins
res.setHeader("Cache-Control", "public, max-age=900, s-maxage=900, stale-while-revalidate");
res.setHeader("Access-Control-Allow-Methods", "GET");
res.setHeader("Access-Control-Allow-Origin", "*");
res.status(200).json(repos);
res.status(200).json(result);
} catch (error) {
console.error(error);
@ -52,58 +48,53 @@ export default async (req: VercelRequest, res: VercelResponse) => {
}
};
const fetchRepos = async (sort: string, limit: number): Promise<Repository[]> => {
// https://docs.github.com/en/graphql/guides/forming-calls-with-graphql
const client = new GraphQLClient(endpoint, {
headers: {
Authorization: `Bearer ${process.env.GH_PUBLIC_TOKEN}`,
Accept: "application/vnd.github.v3+json",
},
});
const fetchRepos = async (sort: string): Promise<Repository[]> => {
// https://docs.github.com/en/graphql/reference/objects#repository
const query = gql`
query ($sort: String, $limit: Int) {
user(login: "${username}") {
repositories(
first: $limit,
isLocked: false,
isFork: false,
ownerAffiliations: OWNER,
privacy: PUBLIC,
orderBy: {
field: $sort,
direction: DESC
}
) {
edges {
node {
name
url
description
pushedAt
stargazerCount
forkCount
primaryLanguage {
const { user } = await graphql<GraphQlQueryResponseData>(
`
query ($username: String!, $sort: String, $limit: Int) {
user(login: $username) {
repositories(
first: $limit
isLocked: false
isFork: false
ownerAffiliations: OWNER
privacy: PUBLIC
orderBy: { field: $sort, direction: DESC }
) {
edges {
node {
name
color
url
description
pushedAt
stargazerCount
forkCount
primaryLanguage {
name
color
}
}
}
}
}
}
`,
{
username: "jakejarvis",
limit: 16,
sort: sort,
headers: {
authorization: `token ${process.env.GH_PUBLIC_TOKEN}`,
},
}
`;
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
const response = await client.request(query, { sort, limit });
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
const currentRepos: Repository[] = response.user.repositories.edges.map(
({ node: repo }: { [key: string]: Repository }) => ({
...repo,
description: encode(repo.description),
})
);
return currentRepos;
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access
const repos: Repository[] = user.repositories.edges.map(({ node: repo }: { [key: string]: Repository }) => ({
...repo,
description: encode(repo.description),
}));
return repos;
};

View File

@ -1,14 +1,13 @@
/// <reference types="./types/tracks" />
// Fetches my Spotify most-played tracks or currently playing track.
// Heavily inspired by @leerob: https://leerob.io/snippets/spotify
import * as Sentry from "@sentry/node";
import * as Tracing from "@sentry/tracing"; // eslint-disable-line @typescript-eslint/no-unused-vars
import { VercelRequest, VercelResponse } from "@vercel/node";
import fetch from "node-fetch";
import * as queryString from "query-string";
import type { Track, TrackSchema, Activity } from "./types/tracks";
const { SPOTIFY_CLIENT_ID, SPOTIFY_CLIENT_SECRET, SPOTIFY_REFRESH_TOKEN } = process.env;
const basic = Buffer.from(`${SPOTIFY_CLIENT_ID}:${SPOTIFY_CLIENT_SECRET}`).toString("base64");
@ -23,7 +22,6 @@ const TOP_TRACKS_ENDPOINT = `https://api.spotify.com/v1/me/top/tracks?time_range
Sentry.init({
dsn: process.env.SENTRY_DSN || "",
environment: process.env.NODE_ENV || process.env.VERCEL_ENV || process.env.SENTRY_ENVIRONMENT || "",
tracesSampleRate: 1.0,
});
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types
@ -104,20 +102,13 @@ const getNowPlaying = async (): Promise<Track> => {
const active: Activity = await response.json();
if (active.is_playing === true && active.item) {
const isPlaying = active.is_playing;
const artist = active.item.artists.map((_artist) => _artist.name).join(", ");
const title = active.item.name;
const album = active.item.album.name;
const imageUrl = active.item.album.images[0].url;
const songUrl = active.item.external_urls.spotify;
return {
isPlaying,
artist,
title,
album,
imageUrl,
songUrl,
isPlaying: active.is_playing,
artist: active.item.artists.map((_artist) => _artist.name).join(", "),
title: active.item.name,
album: active.item.album.name,
imageUrl: active.item.album.images[0].url,
songUrl: active.item.external_urls.spotify,
};
} else {
return { isPlaying: false };

8
api/types/hits.d.ts vendored
View File

@ -1,12 +1,12 @@
type PageStats = {
export type PageStats = {
slug: string;
hits: number;
title?: string;
url?: string;
date?: string;
slug: string;
hits: number;
};
type OverallStats = {
export type OverallStats = {
total: {
hits: number;
};

View File

@ -1,11 +1,10 @@
type Repository = {
import type { Language } from "@octokit/graphql-schema";
export type Repository = {
name: string;
url: string;
description: string;
primaryLanguage?: {
color: string;
name: string;
};
primaryLanguage?: Language;
stargazerCount: number;
forkCount: number;
pushedAt: string;

View File

@ -1,4 +1,4 @@
type TrackSchema = {
export type TrackSchema = {
name: string;
artists: Array<{
name: string;
@ -15,7 +15,7 @@ type TrackSchema = {
};
};
type Track = {
export type Track = {
isPlaying: boolean;
artist?: string;
title?: string;
@ -24,7 +24,7 @@ type Track = {
songUrl?: string;
};
type Activity = {
export type Activity = {
is_playing: boolean;
item?: TrackSchema;
};