The best way to handle a disaster is to avoid it. Let me explain that with an example. If you're building a house and you find out the material used to build it was subpar there is a little you can do to fix it other than building it again. Else keep spending hard-earned money fixing things every 6 months.
The same is true for software engineering. Once buggy/subpar code gets merged into your default branch dealing with it becomes a huge problem. In such a situation you could do one of two things
- You spend time refactoring the inefficient code. In cases when the code in question is fundamental to other features, this becomes increasingly difficult and may throw timelines off track.
- The second option, is you decide to move on. Fix the bugs that QA reports and keep adding features on top in order to meet timelines. This approach may help you meet your timelines in the short term but results in large technical debt. It could even result in instability in the product.
Either way, the best way to handle such situations, is to avoid them!
So how do you do that?
- Have a linter in place. Now the whole team follows the same conventions.
- Compulsory unit tests and set high test coverage thresholds.
- A robust CI pipeline with at least the following stages
- Checkout
- Install dependencies
- Lint
- Build
- Test
- Having integration tests is really helpful but in many cases, timelines and business priorities do not allow for it.
The above are proven ways of reducing the probability of failures. However, when dealing with front-end applications aesthetics and styling play an important role in the acceptance criteria.
It is possible to automate even this but the more time that you spend in writing tests the lesser flexibility you have when it comes to adapting to user feedback and developing software in a truly agile manner.
Every time you iterate based on user research and feedback you will have a bunch of broken tests and you'll spend time rewriting/fixing seemingly meaningless tests.
An alternate approach to writing automated visual tests would be to visually check before it gets merged. In a traditional sense to do this, the reviewer/tester would need to
- pull the source code
- ensure their environment is set up correctly.
- install the dependencies locally.
- run the app
- manually test the application.
Seems like an awful lot of work and something that would take a lot of time. So I automated steps 1 through 4.
In this tutorial, I will take you through how to create a UAT environment for a React application every time a PR is created to the default branch. In my case, the default branch is develop
This tutorial assumes you have a solid understanding of
In this tutorial, you will
- Configure react-router to use relative paths instead of absolute ones
- Setup an s3 bucket that will host the UAT environments. For each branch, we will create a separate folder and copy the artifact in that folder.
- Create a workflow that will handle UAT deployments when a PR is created to the develop branch
Starter Project
Please clone the following repository: https://github.com/wednesday-solutions/react-uat-on-pr. Create a new branch
Configure react-router to add support for relative paths
Copy the following snippet into the utils/history.js file
Add a UAT build script in the package.json
Add the build:uat to the scripts in the package.json
Add internal utils
Create the internalUtils file.
Copy the following code into the newly created file
- isUAT will be used to verify if the current ENVIRONMENT_NAME is uat
- getBasePublicPath function returns a relative base path if isUAT returns true.
Configure webpack to handle relative paths
Step 1
Add this snippet to the internals/webpack/webpack.base.config.js
Step 2
Add this snippet to the internals/webpack/webpack.prod.config.js. Ensure the OfflinePlugin is configured correctly and doesn't break in non-uat environments.
Step 3
Now we'll add a new route to ensure that routing works correctly once the app is hoisted. Add a new route in the app/utils/routeConstants.js
Add this snippet in the containers/App/index.js
Add this snippet to the app/routeConfig.js
Now that we have set up the new route, let's add a button to navigate to it. Add the snippet below in the app/containers/App/index.js
Step 4
Now run the build:uat script
Step 5
Copy the contents of the build folder in a subfolder within it. This is simulating what will happen in the workflow. We will use a separate directory for each branch.
Now run the application
Go to http://localhost:3000/test-relative-route
Now that we've verified that our logic works locally let's setup the workflow to automate deployments to s3.
Setup s3 bucket
Step 1
Login to the AWS console https://console.aws.amazon.com/console/home
Step 2
Go to S3 https://s3.console.aws.amazon.com/s3/home?region=ap-south-1
Step 3
Create a new bucket.
Excuse this brief pause, but here's something worthwhile: Join the ranks of elite C Execs who are already benefiting from LeadReads. For all things digital products.
Join here.
Enable static website hosting
Step 1
Navigate to the newly created bucket and go to the properties tab
Step 2
At the bottom of the page you will see the Static Web Hosting option. Edit and enable it. Save the changes.
Step 3
Once your changes are saved you'll see the link to your bucket. Copy it and keep it handy. You will need it in a few minutes.
Setting up secrets
Step 1
Go to the settings tab
Step 2
Go to the secrets section.
Step 3
Add secrets for AWS_REGION, AWS_SECRET_ACCESS_KEY, AWS_ACCESS_KEY_ID and AWS_S3_BUCKET
Create a new workflow for UAT deployments
Step 1
Create a new file
Step 2
Copy the snippet below into the newly created file
- This workflow will be triggered whenever a PR is created to the develop branch
- In the react-template, the build artefact is stored in build directory. AWS_REGION, AWS_SECRET_ACCESS_KEY, AWS_ACCESS_KEY_ID and AWS_S3_BUCKET are set using Github secrets.
- Checkout the code
- Get the name of the branch
- Install all of the dependencies
- Create a UAT build
- Copy the artefact to s3. The Destination directory is the same as the branch name.
Push your code and create a PR
Upon successful deployment head over to your S3 bucket. You will now see a folder there with the same name as your branch.
Now use the base link of your s3 deployment, the same one that you copied after enable static hosting.
Mine is: http://uat-on-pr.s3-website.ap-south-1.amazonaws.com/
Add the name of your branch to this base URL like this: http://uat-on-pr.s3-website.ap-south-1.amazonaws.com/feat/uat
Now head over to that link and VIOLA!
Where to go from here
I hope you enjoyed this tutorial on how to create an environment for UAT on creation of PR for a React application. If you have any questions or comments, please join the forum discussion on Twitter.
I would highly recommend taking a look at the CI pipeline that comes along with the project here.