Skip to content

Blankeos/solid-launch

Repository files navigation

πŸ’™ Solid Launch

An sophisticated boiler-plate built for simplicity.

Image

Carlo's starter for making a Vike + Solid app with batteries included on stuff I like after experimenting for years.

This is handcrafted from my own research. This might not work for you, but it works for me. πŸ€“

You can also try my other starters:

Benefits

  • 🐭 Handcrafted and minimal - picked and chose "do one thing, do it well" libraries that are just enough to get the job done. Just looks a bit bloated at a glance. (I kinda made my own NextJS from scatch here)
  • ⚑️ Super-fast dev server - way faster than NextJS thanks to Vite. You need to feel it to believe it! It can also literally build your app in seconds.
  • πŸ’¨ Fast, efficient, fine-grained Reactivity - thanks to Solid, it's possibly the most enjoyable framework I used that uses JSX. Has state management primitives out-of-the-box and keeps the experience a breeze.
  • 🐍 Extremely customizable - you're not at the mercy of limited APIs and paradigms set by big frameworks or third-party services. Swap with your preferred JS backend framework/runtime if you want. Vike is just a middleware. Most of the tech I use here are open-source and roll-your-own type of thing. Hack it up! You're a dev aren't you?
  • ☁️ Selfhost-ready - Crafted with simple hosting in mind that'll still probably scale to millions. Just spin up Docker container on a good'ol VPS without locking into serverless. DHH and Shayan influenced me on this. You can still host it on serverless tho. I think? lol
  • πŸ”‹ Batteries-included - took care of the hard stuff for you. A well-thought-out folder structure from years of making projects: a design system, components, utilities, hooks, constants, an adequate backend DDD-inspired sliced architecture that isn't overkill, dockerizing your app, and most importantly---perfectly-crafted those pesky config files.
  • πŸ”‘ Authentication-Ready - One thing devs get stuck on. There's a practical auth implemented from scratch here that doesn't vendor-lock you into any auth provider.
    • Password
    • Transactional Emails (Forgot Password, Email Verification)
    • OAuth
    • Magic Link
    • User Management Dashboard

Tech Stack

  • Bun - Runtime and package manager. You can always switch to Node and PNPM if you wish.
  • SolidJS - Frontend framework that I like. Pretty underrated, but awesome!
  • Vike - Like NextJS, but just a middleware. SSR library on-top of Vite. Use on any JS backend. Flexible, Simple, and Fast!
  • Hono - 2nd fastest Bun framework(?), run anywhere, uses easy-to-understand web-standard paradigms.
  • tRPC - E2E typesafety without context switching. Just amazing DevX.
  • Tailwind - Styling the web has been pretty pleasant with it. I even use it on React Native for work. It's amazing.
  • Prisma - Great migrations workflow, but I want to maximize perf.
  • Kysely - Great typesafe query builder for SQL, minimally wraps around db connection.
  • SQLite/LibSQL (Turso) - Cheapest database, easy to use.
  • Lucia - Makes self-rolling auth easy.
  • SES or MimePost - Emails
  • Backblaze - Cheap blob object storage with an S3-compatible API.
  • Paddle - Accept payments and pay foreign taxes.

QuickStart

I'll assume you don't want to change anything with this setup after cloning so let's get to work!

  1. Copy the environment variables

    cp .env.example .env
  2. Replace the <absolute_url> in the local database with:

    pwd # If it outputs: /User/Projects/solid-launch
    
    # Replace the .env with:
    DATABASE_URL="file:/User/Projects/solid-launch/local.db"
  3. Generate

    bun db:generate # generates Kysely and Prisma client types.
    bun db:migrate # migrates your database.
  4. Install deps and run dev

    bun install
    bun dev

Useful Development Tips

I took care of the painstaking parts to help you develop easily on a SPA + SSR + backend paradigm. You can take take these practices to different projects as well.

  1. Make use of the code-snippets I added. It'll help!

  2. Check all typescript errors (Cmd + Shift + B > tsc:watch tsconfig.json).

  3. Authentication Practices - I have these out-of-the-box for you so you won't have to build it.

    • Getting Current User

      import { useAuthContext } from '@/context/auth.context';
      
      export default function MyComponent() {
       const { user } = useAuthContext();
      }
    • Login, Logout, Register

      import { useAuthContext } from '@/context/auth.context';
      
      export default function MyComponent() {
      const { login, logout, register } = useAuthContext();
      }
    • Hydrating Current User

      This will also automatically hydrate in your layouts. Anywhere you use useAuthStore(), it's magic. (Thanks to Vike's useData(). Fun fact: You actually can't do this in SolidStart because it's architecturally different to Vike).

      // +data.ts
      import { initTRPCSSRClient } from '@/lib/trpc-ssr-client';
      import { PageContext } from 'vike/types';
      
      export type Data = ReturnType<Awaited<typeof data>>;
      
      export async function data(pageContext: PageContext) {
         const { request, response } = pageContext;
      
         const trpcClient = initTRPCSSRClient(request.header(), response.headers); // Pass the headers here.
      
         const result = await trpcClient.auth.currentUser.query();
      
         return {
            user: result.user ?? null,
         };
      }
    • Protecting Routes (Client-Side)

      import ProtectedRoute from '@/components/common/protected-route';
      
      export default MyComponent() {
         return (
            <ProtectedRoute>
              On the server (hydration), this part will not be rendered if unauthenticated.
      
              On the client, you will be redirected to a public route if unauthenticated.
            </ProtectedRoute>
         )
      }
    • Protecting Routes (SSR)

      // +guard.ts (If you don't have +data.ts in the same route).
      export async function guard(pageContext: PageContext) {
         const { request, response } = pageContext;
      
         const trpcClient = initTRPCSSRClient(request.header(), response.headers); // Pass the headers here.
      
         const result = await trpcClient.auth.currentUser.query();
      
         if (!result.user) {
            throw redirect("/") // Must be a public route.
         }
      }
      
      // +guard.ts (If you already have a +data.ts that gets the user).
      // ⚠️ I have not tested this. This depends on `+guard` being called after `+data` is resolved.
      export async function guard(pageContext: PageContext) {
      
         if (!pageContext.data?.user) {
            throw redirect("/"); // Must be a public route.
         }
      }
  4. Dataloading Practices - Also have these out-of-the-box for most usecases since they're tricky to do if you're clueless:

    • Tanstack Query (Client-only) - Use trpc-client.ts
    • Hydrated Tanstack Query (SSR) - Use create-dehydrated-state.ts + trpc-ssr-client.ts

Backend Architecture

My backend architecture is inspired by DDD where I separate in layers, but I keep it pragmatic by not going too overkill with Entities, Domains, and Aggregates. I personally still like the anemic data-driven architecture for most of my apps since the apps I make aren't too business-logic-heavy.

.
└── server/ # - root
    β”œβ”€β”€ dao/ # - data-access-objects
    β”‚   └── *.dao.ts
    β”œβ”€β”€ modules/
    β”‚   └── <module>/
    β”‚       β”œβ”€β”€ services/
    β”‚       β”‚   └── *.service.ts # 1 service usecase
    β”‚       └── <module>.controller.ts
    └── _app.ts # - root TRPC router.
  • dao - abstracted away all queries here to interface with them as plain functions. It actually helps me mentally collocate db queries from service logic when I'm using them inside the service.
  • modules - a vertical slice of each module-group. This just depends on how I feel about grouping them. You get better overtime.
  • <module>.controller.ts - pretty much a group of http endpoints. I can put the DTOs/Validations for each endpoint here without context-switching.
  • services - these are even smaller pieces of logic that can be used inside each endpoint. It's not necessary to use if the app isn't too big, but it helps.
  • _app.ts - The root trpc router where the AppRouter type is exported.

Deployment

Warning

Still in progress

Here are some guides on how to deploy.

  • Kamal (self-host VPS - I recommend)
  • Dokku (self-host VPS)
  • Caprover (self-host VPS)
  • Cloudflare (serverless + static)
  • Vercel (serverless + static)
  • Netlify (static)

Limitations

  • Websockets and Bun
    • It works fine in Prod. But activating Vite HMR + Websockets is not possible in Bun.
    • This is because Bun doesn't work with Connect.Server middlewares (which Vite uses). [Source 1] [Source 2]
    • Bun Workaround: Having a separate process to run the Websocket server and your HTTP Server. Just make sure to use the same pubsub across these two processes (You can do this using Redis). Also make sure to combine them in a single process in production.
    • Alternative recommendation: Use Node instead as it's possible to use Connect.Server middlewares in their http server: PoC.

Future Plans

I'll probably make a swapping guide soon. To replace to these:

  • Runtime: Bun -> Node
  • Package Manager: Bun -> PNPM
  • ORM: Prisma -> Drizzle
  • Database: SQLite -> PostgreSQL, CockroachDB, MongoDB