Skip to content

Latest commit

 

History

History
287 lines (195 loc) · 12.4 KB

CONTRIBUTING.md

File metadata and controls

287 lines (195 loc) · 12.4 KB

Contributing

Getting Started

Development setup

Make sure you have Bun installed on your machine. If you don't have it, you can download it here. Here is some information about Bun in Next.js: Build an app with Next.js and Bun.

Dependencies

First, install dependencies:

bun install

Note

When you merge with the dev branch you may need to rerun bun install in case there are new dependencies added by other contributors.

Environment variables

Then you need to setup environment variables by creating a .env file in the root of the project. Fill the .env file with all variables from the .env.example file. You can supply an empty string as the value for variables that you do not need to set for the things you are using. For example, if you are not working on Feide, you do not need to set the FEIDE_CLIENT_ID, FEIDE_CLIENT_SECRET, etc...

When finished setting the file up it should be on this form:

NODE_ENV="development"
NODE_OPTIONS="--max_old_space_size=16384"
NEXT_TELEMETRY_DISABLED="true"
NEXT_PUBLIC_SITE_URL="http://localhost:3000"
DB_HOST="localhost"
DB_PORT="5432"
DB_USER="user"
# and so on...

If you have not defined an environment variable you will get an error on this form when you start the dev server:

❌ Invalid environment variables: { DB_PORT: [ 'Required' ] }

Here I had forgotten to set the DB_PORT variable.

Note

When you merge with the dev branch you may need to update the .env file in case there are new environment variables added or modified by other contributors.

Docker

Next we are gonna setup the database and storage server to run locally. To achieve this we need to install Docker.

  • To install Docker on Linux you can follow the guide here based on your distribution.

  • For windows you can install Docker Desktop from here. I do not recommend using Docker Desktop as it is very heavy and slow to run + it has a non-commercial license. If you are using windows and know of a better solution please update this guide.

  • For MacOS you can use Colima as a Docker runtime. Just install it using homebrew. You can also use Docker Desktop here if you prefer, but that is not recommended.

    # MacOS only
    brew install colima docker
    # Start colima
    colima start

Then we need to install Docker Compose.

  • For Linux follow the plugin guide for your distro here.

  • For Windows I have no idea, please update this guide.

  • For MacOS you can install Docker Compose using homebrew.

    # MacOS only
    brew install docker-compose

Database and storage server

When you have installed Docker and Docker Compose you can start the database and storage server by running:

bun run dev:setup

Now this command is a fancy all in one command that restarts and resets both the database and storage server pushes any new migrations and seeds the database. If you prefer to use the individual commands they will be outlined below.

You can stop both with:

bun run dev:stop

Tip

In the future we may add a way to SSH tunnel and use the database and storage server for the staging solution locally. This is only beneficial if you are only working on the frontend and do not need to modify the database.

Now for a more in depth guide on all the scripts for the database and storage server. First the basics that they both have in common. Now remember that you in theory only need the fancy command above if you are not doing any debugging. It just runs these other scripts for you under the hood.

To start the database and storage server you can run:

bun run db:start
# or
bun run s3:start

To stop the database and storage server you can run:

bun run db:stop
# or
bun run s3:stop

To delete all the data for the database and storage server you can run:

bun run db:delete
# or
bun run s3:delete

To print the logs continuely from the database and storage server you can run:

bun run db:logs
# or
bun run s3:logs

For migrations we have multiple commands to help us. First to quickly create migrations based on a change you have made to the tables (and not store it) and apply it to the running database you can run:

bun run db:push

This is the command dev:setup uses under the hood to push the migrations to the database.

If you have finished making changes to the database and want to create and store the migration you can run:

bun run db:generate

This will generate new migrations into the @/server/db/migrations directory.

If you want to run the generated migrations on your database you can run:

bun run db:migrate

We also have a command that seeds the database (fills it up with mock data). This command runs the @/server/db/seed.ts file, which needs to be updated when we need data to be available in the database. You can run it with:

bun run db:seed

Lastly we have a very useful command that starts an interactive web server where you can look at the database and run queries on it. This is very useful for debugging and testing. You can use it with:

bun run db:studio

This lets you access the database locally on: https://local.drizzle.studio

If you want to access the storage server locally in a web browser you can always access it on: http://localhost:9001 when it is running. The password and username is what you have defined in the .env file.

Development server

Then you can run the development server and see the website:

bun dev --turbo

Open http://localhost:3000 with your browser to see the result.

Build

When you build the project, you pre-render all the Server Side Generated (SSG) pages. This makes the site load faster and perform better and behave like it will when it is deployed. When serving the built project it will not hot reload when you make changes to the code like it does in development mode.

You can build the project with the following command:

bun run build

To serve the build locally, run:

bun run start

Check linting and formatting

To check linting and formatting you run the respective command:

bun lint

Commit messages

We are using Conventional Commits for our commit messages. This is to ensure that we have a consistent way of writing commit messages to make it easier to understand what has been changed and why. Try to follow the guidelines as closely as possible. You can also use the recommended vscode extension to help you write the commit messages.

Code quality

  • To keep the code as consistent as possible use functions for react components or hooks instead of const variables with arrow function syntax. An exception is when using the forwardRef hook or when creating compound components.
  • Only use default export for pages or layouts etc. since it is required by Next.js. For everything else use named exports. This is to make it easier to find the components in the codebase or change them without ending up with different names for the same component.
  • Use type instead of interface for typescript types. This is to keep the code consistent and to make it easier to read. Also type is more flexible than interface since it can be used for unions and intersections.
  • When using custom icons that are not provided by lucide, make sure to add them as React Components (SVGs) to the components/assets/icons directory. This improves performance since the icons are handled as vectors and not as images.
  • For internationalization use the useTranslations and getTranslations (for async components) hooks from next-intl.
    • For client components pass the translations as props using the t prop for consistency.
    • On the backend when returning error messages, you can just return the key to the translation as the error message and it will be formatted correctly.
    • In the backend context (ctx) the locale is stored in ctx.locale and translations can be accessed using ctx.getTranslations().

Naming conventions

  • All layout components should end with Layout. For example: DefaultLayout.
  • All page components should end with Page to make it clear it is a whole page. For example: AboutPage.

Useful resources

Here is a list of documentations that will help you contribute to the project:

Front-end

  • React - Library for building user interfaces
  • Next.js - Framework for routing and server-side rendering
  • Next-intl - Internationalization library
  • nuqs - Easy to use query params
  • Plate - Tool for rich text editing
  • Tanstack Query - TRPC wraps Tanstack Query which is how we fetch data from the backend
  • Tanstack Table - For dynamic tables with filtering, sorting, pagination etc
  • Tanstack Form - When we need to handle form validation

Styling

Back-end

  • TRPC - Tool for creating API endpoints as functions
  • Lucia - Authentication library
  • Drizzle - ORM for interacting with the database (Postgres under the hood)
  • s3-client - AWS S3 client for uploading files

Infrastructure

  • Docker - Containerization tool for the application, database and storage
  • Docker Compose - Tool for running multi-container applications
  • Colima - Container runtime for docker, I recommend this over Docker Desktop because of performance and license
  • nginx - Reverse proxy for routing requests to the correct service

Other

  • Mozilla - Great resource for looking up documentation for web technologies
  • Can I use - Check browser support for different web technologies (especially useful for CSS)

Development Environment

VS Code

Recommended Extensions

Issues

  • If you are experiencing issues with types, you can restart the typescript server by pressing cmd + shift + p and then type TypeScript: Restart TS Server (You need to have a typescript file open for this to work).
  • You can also try restarting the whole editor by pressing cmd + shift + p and then type Developer: Reload Window.

On windows you can use ctrl instead of cmd.

GitHub

  • For branch names, use kebab-case (e.g. about-us-page)