The website for the Midwest Asian Pacific Islander Desi American Students Union
Production
Staging
- Overview
- Installation
- Workflow
- Netlify Hosting
- REST Functions
- React Framework
- Typescript
- Linting
- Project Structure
- Module Organization
- Styling and CSS
- Branding
- State Management
- Testing
- Responsive Web Design
- Notes
This document summarizes a design process of the MAASU website, what softwares and tools were used, and any executive decisions made in regards to project structure. The Table of Contents provides a summary of all relevant categories.
To install all dependencies, use the Yarn Package Manager:
yarn install
To start a development server:
yarn start
Yarn is faster to install dependencies than most its primary competitor npm
because
it concurrently downloads packages. Yarn's dependency lock versions are placed in a
yarn.lock
rather than a confusing package-lock.json
. In terms of security, npm
allows installed dependencies to execute code, a large security vulnerability in any
project.
On the user-friendly front, yarn
has simpler and more intuitive commands than npm
. Below is a
comparison of commonly used commands where yarn
makes things easier.
command | yarn |
npm |
---|---|---|
install | yarn |
npm install /npm i |
testing | yarn test |
npm run test |
build | yarn build |
npm run build |
custom {script} | yarn script |
npm run script |
In this repository, there are two main branches: staging
and master
, corresponding to the staging
and production
environments, respectively. With the integration of Netlify as a hosting service, any pushed commits to these branches will immediately trigger builds in production.
All other branches are feature
branches, where each branch will contain some type of feature or bugfix. These changes are then merged into staging
. To put these changes into production, create a pull request to merge the staging branch into master
. While there are many ways to merge branches, this is the safest way because creating a Github pull request makes an explicit statement regarding the changes to be made on the public website. These can also be detailed and labeled for future reference. After the changes have been merged into master
, they will automatically be deployed into production.
Pull requests should be labeled with the labels provided and a list of significant changes should be detailed in the request body. Once approved, the requested commits will be squash merged into master
to reduce the amount of commits.
Utilizing git
and github
:
- Check out the staging branch to verify it is up to date with the latest changes in production.
git checkout master && git pull
git branch -D staging
git checkout -b staging
git push origin +staging
This deletes the local staging branch and syncs it with master.
The last command will force push the commits in master to the staging branch. Be cautious when doing so, as it may overwrite any changes you originally had in staging. Make sure you have no changes in staging you may not want to overwrite.
- Make your changes to the
staging
branch, whether through a feature branch or manually. - Commit those changes.
git add *
git commit -m "commit title"
- Pull the latest updates from
master
, fixing any conflicts, then push to Github.
git pull --rebase origin master
git push origin +staging
Note: To avoid conflicts, try not to drastically change too many files at once. It's a better strategy to create short and quick fixes and merge those commits together.
- Create a PR to merge into
master
. - Merge changes into
master
. - Delete any feature branches that may have been created.
git push origin --delete my-feature-name
- Repeat steps 1-7.
For a basic idea of workflow:
feature branch
/staging
→ pull request
→ master
As mentioned before, it's dangerous to have ready access to production without some form of checks. The staging environment is instituted to be able to view and changes as a preview, and allow other members of the eboard to approve any incoming changes (if necessary).
Pull requests allow a second layer of checking, well as a fantastic form of documentation. The only content
pushed to master
should be squashed commits. This is so that:
master
stays clean of small, unnecessary commits.- The commits are organized into categories of work through pull requests.
The pull requests are labeled to make it easier for future developers to find where a specific bug was solved, or a feature was introduced. The template PR makes it clear which year and/or conference for which the commit was intended to be contributed towards. This makes it much easier to backtrack and find old issues or reference old materials.
Since MAASU is a conference-based organization, it is best practice to tag a release for every significant change to the site, i.e promo of upcoming conference, conference registration, and other significant website updates. This will result in around four or five releases in a given year, depending on the circumstances.
Netlify was chosen as a website host due to its continuous integration capabilities. It integrates with Github hooks to be able to deploy new revisions to the web application at any time. It also records logs, allows custom environment variables, functions, and more. It also provides free SSL certificates with any domain, which is a massive positive. I remember the days when SSL certificates had to manually be added to sites, and it was quite a pain to configure.
I originally used to use Firebase Hosting, but the service is not as polished as Netlify. It requires cli tools to deploy and integrates heavily with Google, adding a lot of unecessary strain and bloat (especially with DNS management). Firebase Hosting may one day overtake Netlify, but until then, it will still be considered a beta service.
Netlify also provides more than Firebase Hosting in the free (or "spark") plan(s). Firebase forces a hard
limit of 10GB
of stored data, while Netlify allows unlimited file sizes. Nelify additionally provides a
100GB
bandwidth/month as opposed to Firebase's 10GB
bandwidth. In Firebase's defense, Netlify places a hard
limit of 300
build minutes/month, but it is unlikely that build minutes will ever exceed that limit.
Source: Netlify pricing vs Firebase Pricing
Netlify Functions was chosen for its functions capability over Firebase Functions and AWS Lambda largely due to its pricing tiers. Netlify Function's free tier provides more than either alternate service, and AWS Lambda does not even provide a free tier. While AWS Lambda may be the most scalable in terms of pricing, Netlify Functions provides the simplest integration for smaller organizations, nonprofits, and startups.
Netlify Functions does not monitor RAM or CPU usage, and instead measures usage based on time, which is slightly better than Firebase Functions. This provides a slight edge to Netlify.
Service | Requests/Mo | Runtime/Mo | GB-Sec/Mo | CPU-Sec/Mo |
---|---|---|---|---|
Netlify Functions | 125k | 100hr | ||
Firebase Functions | 125k | 40k | 40k |
The solidifying deal-breaker comes down to this - Firebase Functions restricts outbound networking to Google services only. While Google is used as our primary form of organization and communication, this creates too much of a sole dependency on Google. As a result, Netlify Functions was chosen for serverless function hosting.
Luckily, since Netlify is also coincidentally used for its website hosting service, it easily integrates with Github in terms of deployment and analytics.
Source: Netlify Functions vs Firebase Functions vs AWS Lambda
This website is built using React.js, a Javascript component-based web devleopment framework. React was chosen as a framework for this website as opposed to any other framework because of its component-based philosophy. Breaking down any software into smaller reusable components, or modules, allows for greater flexibility. It allows developers to focus on a specific piece of a project rather than understanding all pieces at once.
This app in particular was created using Create React App because of its ease of use. With Create React App, it's easy to quicky create a simple project with Javascript or Typescript support without having to worry about building, public folders, and other extensions such as CSS modules. I've built projects using only raw Javascript, React, and Webpack, and it is much easier to begin with a lightweight boilerplate including everything you will ever need rather than beginning from scratch.
The reason most web developers are moving from older systems such as Django
and Rails
is that with tools
such as Create React App
and Webpack
is for its static-site generation capabilities. Instead of running
an interpreter or a Python cgi-bin script on some web server, tools like Create React App
make it possible
to build sites in Typescript and compile all sources files into pure HTML, CSS, and Javascript. This makes
the final production code lightweight, minified, and fast.
This project uses Typescript as its primary language for development. Typescript is essentially a strongly-typed javscript language which transpiles into Javascript on compile.
When I first tried to learn Typescript over a year ago, I gave it up and decided it was not for me. After all, Typescript seemed to take away the very thing that made Javascript so powerful - the ability to change types. Why would we regress in functionality? So here's the reasoning.
Typescript catches bugs before they occur. One of the greatest strengths of a compiled language like
C
or Java
is that they can catch type errors before runtime. An interpreted language such as Javascript
doesn't have the luxury of catch compile-time errors. So why does that matter?
Catching bugs before they occur is the whole point of software. It's the whole reason software developers are given technical interviews and coding tests. To see how proficient you are at building a piece of software without bugs. With a strongly typed language, you can verify that arrays don't change types in the middle of code. Javascript allows more room for error, with arrays being able to be rewritten as strings, and functions turning into numbers. Typescript reduces the possibility or type errors by introducing strict typing to the wonderful syntax of Javascript.
And don't worry. If you're a long time Javascript user like me, all it takes is motivation to learn. When I first tried to learn Typescript, I hated it and went immediately back to Javascript. But after learning why Typescript is so powerful and taking the time to relearn it, I realized Typescript is so much better for development than Javascript, and it's now my programming language of choice in web development.
Eslint is a tool used to detect bugs before they have the possibility to occur in production. It does so by adhering the codebase to a specific set of styles. If the codebase does not follow the guidelines, it will provide warnings and errors, as well as instructions on how to fix such errors. It is an indirect form of static testing, and helps prevent code bugs before they happen.
Prettier Is also a tool used for code formatting. Prettier changes code formatting to match a set of styling guidelines, usually on buffer save. These changes will usually be spacing issues and indentation.
Although these tools may seem similar in nature, they both have different purposes and are used in coordination to maintain a proper codebase. In essence, Prettier helps maintain formatting and code readability, while Eslint enforces a strict adherance to code quality and syntax decisions. This summary provides a condensed description of how the two tools are different, and how they can be used in conjunction to improve productivity and prevent bugs before they happen.
The src
file is organized as follows:
src/
api/ - Any and all api calls.
components/ - All functional, or pure, components concerned with UI. Generally,
these components should be small modular units.
constants/ - All constants, strings, and other hard values.
containers/ - Components dealing with data or store. This folder also contains
components which represent entire pages. Simply put, a container
"contains" other components.
css/ - Any and all styling.
functions/ - All REST API functions and handlers are contained within this
folder. All deploy to a separate service.
hooks/ - Reuseable React hooks.
routes/ - Manages all routes.
store/ - All store management and types are contained within this folder.
utils/ - Any and all utility functions.
index.tsx - Main source file which initializes react, store, and renders DOM.
While it may be a common standard to export a folder full of modules using an index.js
or index.ts
file, this methodology of module exporting and importing actually slows down development and can make
codebases confusing for a number of reasons. First, it makes development slightly more tedious when
creating new modules. Now, in addition to creating a new module, a developer needs to make sure the
module is imported and exported properly in an index
file. Depending on the folder and the project,
this file can grow exponentially with hundreds of unnecessary import and export statements.
It also adds diffculty to finding files. When trying to find a module or code file in question, it is
complicated to search through files that all use the same name. Personally, I use a fuzzy file finder
to locate all my files, and it makes things much more complicated when searching for a file and the
results are all named index.js
.
Finally, newer software development tools are beginning to recognize this anti-pattern and move away
from it. Deno is a prime example, removing the ability to "auto-import" index
files. It's a good idea to avoid this pattern to reduce confusion and stay up to date with rising
technologies.
Exporting a default export can also be considered an anti-pattern. The reason for this is simple - the name of the module can be changed when imported. This can cause many headaches when trying to debug modules in the future. for example, using default exports,
export default MyComponent;
...
import HelloRandomlyNamedComponent from './MyComponent';
The same module was renamed to something completely different. In the case of using standard exports,
export const MyComponent;
...
import { MyComponent } from './MyComponent';
The module cannot be renamed as easily. This makes it easier to track down a bug in a deeply-nested module, and prevents multiple modules being given the same name. For these reasons, standard exports are preferred to default exports.
All styles are created using the core concepts of functional CSS, with only one styling file written in SCSS/SASS. SCSS is used to be able to utilize useful tools such as better variables, mathematical calculations, and loops. I simply chose SCSS over SASS because I prefer its style.
With the arrival of React and other modular frontend frameworks, the concept of CSS styling is becoming obsolete. Now, there is no longer a need for massive CSS files containing convoluted class names and confusing styles which conflict with each other. Instead, a new form of CSS called Functional CSS is emerging. This form of styling breaks down classes into single-style classes with priority importance.
.border-red {
border-color: red !important;
}
.border-dashed {
border-style: dashed !important;
}
.border-width-3 {
border-width: 3px !important;
}
<div class='border-red border-dashed border-width-3'>
Functional CSS is a game changer!
</div>
If, like me, you are a long time CSS user, you may be wondering the point of it all. "That's ridiculous! Why would you even think of such a thing?!" I found myself asking the same questions. And here's the kicker.
Functional CSS removes the CSS from the codebase.
What do I mean by that? Imagine working on a large project. You start styling each element by theming, and then you realize you need to refactor some code. Oh, and your boss just told you to replace this component with this one. Where were those styles for that component... let's look in our large css file... what were the class names again?...
I think this situation has happened to everyone, where throughout the duration of a project, your stylesheets become
more and more convoluted with random styles and strange class naming conventions to the point where it becomes nearly
impossible to effectively change styles. And then there's inheritance. You want a div
to be display: block
but it
just happens to be display: inline
from its parent styles, then you need display: block !important
, but then that
breaks the display of its children, etc, etc...
Functional CSS solves all these issues. By providing single-style classes, each style is on its own and has no direct consequences on its parent or its children. It is fully modular in that these classes and meant to be applied to lower level HTML components, which then are used and reused throughout the project. You'll never need to worry about convoluted class names or large css files because once you code a css class, you never have to change it ever again.
Goodbye, old CSS. Hello, functional CSS.
Branding for MAASU is still being determined!
State managment is a key factor in organization of React applications, especially as they grow. While React has built-in state management, this application uses a combination of redux, react-redux, redux-thunk, and redux-devtools-extension. While the former three are necessary for this application to function, the latter simply exists to make development and state debugging much easier.
Why is a popular library like Redux needed for a React application? Doesn't React come with a built-in state? I found myself asking those questions a year ago, and for the longest time, I refused to use Redux because I found no purpose to it other than application bloat and unnecessary debugging stress. So here's the things I learned to change my mind.
-
UI state is different from data state, and the two should be separated.
In the same way frontend is separated from backend, data should not mix with UI, nor should either be interdependent. With Redux, UI state is completely managed by React state, while data is managed completely by Redux. This reduces overlap and UI state bugs tearing apart data state. It also keeps the application more organized overall.
-
Redux prevents unpredictable changes to state.
The most annoying thing about React (and one of its strongest components) is its state management. When state is updated, it's complicated enough to understand how state changes affect rendering, when and how a component's children are updated, and so on. As an application grows new components, data sources, data structures, theis becomes a complicated mess of components updating its siblings, parents, children, and it becomes nearly impossible to keep track of what components update which slices of state.
Redux is different. Redux makes state management predictable by separating state from the components so that a component cannot directly update the state. Instead, actions must be manually dispatched to update the state (known as the store), and each action is accounted for and recorded. Redux-thunk makes logging these actions much simpler.
-
Project organization.
As mentioned before, Redux separates data state from UI state. However, it also manages store separately from components, meaning it becomes much easier to group connected slices of state, regardless of which components use it.
-
Global state
This was the main selling point of Redux for most people before React hooks, but it provides a simple wrapper for providing a global state across components.
With the introduction of React hooks, it could technically be replicated in pure React using context, but at that point, you'd essentially be coding Redux from scratch.
It goes without saying that the core concepts of Redux are complicated to wrap your head around at first. But investing the time to learn the basics of Redux will be worth it.
Testing is one of the most important aspects of software development. It ensures confidence that the software you run will not produce any issues or errors in production. This project focuses on a few different types of testing strategies:
Unit/Integration Tests: Jest
Jest is a fantastic Javascript/Typescript/Babel testing library that can be extended to perform a number of different tasks such as test watching, coverage, and DOM environment testing. It is easy to write tests and mocks, and even simpler to structure tests or run certain subsets of tests.
These tests become especially important with the introduction of components in React's framework, and it becomes extremely important to test each component used in coordination with other components.
End-to-End (e2e) Tests: Cypress
Cypress is an end-to-end testing framework which comes built with assertion libraries, mocks, stubs, and more. It is very user-friendly and can be used to test browser-specific features. The syntax is very clean and easy to use and get started.
Typescript automatically tests data and variables types at compile time, preventing bugs. Eslint also contributes by checking styling and formatting to catch any overlooked errors introduced through poor code styling.
This project is built and pushed to a staging server before the code is finally put into production, allowing current MAASU ECC board members to monitor and test any new additions or features.
The motivation behind this decision is to allow greater transparency within the organization. The technical networking role is such a mysterious position to outsiders, and it provides (to some degree) clarity of progression and improvements made to the other members of the board.
Two breakpoints were chosen to separate web flows into three different sections, designated for mobile-tablet and tablet-desktop borders respectively.
The styling consists of four media queries:
small (mobile only)
not small (tablet and desktop)
not large (mobile and tablet)
large (desktop only)
Responsive web design is a key in serving consistent content to both desktop devices and mobile devices. While Google recommends five breakpoints, their philosophy is designed for large teams of developers seeking fine-tuned control over each and every device's layout. While more device-specific layouts are pleasant, they
- Create edge cases for each specific device
- make it harder for developers to change overall layout
- Disorient users familiar with a different layout
As such, we have opted to provide only two breakpoints. Having a consistent layout among three
device targets makes it much easier and more familiar for users to switch between devices.
Additionally, we have not chosen any queries such as medium only
is because it reduces the
urge to build a third and separate layout from small
and large
, and encourages a similar
layout of all three screen sizes. This ensures that switching between desktop and mobile will
more or less provide a similar experience to users.
One of the most frustrating aspects of component importing is trying to determine the correct relative
path. By using Typescript, import becomes much easier by altering the tsconfig.json
to read directly
from the src/
folder.
{
"compilerOptions": {
"baseUrl": "src"
},
"include": [
"src"
]
}
This makes import much easier, from:
import Header from '../../../components/header'
To:
import Header from 'components/header'
Netlify makes hosting much easier than other hosting services, but there are some caveats. For single page sites like those made
with React, Netlify doesn't redirect other urls to the site by default. This will make any routing with react-router
or reach-router
return a 404 page, which can be scary (and I happened to discover this while in production). To allow Netlify to redirect all urls
to React, add a _redirects
file to the public
folder containing the following line:
/* /index.html 200
This can also be done by adding the following configuration to a netlify.toml
file:
[[redirects]]
from = "/*"
to = "/index.html"
status = 200