Skip to content

Latest commit

 

History

History
342 lines (251 loc) · 23.4 KB

README.md

File metadata and controls

342 lines (251 loc) · 23.4 KB

GO logo


Website for the public library that uses Next.js with the App Router for the frontend and Drupal for content management.
Drupal CMS can be accessed through Lagoon by generating a one-time login in the preferred environment.

Last updated: 27/11/2024

URLs

Description URL
Demo site (may change) https://node.pr-1707.dpl-cms.dplplat01.dpl.reload.dk/
Demo site Drupal CMS (may change, login through lagoon) https://varnish.pr-1707.dpl-cms.dplplat01.dpl.reload.dk/

Table of contents

Getting started

Prerequisites

  • Use the node version registered in the [.nvm] file
  • Preferably managed by nvm
  • This project uses yarn make sure to have this installed globally on your machine

Setup

  1. Make sure you are using the correct Node version:
nvm use
  1. Install dependencies:
yarn
  1. Start the development server:

We run the server in an experimental https state to not get blocked by CORS policy when developing locally.

yarn dev:https

The application is now running at https://localhost:3000

  1. Set up the Drupal CMS (dpl-cms) locally to access configuration variables for the Go app. Ensure the NEXT_PUBLIC_GRAPHQL_SCHEMA_ENDPOINT_DPL_CMS variable in the .env.local file points to the correct endpoint.

Technical Overview

Project structure

In the project, you'll see the following folders and files:

File(s) Description
__tests__/* Unit tests for the application components and utilities. Using vitest
.github/* GitHub configuration files and workflows
.storybook/* Configuration files and stories for Storybook
.vscode.example/* Example settings and recommendations for Visual Studio Code workspace
app/* Routes for the App Router
components/global/* Components that should always be rendered on the page
components/pages/* For page layout components used as children on routes (Useful when making API calls server-side on the route and render page layouts as child components)
components/paragraphs/* Components named according to Drupal CMS conventions for editorial sections on a page
components/shadcn/* Imported shadcn/ui components
components/shared/* Reusable components that can be optionally used across various other components
hooks/* Custom React hooks
lagoon/* TODO: add description
lib/* Library utilities and configurations
lib/config/* Centralized access to environment variables and CMS configuration settings
lib/graphql/* GraphQL-related utilities and configurations
lib/graphql/fetchers/* Custom fetch functions used for fetching data through React Query
lib/graphql/fragments/* GraphQL fragments used to define reusable pieces of data queries
lib/graphql/generated/* Auto-generated GraphQL types and queries based on the GraphQL schema
lib/graphql/queries/* Custom GraphQL queries used throughout the application
lib/helpers/* Utility functions and helpers used across the application
lib/machines/* State machines and related logic for managing complex state transitions through xstate
lib/providers/* Context providers and related logic for managing global state and dependencies
lib/rest/* REST API-related utilities and configurations
lib/rest/cover-service-api/* Custom functions and configurations for interacting with the Cover Service API
lib/rest/publizon-api/* Custom functions and configurations for interacting with the Publizon API
lib/session/* Session management utilities and configurations
lib/shadcn/* Utilities for shadcn/ui components
lib/types/* Manually added types used throughout the application
public/* Non-code, unprocessed assets (fonts, icons, etc.)
styles/* Global Tailwind CSS files
.editorconfig/ Configuration file for maintaining consistent coding styles between different editors and IDEs
.env.example Example environment variables file
.env.local Local environment variables file, specific to your development environment (git ignored)
.env.test Environment variables file for testing
.eslintignore File specifying which files and directories to ignore by ESLint
.eslintrc.json Configuration file for ESLint rules and settings
.gitignore Specifies intentionally untracked files to ignore
.nvmrc Node version manager configuration file
.prettierignore File specifying which files and directories to ignore by Prettier
codegen.ts Configuration file for generating code based on GraphQL schema
components.json Configuration file for defining and managing component metadata (necessary when installing shadcn components)
next.config.mjs Next.js configuration file
orval.config.ts Configuration file for Orval, a tool for generating API clients from OpenAPI specifications
package.json Contains metadata about the project and its dependencies, scripts, and other configurations
postcss.config.mjs Configuration file for PostCSS, a tool for transforming CSS with JavaScript plugins (necessary for Tailwind to compile)
tailwind.config.ts Tailwind CSS configuration file
Taskfile.yml Task automation file used to define and run tasks
tsconfig.json TypeScript configuration file
vitest.config.ts Configuration file for Vitest
yarn.lock Lockfile for Yarn, ensuring consistent installs across different environments

git workflows

File(s) Description
accessibility-test.yml Runs accessibility tests using Axe and Lighthouse to ensure the application meets accessibility standards.
chromatic.yml Runs Chromatic to visualize and test UI components in Storybook, ensuring that changes do not introduce visual regressions.
eslint-check.yml Runs ESLint to check for code quality and adherence to coding standards.
prettier-check.yml Runs Prettier to ensure code formatting consistency across the project.
publish-source.yml Publishes the source code to the specified repository or platform.
type-check.yml Runs TypeScript to check for type errors and ensure type safety across the project.
unit-test.yml Runs unit tests using Vitest to ensure that individual components and utilities function correctly.

Development

UI components from shadcn/ui

We use shadcn/ui to speed up the development of UI components. Imported components are located in /components/shadcn.

If you're developing a component that requires additional logic beyond the initially installed component styles or logic, copy the imported code into a new component in the /components/shared folder. This ensures that components imported by shadcn do not replace previously installed components.

Tailwind

We use Tailwind CSS to style the project. Tailwind's predefined classes are configured in the tailwind.config.ts file, which references CSS variables defined in the globals.css file. This ensures that CSS variables are accessible through both CSS and Tailwind classes when developing components. When creating new variables, add the style as a variable in the globals.css file and include this variable in the Tailwind config. Additionally, the Tailwind CSS IntelliSense extension provides autocomplete features for a smoother development experience.

When introducing new classes, make sure to reuse existing ones and maintain consistency. Use the grid system for placing elements and apply the predefined spacing variables consistently.

Codegen

In this project, we use codegen to generate GraphQL types and queries, which helps streamline the development process and maintain a high level of code quality.

Codegen types

Codegen types are automatically generated TypeScript types based on the specific API schema. These types ensure type safety and provide autocompletion features when working with GraphQL queries and mutations.

To generate the types, run the following command:

yarn codegen:all-rest-services

The custom functions and configurations for these services are located in the lib/rest directory.

yarn codegen:covers

The lib/rest/cover-service-api directory contains functions and configurations for interacting with the Cover Service API. This API is used to fetch cover images and related metadata for various resources.

yarn codegen:graphql

This will create or update the types in the lib/graphql/generated directory.

yarn codegen:publizon

The lib/rest/publizon-api directory contains functions and configurations for interacting with the Publizon API. This API is used to manage and retrieve information about digital publications.

Custom types

Custom types are manually defined TypeScript types that are used throughout the application to ensure type safety and improve code readability.

When creating a custom type, please write a T before the name of the type to refer to the variable as a type. This helps to avoid confusion between types and other functionalities.

To add a custom type, create a new file in the lib/types directory and define your types using TypeScript's type or interface keywords.

xState

TODO: write something if relevant

Config handling

TODO: MIKKEL: write some nice words

Storybook

Storybook is an essential tool in our development workflow for several reasons:

  1. Component Isolation: It allows us to develop and test UI components in isolation, ensuring that each component works as expected without dependencies on the rest of the application.
  2. Visual Documentation: Storybook provides a visual representation of our components, making it easier for developers and designers to understand and collaborate on the UI.
  3. Automated Testing: With integrations like Chromatic, we can automate visual regression testing to catch UI changes and bugs early in the development process.
  4. Reusable Components: By documenting components in Storybook, we promote reusability and consistency across the application, reducing duplication and improving maintainability.

To start Storybook, run the following command:

yarn storybook

This will launch the Storybook server, and you can view the component library in your browser at http://localhost:6006.

We create Storybook stories strictly on a "render component" basis. This means focusing on smaller atomic components that are not specific to generated code.

Deployment

git branches and pull requests

We follow a specific Git branching model as the project is currently in development mode and not yet live. The main branches are:

  • main: Used for deploying code to the demo environment. (In the future this will be the production branch)
  • Feature branches are created from main and are named using the following convention:
DDFBRA-220-opstart-dokumentation
  • DDFBRA: Workspace in Jira
  • 220: Ticket number
  • opstart-dokumentation: Ticket title

Gotha: When creating a branch, use the Jira tickets create branch button to create a branch that refers to the ticket.

When a feature is complete, all test are passing and a review process has taken place, it is merged back into main.

Create pull request

When creating a pull request, follow these steps:

  1. Create a new pull request feature->main.
  2. Add relevant information to the PR template. At a minimum, include a link to the ticket (if available) and a description.
  3. Resolve any merge conflicts, fix errors in workflows, and check Chromatic to ensure new changes do not cause conflicts in the app. The PR creator is responsible for these checks.
  4. Await approval from a relevant team member for the changes to the code.
  5. Merge the code to main and delete the feature branch to clean up.

Reviewing a PR

  1. Review the code changes to ensure they meet the project's coding standards and requirements.
  2. Provide constructive feedback and request any necessary changes. Suggestions are optional for the creator to fix, while change requests must be addressed.
  3. Approve the PR if everything looks good, or request changes if there are issues that need to be addressed.

Updating the demo site

Create a release tag in dpl-go based on sprint number

  1. Navigate to the github project and go to the releases page
  2. Click Draft a new release
  3. Create a new release with the tag format 0.<sprint_number>.<incremental_release_count> (e.g., for sprint 7, the tag would be 0.7.0).
  4. This will trigger the publish-source workflow in the repository

Deploying a release

  1. Navigate to the dpl-cms repository.
  2. Pull the latest changes from the develop branch locally.
  3. Switch to the dpl-go-demo branch and merge the develop branch into it.
  4. Update the dpl-go package version locally to the latest release and push the changes upstream.
  5. Edit the Docker-compose.yml file with the following changes:
node:
  image: ghcr.io/danskernesdigitalebibliotek/dpl-go-node:0.<sprint_number>.<incremental_release_count>
  labels:
    lagoon.type: node
    provenance: false

Quality assurance

Quality assurance (QA) is a critical aspect of our development process, ensuring that the application meets the required standards and functions correctly. By implementing robust QA practices, we can identify and address issues early, maintain code quality, and deliver a reliable product to users.

As an overall rule of thumb, we add different levels of quality assurance depending on the given problem.

GitHub Workflows for quality assurance

  1. Automated Unit Testing: Workflows like unit-test.yml run unit tests automatically, ensuring that individual components and utilities function correctly. This helps catch bugs early in the development process.

  2. Code Quality Checks: Workflows such as eslint-check.yml and prettier-check.yml enforce coding standards and consistent formatting. This ensures that the codebase remains clean, readable, and maintainable.

  3. Type Safety: The type-check.yml workflow runs TypeScript checks to ensure type safety across the project. This helps prevent type-related errors and improves code reliability.

  4. Accessibility Testing: The accessibility-test.yml workflow runs accessibility tests on each Storybook Story using Axe through Playwright. This ensures that the application meets accessibility standards and provides a better user experience for all users.

  5. Visual Regression Testing: The chromatic.yml workflow runs Chromatic to visualize and test UI components in Storybook. This helps catch visual regressions and ensures that UI changes do not introduce unexpected issues.

Developers