Skip to content

Latest commit

 

History

History
363 lines (241 loc) · 18.9 KB

README.md

File metadata and controls

363 lines (241 loc) · 18.9 KB

Boilerplate Client React

Branch Status
develop Shift3
main Shift3

This boilerplate has a wiki which explains the project and its implementation in much greater detail than the code comments.

Note that this repository used to be compatible with our NestJS backend, however we have switched to using django as our primary backend. If you are looking for the NestJS compatible version, we still maintain it in the nestjs-compatibility branch

Demo

You can see a demo of the project on either our production site or to see a demo with new features, you can check out our staging site

Quick Start

To start the project, make sure yarn is installed on your local machine. If you have already installed our laptop script, you should already have yarn.

  1. Install Dependencies via yarn install
  2. Start the Project in development mode via yarn start

Open http://localhost:4200 to view the project in the browser.

The page will reload if you make edits. You will also see any lint errors in the console.

Running unit tests

Run yarn test to execute the unit test runner in the interactive watch mode.
See the section about running tests for more information.

Running test coverage

Run yarn test-cov for a test coverage report in interactive watch mode.
For more information please refer to the react Coverage Reporting documentation.

Build

Run yarn build to build the app for production to the build folder.
It correctly bundles React in production mode and optimizes the build for the best performance.

The build is minified and the filenames include the hashes.
Your app is ready to be deployed!

Learn More

This project was bootstrapped with Create React App.

You can learn more in the Create React App documentation.

To learn React, check out the React documentation.

Deployment

Deploying the application requires having the aws and terraform cli commands installed on your machine. See the following links for OS specific installation instructions:

See the section about deployment for more information.

AWS

Deploying to AWS requires having AWS credentials configured on the machine. Deployment will also require aws cli v2. The deployment script is set to look for an AWS profile named BWTC-Developer. See the following links for documentation on configuring the AWS SSO and creating a named profile:

Terraform

Configuring, building, and changing the AWS infrastructure for the sandbox is handled by Terraform. As a prerequisite, Terraform needs the AWS credentials configured as described in the above section, which developers should already have or can access through Zoho Vault.

Before deployments or initializing terraform you must have aws sso configured and then login with the following commmand:

aws sso login --profile BWTC-Developer

Terraform also needs the project secrets saved in terraform/staging/terraform.tfvars. This file is not committed to version control since it can contain sensitive information such as database credentials and should be added locally. Create the terraform/staging/terraform.tfvars file with the following structure:

profile = "BWTC-Developer"

region = "us-west-2"

web_domain_name = ""

Secret Note
profile This must match the AWS credentials name on the development machine
region This is usually us-west-2
web_domain_name This will be the web domain name for the project, an example may be: example.shift3sandbox.com

After adding the terraform/staging/terraform.tfvars file, cd into the terraform/staging directory and run the following commands to configure and build the AWS infrastructure for the sandbox environment

terraform init
terraform apply

The infrastructure can be updated by changing the Terraform configuration files and running these commands again.

Environment Configuration

After provisioning the AWS instance with Terraform, the project environment variables need to be updated with the new server values.

Update the apiRoute property in environment.staging.ts with the provisioned sandbox instance url.

Build and Deploy

The boilerplate can either be deployed manually, or automatically via CircleCI. The preferred way to deploy is with automatic deployments. The default CircleCI configuration will deploy to the staging environment when there are new commits pushed to the develop branch, and will deploy to production when new commits are pushed to the main branch.

CircleCI Deployments

After your CircleCI project is set up, the only thing you need to do to get deployments working for production and staging is to set 4 environment variables in the CircleCI project settings.

  • STAGING_S3_BUCKET_NAME
    • This is the full S3 bucket name to deploy to.
    • my-app-bucket.shift3sandbox.com
  • STAGING_AWS_ACCESS_KEY_ID
    • The AWS access key ID used to authenticate with AWS.
    • AKIAIOSFODNN7EXAMPLE
  • STAGING_AWS_SECRET_ACCESS_KEY
    • The AWS secret key used to authenticate with AWS.
    • wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY
  • STAGING_AWS_DEFAULT_REGION
    • The default region your infrastructure is deployed to.
    • us-west-2
  • SENTRY_DSN

Once these are set up, your project will be automatically deployed whenever new commits to the develop branch are pushed to Github.

Production deploys from the main branch use the same set of environment variables, just with PRODUCTION instead of STAGING int the names. The list of those variables follow:

  • PRODUCTION_S3_BUCKET_NAME
  • PRODUCTION_AWS_ACCESS_KEY_ID
  • PRODUCTION_AWS_SECRET_ACCESS_KEY
  • PRODUCTION_AWS_DEFAULT_REGION

Manual Deployments

You can also manually deploy, however this requires you to have the AWS CLI setup locally and loaded with your credentials. The automatic deployments are recommended.

In package.json, updated the "deploy:staging" and the deploy:production npm script by replacing the $BUILD_DIRECTORY_PATH and $AWS_SANDBOX_URL placeholders with your project values. For example:

// package.json
{
  ...,
  "scripts": {
    ...
    "deploy:staging": "aws s3 sync ./build s3://example-staging.shift3sandbox.com --profile BWTC-Developer --delete"
    "deploy:production": "aws s3 sync ./build s3://example-prod.shift3sandbox.com --profile BWTC-Developer --delete"
    ...
  }
}

Then use yarn to run the deploy:staging or deploy:production script.

yarn run deploy:staging

Development

Local Development

To work with the project directly, the development machine needs node and yarn installed.

Development Server

Run yarn install to install dependencies. Then run yarn start to start the development server listening on port 4200. You can view the app in the browser by navigating to http://localhost:4200. The app will automatically reload if any source files are changed.

Docker

This project can be run as a Docker container (it is not recommended for involved development because it makes it harder to debug the codebase). Run docker-compose up to build and run the container. The app will be mapped to the port 4200 on your local machine and you can view the app in the browser by navigating to http://localhost:4200. It supports hot reloading so the app will automatically reload if any source files are changed.

Template Repository

This project is configured as a template repository. It creates one commit in the new project based on the template instead of the entire original boilerplate history.

Updating Code Owners

In the repository there should be a .github folder with a CODEOWNERS file inside. This file represents who the owners of the repository code are. When you clone this repo, or use it as a template for a new project, you need to update this file to represent the new owners (you and whomever may be on your project). Simply remove the current owners in the file, and replace them with you and your teamates! The syntax is simply:

@<github username>

Be sure to add the github usernames of all developers on your project. Now anytime a pull request is created, all codeowners are added as reviews automatically! It also becomes a reference point when the project is picked back up in the future. We can easily see who has the best context for the code even years down the line. For more information you can click this link:

Github Codeowners

Updating Issue Templates

Currently the issue templates may have some things you don't want or need in your new project. This can be anything from the tags being set, to the person assigned for each issue. Be sure to go to the settings for the repository, and click Set up templates to configure them in a way that suits your needs. For more information you can click this link:

Setting up issue templates

Initializing the Project

If this project is being cloned to start a new project, there are a few things that need to be updated to make it work. The project name will need to be updated in the README.md, package.json. The README also refers to the boilerplate, both in the text and in the CircleCI badges.

The project environment files will need to be updated with the path to the APIs. The development environment.dev.ts and environment.prod.ts files assumes a local development server of http://localhost:3000, which might need to be updated.

After provisioning and before deploying, the deploy:staging and deploy:production script in package.json needs to be updated.

Prettier

This project uses Prettier to enforce code style. It is highly opinionated by design with relatively scant options for customization. The thought process behind it is to ignore personal styling preferences and instead embrace consistency. There are .prettierrc and .prettierignore configuration files to adjust some options. Prettier is also wired up to a pre-commit hook. This DOES slightly slow down git, as it runs the hook on staged files every time git commit is executed.

Prettier can be configured within editors so that it formats files on save, which helps minimize any changes the pre-commit hook would need to make.

CI

This project is configured to work with CircleCI. CircleCI is a continuous integration and delivery platform for building and deploying your application.

The CI hanldes building the application, running tests, and running linters. All of these jobs must pass in order for the CI build to be successful.

This project has set up the CircleCI configuration here

It is recommended to use the above configuration, however if you choose to alter the configuration please visit the official CircleCI docs for guidance 'https://circleci.com/docs/2.0/config-intro/'.

Wiki Automation

The CircleCI config includes a deploy-wiki job to automatically deploy Wiki pages that are placed in the wiki/ folder. This workflow only runs when changes are commited directly to the develop branch or when a feature branch is merged into the develop branch.

After forking the project, you will need to make the following changes to ensure the deploy-wiki job works properly:

  1. Under the deploy-wiki job, change the working_directory to the name of your GitHub repository.
  2. Follow the CircleCI documentaiton on Creating a GitHub deploy key to add a deploy key with write access, and replace the SSH key fingerprint in the deploy-wiki job.

These steps only need to be performed once by a user with admin access to both GitHub repository and CircleCI project.

Webpack Bundle Analyzer

The project includes webpack-bundle-analyzer, which helps developers figure out the size of the project and its webpack dependencies. To use it, type yarn build in the project's directory in a terminal to create the webpack bundle. Run yarn run analyze, and webpack-bundle-analyzer will launch a server and browser window with a visualization of the project bundle size.

React Hook Form

We are using React Hook Form to extend our forms and make easy to work with validation. for more information see: https://react-hook-form.com/ts for typescript support.

Yup

Yup is a JavaScript object schema validator. With Yup, a developer can define a schema (or structure) which specifies the expected data type of each property in an object. The schema can also specify additional validations such as if a property is optional/required, the min/max length of string properties, the min/max value of numeric properties, a regular expressions to match, etc.

A Yup object schema is created using Yup's object.shape() method. For example, below is a simple schema describing a person object:

import * as yup from 'yup';

const personSchema = yup.object().shape({
  name: yup.string().required(),
  age: yup.number().required().positive().integer(),
});

Once a schema has been defined, existing objects can be validated against the schema using the isValid or isValidSync methods defined on the schema object. isValid performs the validation asynchronously and returns the result in a Promise whereas isValidSync performs the validation synchronously and returns a boolean. For example:

const validPerson = {
  name: 'John',
  age: 25,
};

const invalidPerson = {
  name: 'John',
  age: -25.5,
};

// validate asynchronously

personSchema.isValid(validPerson).then((valid) => {
  // valid === true
});

personSchema.isValid(invalidPerson).then((valid) => {
  // valid === false
});

// or synchronously

personSchema.isValidSync(validPerson); // => true

personSchema.isValidSync(invalidPerson)l // => false

For more information, see the Yup documentation.

resolvers

The useForm hook accepts an optional resolver function which allows you to use any external validation library such as Yup or your own custom validation logic to validate your forms. To simplify the integration with existing validation libraries, React Hook Form provides the optional @hookform/resolvers module which contains utility methods to create resolver functions from library specific object schemas.

For example, below we use the yupResolver from the @hookform/resolvers module to convert a Yup object schema into a resolver function:

import { useForm } from 'react-hook-form';
import { yupResolver } from '@hookform/resolvers/yup';
import * as yup from 'yup';

const personSchema = yup.object().shape({
  name: yup.string().required(),
  age: yup.number().required().positive().integer(),
});

const PersonForm = () => {
  const { register, handleSubmit } = useForm({
    resolver: yupResolver(schema),
  });

  return (
    <form onSubmit={handleSubmit(d => console.log(d))}>
      <input {...register('name')} />
      <input type='number' {...register('age')} />
      <input type='submit' />
    </form>
  );
};

For more information, see the useForm and @hookform/resolvers documentation.

Contributing

For information on contributing to this repo, see the CONTRIBUTING.md file