https://covid.mohamedtayeh.com/
https://www.youtube.com/watch?v=IuJ9-EEIx9g
A simplified COVID-19 vaccination worldwide statistics dashboard. The website leverages data visualization libraries to showcase charts and figures of the data. The application supports search and filtering features that allow the users to navigate the dashboard based on data for a specific timeframe or country. In addition, a user is allowed to sign up and customize the dashboard based on personal preferences for filtering options and colour schemes saved. To support a larger user base, the application provides multi-lingual support.
Task: Leaving deployment aside, explain how the app is built. Please describe the overall code design and be specific about the programming languages, framework, libraries and third-party api that you have used.
- MERN stack - MongoDB, ExpressJs, React, and NodeJs. We also used GraphQL as our API structure. In doing so, we used Apollo Client on our frontend to make the API calls to our backend. Our frontend scripts are written in TypeScript, while our backend scripts are in JavaScript.
Frontend General:
- Context to handle global changes, such date filters, dark mode and chosen language
- Apollo client requests were made using the hooks provided and the requests were made in the component that required them.
- Queries were then put into a separate file to encourage modularity (for the most part)
- Generic components were made such as Loading and Error
- Styles were put into a separate folder except for dyanmic styling and MUI specific styles
- Map:
- One generic map component that supports multiple legends and different geojson polygon shapes was re-used to ensure DRY code
- Components of the map were split into their own components to ensure modularity
- Feature data was stored in a storage URL online on GitHub to avoid a large bundle size
- MUI templates: Material UI for React was extensively used for the design of the website this allowed to have a shared theme throughout the app that can be altered easily
Backend General:
- Schemas were partitioned to encourage modularity, between global, map, authorization, user configs and data pipeline
- Resolvers were partitioned and put into folders that follow this structure
- Mongoose was used to query our data base
- Data Pipeline are in their own folder which are scripts that call onto external endpoints to fetch, tidy, transform and send the data to our backend. This script has its own username and password in the db and is the only user that is allowed to enter vaccination data into the database. To ensure that all data (even past in case of an error from external sources), only the data that is changed is sent to the server. A copy of the vaccination data is serialized and stored in the droplet to ensure that only new or updated data is sent to our database compared to the last update made.
Database General:
- The vaccination data schema is based isoCodes and the data that we have on each isoCode
Frontend libraries:
- @googlemaps/react-wrapper: official google maps library wrapper for react
- Translation: i18next, i18next-browser-languagedetector - to create the translation feature
- mui: material ui for react elements such as buttons, etc.
- chart.js: for the charts
- react-router-dom: routing
- apollo-client: graphql requests
- react-18next: translation feature
Backend libraries:
- node-cron - handles making a script run on schedule (every day at 9AM EST)
- node-fetch - makes server-side request to the data sources that we are using in the project
- csvtojson - used to convert csv format (some data sources) to json format
- lodash - specifically used for cloneDeep because of issues between shallow vs deep copies when doing the data pipline
- graphql-request - used to make graphql requests to the db from the script, light wrapper therefore it was used
- moment - handles creating the logs current time for when the pipeline makes a request to the backend
- mongoose - used to interact with mongodb
- mongodb - to interact with the database
- graphql - to work with graphql
- express-graphql - same as above
- mongo-sanitize - to sanitize inputs into mongoose queries
- jsonwebtoken - to generate a secure token used for authentication
- other: libraries taken from the course - nodemon, bcrypt, body-parser, cors, express, fs
Other:
- mapshaper - used to merge geojson data and lower its precision to make it as light weight as possible
Task: Explain how you have deployed your application.
We deployed our application through a Digital Ocean droplet. This droplet has extra memory so that we can run our application more smoothly. In the droplet, there are 4 docker containers. There is the nginx-proxy container, which houses the reverse proxy. There's a frontend container and a backend container, which both have exposed ports for the reverse proxy. Then, there is the nginx-proxy-acme container, which generates an SSL certificate and handles the renewal of certificates. The setup of these containers are all handled by the docker-compose.yml
](https://github.com/UTSCC09/project-fullstackarmy/blob/main/docker-compose.yml) file. In addition, the docker images for the frontend and backend are built using their respective Dockerfile
's (See frontend Dockerfile
and backend Dockerfile
). Because the proxy is nginx and our website uses React Router, we needed to add some extra options when using the jwilder/nginx-proxy image to change the generated nginx.conf
in order to allow for the routing.
We set up a CI/CD pipeline using Github Actions. Whenever either frontend
, backend
or docker-compose.yml
is changed in the main
branch, the workflow copies the updated file/folder(s) to the VM in our Digital Ocean droplet and runs the necessary commands. That is, if either the frontend
or backend
folders were changed, then there would be a docker-compose build [whichever folder]
run, followed by the docker-compose up -d
command to deploy. docker-compose.yml
only triggers the deployment command. If the workflow is successful, there is a checkmark seen next to the commit in GitHub. If not, then we debug what failed in the workflow. However, to further make sure that everything is running smoothly we double check the deployed website and see if everything looks okay. If absolutely necessary, we can also ssh into our VM and check the status of our docker containers if in case something happened to them during the build/deployment.
Task: Explain how you monitor your deployed app to make sure that everything is working as expected.
-
General: Sentry was set up on frontend, backend and data pipeline to ensure we are aware of any errors that come up. The sample rate for the frontend and backend was put to 0.3 as recommended in production evironments; the sample rate for data pipeline is 1 since these are critical errors. In addition we integrated sentry with the discord API to get updates in case of any errors, since we do hope that our website does get used after the course is complete.
-
Data Pipeline: A collection was specifically made to monitor the number of records that were sent and successfully recorded in the data base set up, this allows us to build our own dashboard in the future to monitor the site. In addition text logs were set up in the backend as well.
Task: What is the top 3 most challenging things that you have learned/developed for you app? Please restrict your answer to only three items.
- Data Pipeline, setting up the schema, wrangling data from sources, setting up scripts with schedulers, and deploying the scripts
- Understanding the google maps API, and ensuring that the map code is optimized, without overloading the main thread and follows DRY principles
- Deployment takes time, and the VMs at school are not as reliable as a VM from outide, like Digital Ocean
Task: Describe the contribution of each team member to the project. Please provide the full name of each team member (but no student number).
-
Mohamed Tayeh
- Deployment
- CI/CD
- Maps Implementation
- Data Pipelines
- Data Sources
- Data Wrangling and Transformation
- Schema Structure
- Plans for Data Visualization
- Write Up / Translations
- Translation feature
- Excel download
- Researching best practices - linting, typescript, etc.
- Frontend sign/signup component
- Small frontend components, such as Loading and Error, and some style fixes
- Sentry
-
Aaren Chu:
- Deployment
- CI/CD
- Herd Immunity Bar and Series Charts, including GraphQL API calls
- Saved User configurations
- Set up of organization and clusters in MongoDB Atlas
- Design of tab structure to display one heat map at a time
- Schema Structure
- Authentication
- Sanitizing Mongoose queries
-
Raha Gharadaghi
- Design of UI/frontend skeleton
- Implementation of information tab parallax scroll
- Filtering by region and date range (making queries for the charts responsive to the selected values and applying changes to the chart display)
- Dark Mode/Light Mode feature
- Vaccine Distribution chart (including GraphQL API calls)
- Languages dropdown for translation
Task: Any additional comment you want to share with the course staff?
-
Chrome violations
-
These are not errors nor bugs, they are indications of where improvement can be made to the code
-
Our violations are coming from google maps api library rather than our code.
-
There are 4 violations that occur due to the google maps, particularly due to having many features in the map
[Violation] 'requestAnimationFrame' handler took 125ms
, in the console this occurs in themap.js
(not our files)
This mainly occurs when zooming in and out the map, when panning the camera too fast. The animation of having the features on the map takes too long, which outside of our controls since we are using the google maps API to draw the features on the map.
[Violation] 'setTimeout' handler took 81ms
, in the console this occurs in thepoly.js
(not our files)
This occurs when hovering over a feature that is big and outside of the maps bounds. This animation/feature is from the google maps api libary therefore its speed is outside of our control
[Violation] 'setTimeout' handler took 221ms
, in the console this occurs in themap.js
(not our files)
This occurs when clicking the scaled legend switch in the google maps, this is caused due to it takes a bit of time for google maps to recolor all the features present on the map
[Violation] Added non-passive event listener to a scroll-blocking 'touchstart' event. Consider marking event handler as 'passive' to make the page more responsive. See https://www.chromestatus.com/feature/5745543795965952
Mohamed has asked about this on Piazza and implemented the solution in
FeaturePolygon.ts
however this violation occured instead. Mohamed has talked with google maps developers/maintainers in their discord as well and it was not from his code. He has sent this portion of the code to them to be able to create a repo with this issue. -