Skip to content

CDLUC3/dmsp_frontend_prototype

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

dmsp_frontend_prototype app

Next.js Node.js License

Table of Contents

Introduction

The dmsp_frontend_app is a web application built using NextJS React framework and TypeScript. The app serves up all the pages for the DMP Tool, and uses Apollo GraphQL Client to make requests to the backend server.

The app is a hybrid framework using both frontend and backend logic. The app includes some API endpoints for dealing with authentication - namely for creating an http-only auth cookie, and for checking auth state, and deleting the auth cookie. It also uses middleware to control access to pages based on authentication and roles.

Graphql Queries and Mutations

This app is using graphql-codegen to generate types from graphql queries and mutations. Please place your queries in the graphql directory and name the file like graphql/<name>.<query|mutation>.graphql per the codegen.ts config file. This will generate the generated/graphql.tsx file, and you can use the generated functions or documents to make the graphql requests.

Once the schema has been added, you will need to run npm run generate this kicks off a script that builds out Typescript Types for the new schema and queries. The schema is dependent on the graphql server running at dmsp_backend_prototype in order to successfully generate.

This app was bootstrapped with create-next-app.

Features

  • App Router: Leverages the new App Router for improved routing and page structure.
  • API Routes: Custom API routes for handling authentication using JWT tokens.
  • Apollo Client: Uses Apollo Client to make GraphQL requests to the backend
  • SCSS/CSS Modules: Uses Sass along with CSS Modules for styling.
  • Unit Testing: Uses Jest with React Testing Library
  • Accessibility Testing: Uses jest-axe package
  • UI Components: React Aria Components library used with custom components
  • Authentication: Auth tokens from backend stored in cookie. Token is passed via headers in requests to backend

Getting Started

Prerequisites

  • Node.js v20.9.0 or higher
  • npm v10.7.0 or higher
  • Docker Desktop -- Docker Desktop gives you Docker Engine, Docker CLI client, and Docker Compose

Installation

  1. Clone the repository
git clone [email protected]:CDLUC3/dmsp_frontend_prototype.git
cd dmsp_frontend_prototype
npm install
  1. Install dependencies
npm install

Localization

The app is using next-intl for internationalization (i18n) with prefix-based routing (i.e., a dynamic [locale] segment will be included in the path, like /en-US/account/connectsion). We will be using the four-letter locales, meaning a combination of the two-letter language code and the two-letter country code(i.e., en_US).

The i18n/request.js loads translation files, and i18n/routing.js tells next-intl which locales shall be supported.

All content to be translated will be located under the messages directory, with content being separated out under locales, messages/en-US/errors.json and files being organized by components, messages/en-US/home.json.

The NextIntlClientProvider is wrapped around the children in layout.tsx

        <NextIntlClientProvider messages={messages}>
          {children}
        </NextIntlClientProvider>

This allows components to reference the content from a page, like:

import {useTranslations} from 'next-intl';
import {Link} from '@/i18n/routing';
 
export default function HomePage() {
  const t = useTranslations('HomePage');
  return (
    <div>
      <h1>{t('title')}</h1>
      <Link href="/about">{t('about')}</Link>
    </div>
  );
}

For more documentation on what is available in next-intl, go to: https://next-intl-docs.vercel.app/docs/getting-started

Crowdin - Localization Management Platform

We have a free, open-source license with Crowdin: http://crowdin.com.

In the app, we are using the Crowdin Cli to upload content, and download translated files from the site.

You can log into the site with the dmsp-translate account, and navigate to https://crowdin.com/project/dmptool to see the source files and translations.

In order to upload new changes to the content, run the following scripts from package.json:

  • crowdin:upload - to upload content changes in english
  • crowdin:download - to download any new translations

For more documentation on Crowdin, go to:

Environment Variables

For the development environment, the environment variables are stored at .env.local. This is set as the default env file in jest.setup.ts. These variables must be set in order for the app to work.

Running the App

npm run dev

Building for Production

npm run build
npm run start

Docker Setup

This project users Docker for both development and production environments. The setup is managed with docker-compose.yml files that build off Dockerfile.dev for development and Dockerfile.prod for production. Additionally, the project includes a buildspec.yaml file for AWS CodeBuild, which also references these Dockerfiles.

Development

  1. Build and run the development Docker containers
docker-compose build
docker-compose up
  1. To stop the docker container, run
docker-compose down
  1. Run the following to check that your container is up
docker container ls -a
  1. Access the site at http://localhost:3000

Production

  1. Build and run the production Docker container
docker build -f Dockerfile.prod -t dmsp_frontend_prototype:dmsp_frontend_prod .

docker run -p 3000:3000 dmsp_frontend_prototype:dmsp_frontend_prod
  1. Access the Next.js app at http://localhost:3000

Project Structure

|-- app/
|   |-- api
|       |-- setCookie
|           |-- __tests__
|           |-- route.ts
|   |-- login
|       |-- __tests__
|       |-- login.module.scss
|       |-- page.tsx
|   |-- layout.tsx
|   |-- page.tsx
|   |-- not-found.tsx
|-- components/
|   |-- Header
|       |-- __tests__
|       |-- header.module.scss
|       |-- index.tsx
|   |-- context
|       |-- AuthContext.tsx
|       |-- CsrfContext.tsx
|-- cypress
|   |-- e2e
|       |-- spec.cy.ts
|-- generated
|   |-- graphql.tsx
|-- graphql
|   |-- affiliations.query.graphql
|-- lib
|   |-- graphql
|       |-- client
|           |-- apollo-client.ts
|-- public
|   |-- images
|       |-- dmptool_logo_u166.svg
|-- styles
|   |-- globals.scss
|   |-- elements.scss
|-- utils
|   |-- logger.ts
|   |-- authHelper.ts
|   |-- authLink.ts
|   |-- errorHandler.ts
|-- buildspec.yaml
|-- codegen.ts
|-- docker-compose.yml
|-- DockerFile.dev
|-- Dockerfile.prod
|-- middleware.ts
|-- next.config.mjs
|-- package.json
|-- tsconfig.json

Authentication

There are currently two context files, AuthContext.tsx and CsrfContext.tsx, that help manage the auth state and the csrf token state.

All forms submitted to the backend need to include the CSRF token in the header of the request. GraphQL queries have inherent handling of csrf protection as long as we remember to include the CONTENT-TYPE as application/jsonin the header, which is currently added by the utils/authLink used in the apollo client instance.

The endpoint to refresh the csrf token is called when getting a 403 returned from the backend server.

Testing

When new components or functions are added, corresponding unit tests should ideally be added. We want to shoot for above 70% coverage. React components are tested using jest and React Testing Library.

Unit tests should generally be placed in a __tests__ folder adjacent to the component or file being tested.

Functional tests are conducted using Cypress. These tests can be run in headless mode using the command npm run cypress:run or in headed mode is npm run cypress:open. Functional tests should be placed in the cypress directory.

API Routes

  • GET /api/check-auth: returns whether user is authenticated based on presence of auth token in cookie

Contributing

  1. Clone the repo from github (git clone [email protected]:CDLUC3/dmsp_frontend_prototype.git)
  2. Create a new branch prefixing branch with bug or feature based on type of update (git checkout -b feature/your-feature)
  3. Add your changes and add commit message (git add .; git commit -m "added new feature)
    • A pre-commit is run with the commit which checks to make sure linting and coverage pass before a commit goes through
  4. Push to the branch (git push --set-upstream origin feature/your-feature)
  5. Open a Pull Request in github

Contributors

License

This project is licensed under the MIT License - see the LICENSE for for details.