From 5259a3276ebc65568c49bf5b0baebb91e6229c54 Mon Sep 17 00:00:00 2001 From: Faisal Bhuiyan Date: Tue, 26 Nov 2024 21:17:22 +0600 Subject: [PATCH] init project and setup neondb --- .env.example | 14 ++++ .eslintrc.cjs | 42 ++++++++++++ .gitignore | 46 ++++++++++++++ README.md | 29 +++++++++ bun.lockb | Bin 0 -> 164805 bytes next.config.js | 10 +++ package.json | 57 +++++++++++++++++ postcss.config.js | 5 ++ prettier.config.js | 4 ++ prisma/schema.prisma | 20 ++++++ public/favicon.ico | Bin 0 -> 23600 bytes src/app/_components/post.tsx | 50 +++++++++++++++ src/app/api/trpc/[trpc]/route.ts | 34 ++++++++++ src/app/layout.tsx | 24 +++++++ src/app/page.tsx | 53 ++++++++++++++++ src/env.js | 44 +++++++++++++ src/server/api/root.ts | 23 +++++++ src/server/api/routers/post.ts | 31 +++++++++ src/server/api/trpc.ts | 106 +++++++++++++++++++++++++++++++ src/server/db.ts | 17 +++++ src/styles/globals.css | 3 + src/trpc/query-client.ts | 25 ++++++++ src/trpc/react.tsx | 76 ++++++++++++++++++++++ src/trpc/server.ts | 30 +++++++++ start-database.sh | 60 +++++++++++++++++ tailwind.config.ts | 14 ++++ tsconfig.json | 42 ++++++++++++ 27 files changed, 859 insertions(+) create mode 100644 .env.example create mode 100644 .eslintrc.cjs create mode 100644 .gitignore create mode 100644 README.md create mode 100644 bun.lockb create mode 100644 next.config.js create mode 100644 package.json create mode 100644 postcss.config.js create mode 100644 prettier.config.js create mode 100644 prisma/schema.prisma create mode 100644 public/favicon.ico create mode 100644 src/app/_components/post.tsx create mode 100644 src/app/api/trpc/[trpc]/route.ts create mode 100644 src/app/layout.tsx create mode 100644 src/app/page.tsx create mode 100644 src/env.js create mode 100644 src/server/api/root.ts create mode 100644 src/server/api/routers/post.ts create mode 100644 src/server/api/trpc.ts create mode 100644 src/server/db.ts create mode 100644 src/styles/globals.css create mode 100644 src/trpc/query-client.ts create mode 100644 src/trpc/react.tsx create mode 100644 src/trpc/server.ts create mode 100644 start-database.sh create mode 100644 tailwind.config.ts create mode 100644 tsconfig.json diff --git a/.env.example b/.env.example new file mode 100644 index 0000000..cd1ae0b --- /dev/null +++ b/.env.example @@ -0,0 +1,14 @@ +# Since the ".env" file is gitignored, you can use the ".env.example" file to +# build a new ".env" file when you clone the repo. Keep this file up-to-date +# when you add new variables to `.env`. + +# This file will be committed to version control, so make sure not to have any +# secrets in it. If you are cloning this repo, create a copy of this file named +# ".env" and populate it with your secrets. + +# When adding additional environment variables, the schema in "/src/env.js" +# should be updated accordingly. + +# Prisma +# https://www.prisma.io/docs/reference/database-reference/connection-urls#env +DATABASE_URL="postgresql://postgres:password@localhost:5432/gitsage" diff --git a/.eslintrc.cjs b/.eslintrc.cjs new file mode 100644 index 0000000..a92fb0b --- /dev/null +++ b/.eslintrc.cjs @@ -0,0 +1,42 @@ +/** @type {import("eslint").Linter.Config} */ +const config = { + "parser": "@typescript-eslint/parser", + "parserOptions": { + "project": true + }, + "plugins": [ + "@typescript-eslint" + ], + "extends": [ + "next/core-web-vitals", + "plugin:@typescript-eslint/recommended-type-checked", + "plugin:@typescript-eslint/stylistic-type-checked" + ], + "rules": { + "@typescript-eslint/array-type": "off", + "@typescript-eslint/consistent-type-definitions": "off", + "@typescript-eslint/consistent-type-imports": [ + "warn", + { + "prefer": "type-imports", + "fixStyle": "inline-type-imports" + } + ], + "@typescript-eslint/no-unused-vars": [ + "warn", + { + "argsIgnorePattern": "^_" + } + ], + "@typescript-eslint/require-await": "off", + "@typescript-eslint/no-misused-promises": [ + "error", + { + "checksVoidReturn": { + "attributes": false + } + } + ] + } +} +module.exports = config; \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..c24a835 --- /dev/null +++ b/.gitignore @@ -0,0 +1,46 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.js + +# testing +/coverage + +# database +/prisma/db.sqlite +/prisma/db.sqlite-journal +db.sqlite + +# next.js +/.next/ +/out/ +next-env.d.ts + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* +.pnpm-debug.log* + +# local env files +# do not commit any .env files to git, except for the .env.example file. https://create.t3.gg/en/usage/env-variables#using-environment-variables +.env +.env*.local + +# vercel +.vercel + +# typescript +*.tsbuildinfo + +# idea files +.idea \ No newline at end of file diff --git a/README.md b/README.md new file mode 100644 index 0000000..67943c7 --- /dev/null +++ b/README.md @@ -0,0 +1,29 @@ +# Create T3 App + +This is a [T3 Stack](https://create.t3.gg/) project bootstrapped with `create-t3-app`. + +## What's next? How do I make an app with this? + +We try to keep this project as simple as possible, so you can start with just the scaffolding we set up for you, and add additional things later when they become necessary. + +If you are not familiar with the different technologies used in this project, please refer to the respective docs. If you still are in the wind, please join our [Discord](https://t3.gg/discord) and ask for help. + +- [Next.js](https://nextjs.org) +- [NextAuth.js](https://next-auth.js.org) +- [Prisma](https://prisma.io) +- [Drizzle](https://orm.drizzle.team) +- [Tailwind CSS](https://tailwindcss.com) +- [tRPC](https://trpc.io) + +## Learn More + +To learn more about the [T3 Stack](https://create.t3.gg/), take a look at the following resources: + +- [Documentation](https://create.t3.gg/) +- [Learn the T3 Stack](https://create.t3.gg/en/faq#what-learning-resources-are-currently-available) — Check out these awesome tutorials + +You can check out the [create-t3-app GitHub repository](https://github.com/t3-oss/create-t3-app) — your feedback and contributions are welcome! + +## How do I deploy this? + +Follow our deployment guides for [Vercel](https://create.t3.gg/en/deployment/vercel), [Netlify](https://create.t3.gg/en/deployment/netlify) and [Docker](https://create.t3.gg/en/deployment/docker) for more information. diff --git a/bun.lockb b/bun.lockb new file mode 100644 index 0000000000000000000000000000000000000000..88aba448edf9837b703012bba694e2cd7f98b091 GIT binary patch literal 164805 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 literal 0 HcmV?d00001 diff --git a/next.config.js b/next.config.js new file mode 100644 index 0000000..121c4f4 --- /dev/null +++ b/next.config.js @@ -0,0 +1,10 @@ +/** + * Run `build` or `dev` with `SKIP_ENV_VALIDATION` to skip env validation. This is especially useful + * for Docker builds. + */ +import "./src/env.js"; + +/** @type {import("next").NextConfig} */ +const config = {}; + +export default config; diff --git a/package.json b/package.json new file mode 100644 index 0000000..7e75bdb --- /dev/null +++ b/package.json @@ -0,0 +1,57 @@ +{ + "name": "gitsage", + "version": "0.1.0", + "private": true, + "type": "module", + "scripts": { + "build": "next build", + "check": "next lint && tsc --noEmit", + "db:generate": "prisma migrate dev", + "db:migrate": "prisma migrate deploy", + "db:push": "prisma db push", + "db:studio": "prisma studio", + "dev": "next dev --turbo", + "format:check": "prettier --check \"**/*.{ts,tsx,js,jsx,mdx}\" --cache", + "format:write": "prettier --write \"**/*.{ts,tsx,js,jsx,mdx}\" --cache", + "postinstall": "prisma generate", + "lint": "next lint", + "lint:fix": "next lint --fix", + "preview": "next build && next start", + "start": "next start", + "typecheck": "tsc --noEmit" + }, + "dependencies": { + "@prisma/client": "^5.14.0", + "@t3-oss/env-nextjs": "^0.10.1", + "@tanstack/react-query": "^5.50.0", + "@trpc/client": "^11.0.0-rc.446", + "@trpc/react-query": "^11.0.0-rc.446", + "@trpc/server": "^11.0.0-rc.446", + "geist": "^1.3.0", + "next": "^15.0.1", + "react": "^18.3.1", + "react-dom": "^18.3.1", + "server-only": "^0.0.1", + "superjson": "^2.2.1", + "zod": "^3.23.3" + }, + "devDependencies": { + "@types/eslint": "^8.56.10", + "@types/node": "^20.14.10", + "@types/react": "^18.3.3", + "@types/react-dom": "^18.3.0", + "@typescript-eslint/eslint-plugin": "^8.1.0", + "@typescript-eslint/parser": "^8.1.0", + "eslint": "^8.57.0", + "eslint-config-next": "^15.0.1", + "postcss": "^8.4.39", + "prettier": "^3.3.2", + "prettier-plugin-tailwindcss": "^0.6.5", + "prisma": "^5.14.0", + "tailwindcss": "^3.4.3", + "typescript": "^5.5.3" + }, + "ct3aMetadata": { + "initVersion": "7.38.1" + } +} diff --git a/postcss.config.js b/postcss.config.js new file mode 100644 index 0000000..01bf743 --- /dev/null +++ b/postcss.config.js @@ -0,0 +1,5 @@ +export default { + plugins: { + tailwindcss: {}, + }, +}; diff --git a/prettier.config.js b/prettier.config.js new file mode 100644 index 0000000..da332bd --- /dev/null +++ b/prettier.config.js @@ -0,0 +1,4 @@ +/** @type {import('prettier').Config & import('prettier-plugin-tailwindcss').PluginOptions} */ +export default { + plugins: ["prettier-plugin-tailwindcss"], +}; diff --git a/prisma/schema.prisma b/prisma/schema.prisma new file mode 100644 index 0000000..8aaf69d --- /dev/null +++ b/prisma/schema.prisma @@ -0,0 +1,20 @@ +// This is your Prisma schema file, +// learn more about it in the docs: https://pris.ly/d/prisma-schema + +generator client { + provider = "prisma-client-js" +} + +datasource db { + provider = "postgresql" + url = env("DATABASE_URL") +} + +model Post { + id Int @id @default(autoincrement()) + name String + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt + + @@index([name]) +} diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..b5336a48e6e7341e6e866c5f477d1a006dc6bf9e GIT binary patch literal 23600 zcmeI4YmXFF5QckMt_#9~A|eR0un<@-0Td&mM&yGDL5La*i5i4>L8E>%(V+2{5q^?? z!ER>HQ`Oydy3aY?Jv$3Mf%G$Vs@|$o)6>(}&9a4TF zKg<4m_B6L%y0q?b`nly`mYvD{@~}K=I+8Co#VeoB^xtjhhAtfID%1d1b*N@D#yK!} z14@QRPE;>VwZegct&j;zCFe!c2jiPMbtw>+9Gnxjf1rHjk{8xxNJDMDdJq=5I^ksZ z;MZyd{u-)I`aOORAPrZvZ^3#H)=n5j@O~-q8$2{<3CDUIR)hJM8M6srVBHJ-2IZTI z`ioHw+6{ZligsHERT+^(&}&{F`=clDQx}Nk4%9n*j$%ty&$m-J--U*Qw~*cj{S5&- zUAs2xviLEZqF~D$OqSbISAMd4F{)u zRru`TrQEV42tNjG6x_Ay^V!0StWBHHeNZyUY9)Is$zmV*CR??KXYfxxTqE~9>#bl< zln%~2s)E*gyj!6(O}y=WYKALt!YZM#;n%XmhTjw(`1G{B64UUuG{sU`Ps}Kt#bSa! zgnVFyS8cx7y1h1$LJBvTjQ42Mr6y(z)BrB3Y1t5788Ro)BwZb==$UV;G1hxC&@^@` z8AUk`!c+&V(5kH-<+Jot_|HCh?9~db_c*sga@8oGfMSUxiEK-Y9C zta5mL4_gRmu_g}Dn)-O>gT$AR57;8%um;7#-ll2kv2bB?ii8JkB=H6=yIQck2br+X z1%xb31=DCwu1JCRE+`7lyE=(VI@ln(6}&Sk&1R`p$8;!;?UP@oqG@cLLy@1T#QnoO zel>L-B0_X15cPD<&2#>FANi z4!n#VC^k98W$N$p|Kk;+yb0rL?7-N8u> zC0*N2##f+QjdY#SpvyF_KB6YjPtZUh7m#Vb8pd&yW@f| z4R>_=d`y6x62cS<+)nFD=ChM%sBWGmLoEDBK=8UkM0!>vw)k4sWGA}kb{09&{tF6$ zNmISnScqt|PyPr?{0qq;TPIgIpODXlh1dz8lD?5_y})VJMBV|~R^&rpS|{t@4XdIy z8;WuuizyI-ZgZ>Ff%ZV1a{0&;y$;nBkP3t!oI=bJ?ScCeQ*uD&J3THQCFK@GAj4JoyXP%bdp#m@4(F?j~pc4#O4FJd9k!dfV-R`$J7@b zk8Py(r-)fKRUQ~8bs&_Uqaa4kamy|%g9#uXWYWQ*>n)iF>?=81o_NJC9EuZh#U=Xn z7lHBsE2cnLb6Q=NvajO*~BQ$s(JFmBR@A3 z(FIDEjS!M=!S<#$#EzjqaNAs66I4ppIUETR%fu5b8r^2cYj8NvIIGsRd8Rr z%u>!K$irt0c7fvWJ9%FS%kwPOMbH)eUS5(*XzZ63ZKX5mTU8`9cps$0=^iw~KpiOk zO{>zUEyroSq<+KthNAos8eujaIHErfQpz@39=v0)3+&P9O6%G9(Xaodzb@@fj;(zy zQMv~80=xAJfZ}~3F~N?M&PZ@bGy_OwQ5DPS#1LJ<_wM%{`dVtJ*!&j4VJq eAi2Y^nY=fz>F@5yV+Y0#j2##|Fm_;GJMceGE{$0L literal 0 HcmV?d00001 diff --git a/src/app/_components/post.tsx b/src/app/_components/post.tsx new file mode 100644 index 0000000..b6b5cb4 --- /dev/null +++ b/src/app/_components/post.tsx @@ -0,0 +1,50 @@ +"use client"; + +import { useState } from "react"; + +import { api } from "@/trpc/react"; + +export function LatestPost() { + const [latestPost] = api.post.getLatest.useSuspenseQuery(); + + const utils = api.useUtils(); + const [name, setName] = useState(""); + const createPost = api.post.create.useMutation({ + onSuccess: async () => { + await utils.post.invalidate(); + setName(""); + }, + }); + + return ( +
+ {latestPost ? ( +

Your most recent post: {latestPost.name}

+ ) : ( +

You have no posts yet.

+ )} +
{ + e.preventDefault(); + createPost.mutate({ name }); + }} + className="flex flex-col gap-2" + > + setName(e.target.value)} + className="w-full rounded-full px-4 py-2 text-black" + /> + +
+
+ ); +} diff --git a/src/app/api/trpc/[trpc]/route.ts b/src/app/api/trpc/[trpc]/route.ts new file mode 100644 index 0000000..c397112 --- /dev/null +++ b/src/app/api/trpc/[trpc]/route.ts @@ -0,0 +1,34 @@ +import { fetchRequestHandler } from "@trpc/server/adapters/fetch"; +import { type NextRequest } from "next/server"; + +import { env } from "@/env"; +import { appRouter } from "@/server/api/root"; +import { createTRPCContext } from "@/server/api/trpc"; + +/** + * This wraps the `createTRPCContext` helper and provides the required context for the tRPC API when + * handling a HTTP request (e.g. when you make requests from Client Components). + */ +const createContext = async (req: NextRequest) => { + return createTRPCContext({ + headers: req.headers, + }); +}; + +const handler = (req: NextRequest) => + fetchRequestHandler({ + endpoint: "/api/trpc", + req, + router: appRouter, + createContext: () => createContext(req), + onError: + env.NODE_ENV === "development" + ? ({ path, error }) => { + console.error( + `❌ tRPC failed on ${path ?? ""}: ${error.message}` + ); + } + : undefined, + }); + +export { handler as GET, handler as POST }; diff --git a/src/app/layout.tsx b/src/app/layout.tsx new file mode 100644 index 0000000..24f2f27 --- /dev/null +++ b/src/app/layout.tsx @@ -0,0 +1,24 @@ +import "@/styles/globals.css"; + +import { GeistSans } from "geist/font/sans"; +import { type Metadata } from "next"; + +import { TRPCReactProvider } from "@/trpc/react"; + +export const metadata: Metadata = { + title: "Create T3 App", + description: "Generated by create-t3-app", + icons: [{ rel: "icon", url: "/favicon.ico" }], +}; + +export default function RootLayout({ + children, +}: Readonly<{ children: React.ReactNode }>) { + return ( + + + {children} + + + ); +} diff --git a/src/app/page.tsx b/src/app/page.tsx new file mode 100644 index 0000000..d473a0d --- /dev/null +++ b/src/app/page.tsx @@ -0,0 +1,53 @@ +import Link from "next/link"; + +import { LatestPost } from "@/app/_components/post"; +import { api, HydrateClient } from "@/trpc/server"; + +export default async function Home() { + const hello = await api.post.hello({ text: "from tRPC" }); + + void api.post.getLatest.prefetch(); + + return ( + +
+
+

+ Create T3 App +

+
+ +

First Steps →

+
+ Just the basics - Everything you need to know to set up your + database and authentication. +
+ + +

Documentation →

+
+ Learn more about Create T3 App, the libraries it uses, and how + to deploy it. +
+ +
+
+

+ {hello ? hello.greeting : "Loading tRPC query..."} +

+
+ + +
+
+
+ ); +} diff --git a/src/env.js b/src/env.js new file mode 100644 index 0000000..6ca7f3e --- /dev/null +++ b/src/env.js @@ -0,0 +1,44 @@ +import { createEnv } from "@t3-oss/env-nextjs"; +import { z } from "zod"; + +export const env = createEnv({ + /** + * Specify your server-side environment variables schema here. This way you can ensure the app + * isn't built with invalid env vars. + */ + server: { + DATABASE_URL: z.string().url(), + NODE_ENV: z + .enum(["development", "test", "production"]) + .default("development"), + }, + + /** + * Specify your client-side environment variables schema here. This way you can ensure the app + * isn't built with invalid env vars. To expose them to the client, prefix them with + * `NEXT_PUBLIC_`. + */ + client: { + // NEXT_PUBLIC_CLIENTVAR: z.string(), + }, + + /** + * You can't destruct `process.env` as a regular object in the Next.js edge runtimes (e.g. + * middlewares) or client-side so we need to destruct manually. + */ + runtimeEnv: { + DATABASE_URL: process.env.DATABASE_URL, + NODE_ENV: process.env.NODE_ENV, + // NEXT_PUBLIC_CLIENTVAR: process.env.NEXT_PUBLIC_CLIENTVAR, + }, + /** + * Run `build` or `dev` with `SKIP_ENV_VALIDATION` to skip env validation. This is especially + * useful for Docker builds. + */ + skipValidation: !!process.env.SKIP_ENV_VALIDATION, + /** + * Makes it so that empty strings are treated as undefined. `SOME_VAR: z.string()` and + * `SOME_VAR=''` will throw an error. + */ + emptyStringAsUndefined: true, +}); diff --git a/src/server/api/root.ts b/src/server/api/root.ts new file mode 100644 index 0000000..59dac6a --- /dev/null +++ b/src/server/api/root.ts @@ -0,0 +1,23 @@ +import { postRouter } from "@/server/api/routers/post"; +import { createCallerFactory, createTRPCRouter } from "@/server/api/trpc"; + +/** + * This is the primary router for your server. + * + * All routers added in /api/routers should be manually added here. + */ +export const appRouter = createTRPCRouter({ + post: postRouter, +}); + +// export type definition of API +export type AppRouter = typeof appRouter; + +/** + * Create a server-side caller for the tRPC API. + * @example + * const trpc = createCaller(createContext); + * const res = await trpc.post.all(); + * ^? Post[] + */ +export const createCaller = createCallerFactory(appRouter); diff --git a/src/server/api/routers/post.ts b/src/server/api/routers/post.ts new file mode 100644 index 0000000..f1fd661 --- /dev/null +++ b/src/server/api/routers/post.ts @@ -0,0 +1,31 @@ +import { z } from "zod"; + +import { createTRPCRouter, publicProcedure } from "@/server/api/trpc"; + +export const postRouter = createTRPCRouter({ + hello: publicProcedure + .input(z.object({ text: z.string() })) + .query(({ input }) => { + return { + greeting: `Hello ${input.text}`, + }; + }), + + create: publicProcedure + .input(z.object({ name: z.string().min(1) })) + .mutation(async ({ ctx, input }) => { + return ctx.db.post.create({ + data: { + name: input.name, + }, + }); + }), + + getLatest: publicProcedure.query(async ({ ctx }) => { + const post = await ctx.db.post.findFirst({ + orderBy: { createdAt: "desc" }, + }); + + return post ?? null; + }), +}); diff --git a/src/server/api/trpc.ts b/src/server/api/trpc.ts new file mode 100644 index 0000000..fb3fb22 --- /dev/null +++ b/src/server/api/trpc.ts @@ -0,0 +1,106 @@ +/** + * YOU PROBABLY DON'T NEED TO EDIT THIS FILE, UNLESS: + * 1. You want to modify request context (see Part 1). + * 2. You want to create a new middleware or type of procedure (see Part 3). + * + * TL;DR - This is where all the tRPC server stuff is created and plugged in. The pieces you will + * need to use are documented accordingly near the end. + */ +import { initTRPC } from "@trpc/server"; +import superjson from "superjson"; +import { ZodError } from "zod"; + +import { db } from "@/server/db"; + +/** + * 1. CONTEXT + * + * This section defines the "contexts" that are available in the backend API. + * + * These allow you to access things when processing a request, like the database, the session, etc. + * + * This helper generates the "internals" for a tRPC context. The API handler and RSC clients each + * wrap this and provides the required context. + * + * @see https://trpc.io/docs/server/context + */ +export const createTRPCContext = async (opts: { headers: Headers }) => { + return { + db, + ...opts, + }; +}; + +/** + * 2. INITIALIZATION + * + * This is where the tRPC API is initialized, connecting the context and transformer. We also parse + * ZodErrors so that you get typesafety on the frontend if your procedure fails due to validation + * errors on the backend. + */ +const t = initTRPC.context().create({ + transformer: superjson, + errorFormatter({ shape, error }) { + return { + ...shape, + data: { + ...shape.data, + zodError: + error.cause instanceof ZodError ? error.cause.flatten() : null, + }, + }; + }, +}); + +/** + * Create a server-side caller. + * + * @see https://trpc.io/docs/server/server-side-calls + */ +export const createCallerFactory = t.createCallerFactory; + +/** + * 3. ROUTER & PROCEDURE (THE IMPORTANT BIT) + * + * These are the pieces you use to build your tRPC API. You should import these a lot in the + * "/src/server/api/routers" directory. + */ + +/** + * This is how you create new routers and sub-routers in your tRPC API. + * + * @see https://trpc.io/docs/router + */ +export const createTRPCRouter = t.router; + +/** + * Middleware for timing procedure execution and adding an artificial delay in development. + * + * You can remove this if you don't like it, but it can help catch unwanted waterfalls by simulating + * network latency that would occur in production but not in local development. + */ +const timingMiddleware = t.middleware(async ({ next, path }) => { + const start = Date.now(); + + if (t._config.isDev) { + // artificial delay in dev + const waitMs = Math.floor(Math.random() * 400) + 100; + await new Promise((resolve) => setTimeout(resolve, waitMs)); + } + + const result = await next(); + + const end = Date.now(); + console.log(`[TRPC] ${path} took ${end - start}ms to execute`); + + return result; +}); + +/** + * Public (unauthenticated) procedure + * + * This is the base piece you use to build new queries and mutations on your tRPC API. It does not + * guarantee that a user querying is authorized, but you can still access user session data if they + * are logged in. + */ +export const publicProcedure = t.procedure.use(timingMiddleware); diff --git a/src/server/db.ts b/src/server/db.ts new file mode 100644 index 0000000..9255447 --- /dev/null +++ b/src/server/db.ts @@ -0,0 +1,17 @@ +import { PrismaClient } from "@prisma/client"; + +import { env } from "@/env"; + +const createPrismaClient = () => + new PrismaClient({ + log: + env.NODE_ENV === "development" ? ["query", "error", "warn"] : ["error"], + }); + +const globalForPrisma = globalThis as unknown as { + prisma: ReturnType | undefined; +}; + +export const db = globalForPrisma.prisma ?? createPrismaClient(); + +if (env.NODE_ENV !== "production") globalForPrisma.prisma = db; diff --git a/src/styles/globals.css b/src/styles/globals.css new file mode 100644 index 0000000..b5c61c9 --- /dev/null +++ b/src/styles/globals.css @@ -0,0 +1,3 @@ +@tailwind base; +@tailwind components; +@tailwind utilities; diff --git a/src/trpc/query-client.ts b/src/trpc/query-client.ts new file mode 100644 index 0000000..bda6439 --- /dev/null +++ b/src/trpc/query-client.ts @@ -0,0 +1,25 @@ +import { + defaultShouldDehydrateQuery, + QueryClient, +} from "@tanstack/react-query"; +import SuperJSON from "superjson"; + +export const createQueryClient = () => + new QueryClient({ + defaultOptions: { + queries: { + // With SSR, we usually want to set some default staleTime + // above 0 to avoid refetching immediately on the client + staleTime: 30 * 1000, + }, + dehydrate: { + serializeData: SuperJSON.serialize, + shouldDehydrateQuery: (query) => + defaultShouldDehydrateQuery(query) || + query.state.status === "pending", + }, + hydrate: { + deserializeData: SuperJSON.deserialize, + }, + }, + }); diff --git a/src/trpc/react.tsx b/src/trpc/react.tsx new file mode 100644 index 0000000..c3772e5 --- /dev/null +++ b/src/trpc/react.tsx @@ -0,0 +1,76 @@ +"use client"; + +import { QueryClientProvider, type QueryClient } from "@tanstack/react-query"; +import { loggerLink, unstable_httpBatchStreamLink } from "@trpc/client"; +import { createTRPCReact } from "@trpc/react-query"; +import { type inferRouterInputs, type inferRouterOutputs } from "@trpc/server"; +import { useState } from "react"; +import SuperJSON from "superjson"; + +import { type AppRouter } from "@/server/api/root"; +import { createQueryClient } from "./query-client"; + +let clientQueryClientSingleton: QueryClient | undefined = undefined; +const getQueryClient = () => { + if (typeof window === "undefined") { + // Server: always make a new query client + return createQueryClient(); + } + // Browser: use singleton pattern to keep the same query client + return (clientQueryClientSingleton ??= createQueryClient()); +}; + +export const api = createTRPCReact(); + +/** + * Inference helper for inputs. + * + * @example type HelloInput = RouterInputs['example']['hello'] + */ +export type RouterInputs = inferRouterInputs; + +/** + * Inference helper for outputs. + * + * @example type HelloOutput = RouterOutputs['example']['hello'] + */ +export type RouterOutputs = inferRouterOutputs; + +export function TRPCReactProvider(props: { children: React.ReactNode }) { + const queryClient = getQueryClient(); + + const [trpcClient] = useState(() => + api.createClient({ + links: [ + loggerLink({ + enabled: (op) => + process.env.NODE_ENV === "development" || + (op.direction === "down" && op.result instanceof Error), + }), + unstable_httpBatchStreamLink({ + transformer: SuperJSON, + url: getBaseUrl() + "/api/trpc", + headers: () => { + const headers = new Headers(); + headers.set("x-trpc-source", "nextjs-react"); + return headers; + }, + }), + ], + }) + ); + + return ( + + + {props.children} + + + ); +} + +function getBaseUrl() { + if (typeof window !== "undefined") return window.location.origin; + if (process.env.VERCEL_URL) return `https://${process.env.VERCEL_URL}`; + return `http://localhost:${process.env.PORT ?? 3000}`; +} diff --git a/src/trpc/server.ts b/src/trpc/server.ts new file mode 100644 index 0000000..4edb0b1 --- /dev/null +++ b/src/trpc/server.ts @@ -0,0 +1,30 @@ +import "server-only"; + +import { createHydrationHelpers } from "@trpc/react-query/rsc"; +import { headers } from "next/headers"; +import { cache } from "react"; + +import { createCaller, type AppRouter } from "@/server/api/root"; +import { createTRPCContext } from "@/server/api/trpc"; +import { createQueryClient } from "./query-client"; + +/** + * This wraps the `createTRPCContext` helper and provides the required context for the tRPC API when + * handling a tRPC call from a React Server Component. + */ +const createContext = cache(async () => { + const heads = new Headers(await headers()); + heads.set("x-trpc-source", "rsc"); + + return createTRPCContext({ + headers: heads, + }); +}); + +const getQueryClient = cache(createQueryClient); +const caller = createCaller(createContext); + +export const { trpc: api, HydrateClient } = createHydrationHelpers( + caller, + getQueryClient +); diff --git a/start-database.sh b/start-database.sh new file mode 100644 index 0000000..6682ea6 --- /dev/null +++ b/start-database.sh @@ -0,0 +1,60 @@ +#!/usr/bin/env bash +# Use this script to start a docker container for a local development database + +# TO RUN ON WINDOWS: +# 1. Install WSL (Windows Subsystem for Linux) - https://learn.microsoft.com/en-us/windows/wsl/install +# 2. Install Docker Desktop for Windows - https://docs.docker.com/docker-for-windows/install/ +# 3. Open WSL - `wsl` +# 4. Run this script - `./start-database.sh` + +# On Linux and macOS you can run this script directly - `./start-database.sh` + +DB_CONTAINER_NAME="gitsage-postgres" + +if ! [ -x "$(command -v docker)" ]; then + echo -e "Docker is not installed. Please install docker and try again.\nDocker install guide: https://docs.docker.com/engine/install/" + exit 1 +fi + +if ! docker info > /dev/null 2>&1; then + echo "Docker daemon is not running. Please start Docker and try again." + exit 1 +fi + +if [ "$(docker ps -q -f name=$DB_CONTAINER_NAME)" ]; then + echo "Database container '$DB_CONTAINER_NAME' already running" + exit 0 +fi + +if [ "$(docker ps -q -a -f name=$DB_CONTAINER_NAME)" ]; then + docker start "$DB_CONTAINER_NAME" + echo "Existing database container '$DB_CONTAINER_NAME' started" + exit 0 +fi + +# import env variables from .env +set -a +source .env + +DB_PASSWORD=$(echo "$DATABASE_URL" | awk -F':' '{print $3}' | awk -F'@' '{print $1}') +DB_PORT=$(echo "$DATABASE_URL" | awk -F':' '{print $4}' | awk -F'\/' '{print $1}') + +if [ "$DB_PASSWORD" = "password" ]; then + echo "You are using the default database password" + read -p "Should we generate a random password for you? [y/N]: " -r REPLY + if ! [[ $REPLY =~ ^[Yy]$ ]]; then + echo "Please change the default password in the .env file and try again" + exit 1 + fi + # Generate a random URL-safe password + DB_PASSWORD=$(openssl rand -base64 12 | tr '+/' '-_') + sed -i -e "s#:password@#:$DB_PASSWORD@#" .env +fi + +docker run -d \ + --name $DB_CONTAINER_NAME \ + -e POSTGRES_USER="postgres" \ + -e POSTGRES_PASSWORD="$DB_PASSWORD" \ + -e POSTGRES_DB=gitsage \ + -p "$DB_PORT":5432 \ + docker.io/postgres && echo "Database container '$DB_CONTAINER_NAME' was successfully created" diff --git a/tailwind.config.ts b/tailwind.config.ts new file mode 100644 index 0000000..5fd44e8 --- /dev/null +++ b/tailwind.config.ts @@ -0,0 +1,14 @@ +import { type Config } from "tailwindcss"; +import { fontFamily } from "tailwindcss/defaultTheme"; + +export default { + content: ["./src/**/*.tsx"], + theme: { + extend: { + fontFamily: { + sans: ["var(--font-geist-sans)", ...fontFamily.sans], + }, + }, + }, + plugins: [], +} satisfies Config; diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..c5eef6e --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,42 @@ +{ + "compilerOptions": { + /* Base Options: */ + "esModuleInterop": true, + "skipLibCheck": true, + "target": "es2022", + "allowJs": true, + "resolveJsonModule": true, + "moduleDetection": "force", + "isolatedModules": true, + + /* Strictness */ + "strict": true, + "noUncheckedIndexedAccess": true, + "checkJs": true, + + /* Bundled projects */ + "lib": ["dom", "dom.iterable", "ES2022"], + "noEmit": true, + "module": "ESNext", + "moduleResolution": "Bundler", + "jsx": "preserve", + "plugins": [{ "name": "next" }], + "incremental": true, + + /* Path Aliases */ + "baseUrl": ".", + "paths": { + "@/*": ["./src/*"] + } + }, + "include": [ + ".eslintrc.cjs", + "next-env.d.ts", + "**/*.ts", + "**/*.tsx", + "**/*.cjs", + "**/*.js", + ".next/types/**/*.ts" + ], + "exclude": ["node_modules"] +}