#monorepo
#workspace
#infra
Written by Paul
Yarn Workspaces
For my side project, I created a new GitHub organization to manage everything there. Initially, I faced a few challenges:
- Deploying via Vercel made it challenging to manage in a monorepo format.
- Thus, instead of using a monorepo structure, I managed separate, scattered repositories.
I initially considered a monorepo setup, but Vercel’s limitations for monorepo management added complexity. So, I stuck with separate repositories until I began customizing server management on EC2, which allowed me to transition to a monorepo.
Project Structure
The structure is as follows:
root |-- apps | |-- client | |-- server | |-- react-native-app |-- packages | |-- design-system | |-- notion-utils | |-- prisma-schema | |-- storybook |-- docker |-- terraform
In
apps
, I have actual service projects like web, mobile, and server apps. These services are deployed on EC2 using Docker, which I’ll cover shortly.In
packages
, I collected reusable libraries like utility functions and the design system. With a monorepo setup, I didn’t need to publish these to npm, which simplified reuse.React Native in a Monorepo
An interesting aspect is the integration of the React Native project as a package in the monorepo. I referred to react-native-universal-monorepo for guidance. By adjusting paths for
node_modules
and Podfile
references, the setup worked smoothly.Using the react-native-monorepo-tools npm package, I modified
metro.config.js
as follows:/* eslint-disable @typescript-eslint/no-var-requires */ const { getDefaultConfig, mergeConfig } = require('@react-native/metro-config') const exclusionList = require('metro-config/src/defaults/exclusionList') const { getMetroTools, getMetroAndroidAssetsResolutionFix } = require('react-native-monorepo-tools') const monorepoMetroTools = getMetroTools() const androidAssetsResolutionFix = getMetroAndroidAssetsResolutionFix() const config = { transformer: { publicPath: androidAssetsResolutionFix.publicPath, getTransformOptions: async () => ({ transform: { experimentalImportSupport: false, inlineRequires: false, }, }), }, server: { enhanceMiddleware: (middleware) => { return androidAssetsResolutionFix.applyMiddleware(middleware) }, }, watchFolders: monorepoMetroTools.watchFolders, resolver: { blockList: exclusionList(monorepoMetroTools.blockList), extraNodeModules: monorepoMetroTools.extraNodeModules, }, } module.exports = mergeConfig(getDefaultConfig(__dirname), config)
For Android, I modified paths in
app/build.gradle
to reference the packages correctly.Docker and Docker Compose
With the monorepo established, I moved to set up CI/CD pipelines.
Initially, I used Load Balancer + EC2 for the infrastructure, but since this is a side project, I realized that a 24/7 setup was unnecessary and costly. So, I switched to an NGINX + EC2 stack, significantly reducing costs.
I also set up a naming convention for Docker files and wrote the
docker-compose.yml
file as follows:version: '3.8' services: coldsurf-io: platform: linux/amd64 build: context: ../ dockerfile: ./docker/Dockerfile.coldsurf-io args: GITHUB_TOKEN: '${GITHUB_TOKEN}' PORT: 4001 image: '${ECR_REMOTE_HOST}/coldsurf/coldsurf-io:latest' env_file: - ../apps/coldsurf-io/.env ports: - '4001:4001' ...
After building Docker containers, I uploaded them to ECR and pulled them from EC2. Using GitHub Actions, I set up manual deployment triggers.
Terraform Introduction
For minor needs, I tried using Terraform. One task involved granting access to EC2 on port 22 (SSH) to the GitHub Actions runner’s IP. I automated this security rule using Terraform:
provider "aws" { region = "ap-northeast-2" } data "http" "myip" { url = "<http://ipv4.icanhazip.com>" } resource "aws_security_group" "coldsurf-terraform-sg" { name = "coldsurf-terraform-sg" description = "Security group for coldsurf and terraform" ingress { from_port = 22 to_port = 22 protocol = "tcp" cidr_blocks = ["${chomp(data.http.myip.body)}/32"] } ... }
This configuration allows only the current GitHub Actions runner’s IP access to port 22 during CI runs.
NGINX Configuration on EC2
Deploying multiple services via Docker Compose required a root NGINX configuration on the EC2 server. I set up NGINX to direct traffic based on domain as follows:
server { if ($host = api.billets.coldsurf.io) { return 301 https://$host$request_uri; } server_name api.billets.coldsurf.io api.wa-museum.coldsurf.io wamuseum.coldsurf.io blog.coldsurf.io coldsurf.io; ... }
Each service is then routed via NGINX config:
server { server_name wamuseum.coldsurf.io; location / { proxy_pass <http://127.0.0.1:4002>; ... } }
If
403 Forbidden
errors occur, it may be due to filesystem permission issues. Adding www-data
to the appropriate group can help resolve this:$ sudo usermod -aG ubuntu www-data $ sudo chmod g+rw /path/to/file_or_directory
Managing SSL with Let’s Encrypt
Replacing the original Load Balancer + ACM setup, I transitioned to SSL management with Let’s Encrypt on NGINX.
In Closing
I’m considering Fargate for the server infrastructure because, as a side project, the service doesn’t need continuous uptime, and Fargate’s usage-based billing is a better fit for variable traffic.
EC2 + ECR vs. Fargate
- EC2 + ECR:
- Cost-effective if resources are needed 24/7.
- Auto Scaling is possible but less flexible.
- Pros: Cost-saving with reserved or spot instances.
- Cons: Higher management overhead.
- Fargate:
- Serverless: Start/stop based on requests; ideal for irregular traffic.
- Billing: Charges per second of CPU/memory usage.
- Auto Scaling is more flexible.
- Cons: Higher costs if resources are needed constantly.
Summary:
- 24/7 resource needs: EC2 + ECR is more cost-effective.
- Intermittent usage with traffic spikes: Fargate is ideal for optimizing resource costs.
For anyone interested, you can check out the monorepo at https://github.com/coldsurfers/surfers-root 🙂