/testmaps

Python API and React client microservices. Running via Docker, GitLab CI and Heroku. API using Flask-RESTplus, PostgreSQL and Pytest. React client using react-testing-library for testing.

Primary LanguageJavaScript

pipeline status

Udacity Fullstack Developer Capstone Project:

TLDR

Users

I have created 3 users for the reviewer to make use of. These are listed below, the password was provided in the submission message.

The emails below are not real email addresses, they are just used for purposes of logging in.

You can view the "Profile" details including the user id, token and permissions by clicking on the user email address in the navbar at top right. From there, you can use the token for making requests via Swagger.

Note: You will see several endpoints documented in Swagger, however, for the purposes of this project, only the "wakeparks" endpoint is in scope for this project.

Motivation

I wanted to create a project that pulled together many of the skills I have learned in both this Udacity Fullstack Nanodegree, as well as some skills I learned from my previous React Developer Nanodegree. Additionally, there were elements I felt were missing from both courses that I needed to learn and incorporate.

What was missing from the Nanodegrees?

The Fullstack Nanodegree contained a section on running applications through Docker, the example was somewhat straightforward and, importantly, did not include a database or frontend. To my mind, the trickiest part of working with Docker is getting multiple technologies working together cohesively. Hence, I worked to implement such an approach in this project.

The React Developer Nanodegree, while fairly broad in scope - covering React, Redux and React Native - the React Developer ND did not include anything in regards to testing. As such, I wanted to learn how to write tests for React. Additionally, the React Developer Nanodegree did not cover how to manage authentication and authorisation.

Goal

My goal was to create a project that would include several elements of a modern production workflow. This project, while not wide in scope or features, has more depth in terms of tooling than what I have done before. In brief, the goal was to achieve the following:

  1. API documented via Swagger.
  2. API well covered by automated tests.
  3. Application running on Docker.
  4. Frontend built in React.
  5. Frontend supported by automated tests (Jest and React Testing Library).
  6. Inclusion of end-to-end tests. (Cypress).
  7. API and frontend deployed via CI/CD pipeline to the web. (GitLab CI and Heroku)

Overview

The application is made up of two parts:

  • API
  • Client

The API, database, and client all run from Docker and get pushed through a CI/CD pipeline using GitLab CI to Heroku.

API

The API is built in Python, Flask and Flask-RESTplus. I selected Flask-RESTplus since it was similar to Flask-RESTful and comes with Swagger support out of the box. Coming from a frontend background, I feel Swagger (or a good Postman collection) are important so that others can explore and experiment with the API.

Client

The frontend is built in React with tests in Jest and React Testing Library. The test coverage for the frontend is lower than the API since I am still figuring out how to "mock Auth0" amongst other things.

While Auth0 has a wide array of tutorials an documentation, they seem to have comparitively little on how to unit test against it. That being said, I did (thankfully) find resources that helped me manage some of this. However, I still have plenty to learn when it comes to mocking entire services. The frontend test coverage needs work.

Deployment

When it comes to deployment, the application is run through a GitLab CI/CD pipeline, and will deploy to Heroku only if all linting, formatting and testing passes for both the API and the client. (You can see the pipeline badge on the GitLab version of the repo here)

API Documentation

I am not listing the API documentation in Markdown here since I have incorporated Swagger into the project. Please take a look at the link below:

Swagger: https://waketestmaps.herokuapp.com/docs

In order to use Swagger for endpoints that require authentication, visit the client app, login and then visit the /profile page by clicking on your username at top right. On that page I have displayed the relevant user details including the JSON Web Token which can be pasted into Swagger for use.

Roles

The application uses Auth0 for authentication and authorisation along with role-based access (RBAC). The roles are:

  • No role - all visitors do not have a role and can only "GET" wakeparks.
  • Parkadmin - users that "own/create" a wakepark will be assigned the parkadmin role, this effectively lets them post, put or patch wakeparks they have created.
  • Superadmin - user that has all permissions and the only role with permission to delete wakeparks.

What this means in real terms is the following:

  • All users/visitors without a role can only view (GET) wakepark listings.
  • Users with the "parkadmin" role can create or update wakeparks they have created.
  • Users with the "superadmin" role can create, update and delete wakeparks created by any user.

Permissions

The available permissions are:

  • get:cableparks (All)
  • post:cableparks (Superadmin, Parkadmin)
  • put:cableparks (Superadmin, Parkadmin)
  • delete:cableparks (Superadmin)

Reflection on Project

While this app is limited in scope and features, I feel it represents a solid foundation from which to develop future features from. Learning how to pull together all these various strands into a cohesive project was a large learning curve. Combining Docker, backend, frontend, testing and a CI/CD pipeline I think this project really is "fullstack" despite the scope and feature set being very small at this stage.

Where to find the app?

Reviewer Notes

I feel that this project has expanded my horizons. As a front-end developer coming from the Angular world, pretty much everything used in this project was more or less unknown to me before October 2019 (I did have a bit of experience with React prior to my React Developer Nanodegree). Outside of the course materials, I had to learn about Docker, GitLab CI, Cypress, Jest, React Testing Library, Swagger, Flask-RESTplus not to mention the Auth0 nuances. Finally, getting them all working together nicely was yet another learning process.

Development Notes

Below are reference notes related to developing and running the project. Please read through them to get accustomed and started.

GitHub

Check project out at: https://github.com/nicholaspretorius/testmaps Git Clone: git clone https://github.com/nicholaspretorius/testmaps.git

GitLab

Same repo is present on GitLab at: https://gitlab.com/nicholaspretorius/testmaps

CI/CD pipeline is run from GitLab.

Instructions

Local development requires the following:

Install and activate a Python 3.7 virtual environment as follows:

  • python3.7 -m venv env
  • source env/bin/activate
  • pip list
  • pip install -r requirements.txt

Swagger - API Documentation

When viewing on web:

When running locally:

Setup Local DB

If you are not using Docker, you can run the app locally as follows - however, I would discourage this as it is not "intended" to be run this way:

  • dropdb users_dev -p 5433
  • createdb users_dev -p 5433
  • python manage.py recreate_db
  • python manage.py seed_db
  • Comment line #22 and uncomment line #23 of config.py

Run Local

To run locally via Python:

  • export FLASK_APP=project/__init__.py
  • export APP_SETTINGS=project.config.DevelopmentConfig
  • export FLASK_ENV=development
  • python manage.py run

Run docker-compose

The recommended way of running this application is via docker-compose:

Note: Anywhere you see "dc" this is my shortcut for "docker-compose", which has been setup via an alias.

  • docker-compose up --build
  • To sanity check that the API is running, visit: http://localhost:5001/hello
  • See logs docker-compose logs -f (Ctrl + c to quit)
  • docker-compose -f docker-compose.prod.yml up -d --build
  • docker-compose -f docker-compose.prod.yml exec users python manage.py recreate_db
  • docker-compose -f docker-compose.prod.yml exec users python manage.py seed_db

Docker Housekeeping

Docker can be a resource hog on your local machine, as such it helps to do some Docker "housekeeping" occassionally.

  • Show Docker disk usage: docker system df
  • Remove Docker images: docker rmi $(docker images -f "dangling=true" -q)
  • Removed unused volumes: docker volume prune
  • Removed unused Docker resource: docker system prune

Database

To setup and populate the database with some data, do as follows:

  • docker-compose exec users python manage.py recreate_db
  • docker-compose exec users python manage.py seed_db
  • dc exec users python manage.py reset_db

In order to view Postgres from the Docker container, do as follows:

  • docker-compose exec users-db psql -U postgres
  • \l list the databases
  • \c users_dev connect to the 'users_dev' db
  • \dt list relations for that db

Run tests

To run the API tests, you can do the following. Please note the various options that can be run to specify different areas of testing. there are currently 101 tests covering just over 90% of the application.

  • docker-compose exec users pytest "project/tests"
  • docker-compose exec users pytest "project/tests" --disable-warnings
  • docker-compose exec users pytest "project/tests" -p no:warnings
  • dc exec users pytest "project/tests" -p no:warnings --cov="project"
  • dc exec users pytest "project/tests" -p no:warnings --cov="project" --cov-report html
  • Rerun tests that failed in last run: docker-compose exec users pytest "project/tests" --lf
  • Run tests with only "config" in the name: docker-compose exec users pytest "project/tests" -k config
  • Run tests with only "production" in the name: docker-compose exec users pytest "project/tests" -k production
  • Stop tests after first failure: docker-compose exec users pytest "project/tests" -x
  • Enter PBD after first failure and end the test session: docker-compose exec users pytest "project/tests" -x --pdb
  • Stop after two failures: docker-compose exec users pytest "project/tests" --maxfail=2
  • Show local variables in traceback: docker-compose exec users pytest "project/tests" -l
  • List the two slowest tests: docker-compose exec users pytest "project/tests" --durations=2

Flake8, black, isort

Flake8 has been used for linting,, black for formatting and isort for ordering of imports.

  • dc exec users flake8 project

  • dc exec users black project --check

  • dc exec users black project --diff

  • dc exec users black project

  • dc exec users /bin/sh -c "isort project/*/*.py --check-only"

  • dc exec users /bin/sh -c "isort project/*/*.py --diff"

  • dc exec users /bin/sh -c "isort project/*/*.py"

  • Pycodestyle (formerly known as Pep8) dc exec users pycodestyle project

Run the shell

You can run the Flask shell as follows:

  • docker-compose exec users flask shell

Logging

While this has not been used in this project, it was initially written to help with logging. This is not considered part of the active project, but I have kept it for possible future reference if needed.

  • Import: from project.utils.logger import log
  • Usage: log.debug(f"Message goes here: {var_name_here}") or log.error(f"Message goes here: {var_name_here}")
  • Levels: info, debug, error
  • Check the logging.conf file for how console and file are setup as handlers.

Ultimately, this will not be used. Sentry would be used in order to log for both the API and the frontend.

Run production Dockerfile

To run the production Docker, in preparation for deploying to Heroku, do as follows:

  • docker build -f Dockerfile.prod -t registry.heroku.com/waketestmaps/web .
  • docker run --name waketestmaps -e "PORT=8765" -p 5002:8765 registry.heroku.com/waketestmaps/web:latest
  • Visit http://localhost:5002/hello
  • docker stop waketestmaps
  • docker rm waketestmaps
  • docker push registry.heroku.com/waketestmaps/web:latest
  • heroku container:release web
  • Visit https://waketestmaps.herokuapp.com/hello
  • heroku run python manage.py reset_db

Flask Admin

To view the Flask-Admin console:

  • Visit http://localhost:5001/admin/user/

Client

Install dependencies with Yarn: yarn install

Note: For some reason create-react-app running after an npm install has problems using fsevents with npm test and the client tests will not run. Using yarn install fixes this and npm test and npm start will work as expected.

Run on Docker

To run via Docker, run the following:

  • export REACT_APP_USERS_SERVICE_URL=http://localhost:5001 && dc up --build
  • From project root run: dc up --build
  • Visit http://localhost:3007

Run on local

To run on local, run the following commands:

  • export REACT_APP_USERS_SERVICE_URL=http://localhost:5000
  • From /services/client run: npm start
  • Visit http://localhost:3000

Tests

  • From the /services/client folder run: npm test
  • Runs all tests in --watch mode
  • Run with coverage: ./node_modules/.bin/react-scripts test --coverage
  • Note: The --coverage option seems to be inconsistent. At times it reports actual coverage and at others it just displays nothing. I have investigated this, and got it working, only to find at a later date it no longer reports coverage. This is an open issue. Feedback or advice on this is welcomed.

e2e Tests

  • To run Cypress, change directory into /services/client, run: ./node_modules/.bin/cypress open
  • To run Cypress from Docker run: docker-compose run cypress ./node_modules/.bin/cypress run

Linting

To lint the frontend client code:

  • npm run lint
  • dc exec client npm run lint

Formatting

To format the frontend client code:

  • npm run prettier:check
  • npm run prettier:write
  • dc exec client npm run prettier:check
  • dc exec client npm run prettier:write

Run Dockerfile.deploy locally

To test/run the "deployed" Dockerfile, run the following:

  • export REACT_APP_USERS_SERVICE_URL=http://localhost:8007
  • export DATABASE_URL=db_name_goes_here
  • docker build -f Dockerfile.deploy -t registry.heroku.com/waketestmaps/web .
  • docker run --name waketestmaps -e PORT=8765 -e DATABASE_URL="$(echo $DATABASE_URL)" -e REACT_APP_USERS_SERVICE_URL="$(echo $REACT_APP_USERS_SERVICE_URL)" -e "SECRET_KEY=secret_key_here" -p 8007:8765 registry.heroku.com/waketestmaps/web:latest
  • docker exec -it waketestmaps python manage.py reset_db
  • For client visit: http://localhost:8007/
  • For api visit: http://localhost:8007/wakeparks/
  • For Swagger visit: http://localhost:8007/docs
  • docker stop waketestmaps
  • docker rm waketestmaps

Deployment

To release to Heroku (for the first time or if the Docker setup has changed) the following is required:

Notes on Auth0

  • When using Google (or social login), auth.checkSession will require login on refresh for localhost (see here).
  • When mocking Auth0, you can use this approach.