mirror of
https://github.com/r-freeman/portfolio.git
synced 2025-01-18 07:25:41 +00:00
This commit is contained in:
parent
66eea98efc
commit
ceaf81d59b
@ -8,7 +8,7 @@ and skills, as well as provide information about me and my interests.
|
||||
- Language: [TypeScript](https://www.typescriptlang.org/)
|
||||
- Framework: [Next.js](https://nextjs.org/)
|
||||
- Database: [Supabase](https://supabase.com/)
|
||||
- Deployment: ~~[Vercel](https://vercel.com/)~~ Self-Hosted on Raspberry Pi 5
|
||||
- Deployment: [Self-Hosted on Raspberry Pi 5](https://ryanfreeman.dev/writing/migrating-from-vercel-to-raspberry-pi-5)
|
||||
- Styling: [Tailwind CSS](https://tailwindcss.com/)
|
||||
- Integrations: [Spotify](https://spotify.com/)
|
||||
|
||||
|
BIN
app/writing/migrating-from-vercel-to-raspberry-pi-5/casaos.png
Normal file
BIN
app/writing/migrating-from-vercel-to-raspberry-pi-5/casaos.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.2 MiB |
@ -1,7 +1,8 @@
|
||||
import {ArticleLayout} from '../../../components/layouts/ArticleLayout'
|
||||
import {createSlug} from '../../../lib/createSlug'
|
||||
import raspberryPi from './vishnu-mohanan-rZKdS0wI8Ks-unsplash.jpg'
|
||||
import Image from 'next/image';
|
||||
import casaOS from './casaos.png'
|
||||
import Image from 'next/image'
|
||||
|
||||
export const metadata = {
|
||||
authors: 'Ryan Freeman',
|
||||
@ -23,12 +24,146 @@ Recently, I decided to migrate this website from Vercel to a self-hosted environ
|
||||
|
||||
## The Why: Leaving Vercel behind
|
||||
|
||||
Don't get me wrong, the developer experience with Vercel is in my opinion second to none. Push code and forget about it, while Vercel automagically deploys your apps. Vercel abstracts all the complicated CI/CD processes behind a nice UI and takes care of everything under the hood.
|
||||
Don't get me wrong, the developer experience with Vercel is in my opinion second to none. Push code and forget about it, while Vercel automagically deploys your apps. Vercel abstracts all the complicated CI/CD processes behind a nice-looking UI and takes care of everything under the hood.
|
||||
|
||||
However, I wanted to explore a more hands-on approach, one that would give me more control over my deployment process and for this I needed a new home for my website.
|
||||
However, I wanted to explore a more hands-on approach, one that would give me more control over my deployment process and reduce my costs. For this I needed a new home for my website.
|
||||
|
||||
## The Hardware
|
||||
|
||||
The Raspberry Pi 5 seemed like the perfect solution to run my website from. It's a powerful little device, cheap to run and since I'm already running a small media server on another Pi, I was at least somewhat familiar how to set everything up.
|
||||
|
||||
I wanted fast and reliable storage so that websites and apps I'm going to deploy would be responsive. I settled on this [Geeekpi NVMe adapter](https://amzn.to/3yH64bf) and [Kingston 500GB NVMe](https://amzn.to/4g07NsI) which had sufficient storage capacity for my needs.
|
||||
I wanted fast and reliable storage so the websites and apps I'm going to deploy would be responsive. I settled on this [Geeekpi NVMe adapter](https://amzn.to/3yH64bf) and [Kingston 500GB NVMe](https://amzn.to/4g07NsI) which had sufficient storage capacity for my needs.
|
||||
|
||||
Next up I needed a cooling solution and a case for the Raspberry Pi, so I chose the [GeeekPi Raspberry Pi 5 Metal Case with Official Active Cooler](https://amzn.to/3AA2MXL). The active cooler is low-profile and compatible with the NVMe adapter so it was perfect for keeping the Pi cool.
|
||||
|
||||
With a small bit of assembly the Raspberry Pi was ready to go, so now I needed to set up the software.
|
||||
|
||||
## The Software
|
||||
|
||||
The next step was to get the Raspberry Pi up and running with an OS. While there are many Linux distros to choose from, I decided to go with Ubuntu Server 64bit as I wanted to use the Pi in a headless configuration.
|
||||
|
||||
Next, I set up and configured SSH to enable me to connect remotely to the Raspberry Pi and continue to configure my new deployment stack.
|
||||
|
||||
### CasaOS
|
||||
|
||||
I wanted to manage the Pi from a browser for certain tasks, so I chose [CasaOS](https://casaos.zimaspace.com/) for this. CasaOS is free and open-source software which takes care of installing Docker and features a nice frontend with an app store for installing other Docker based apps.
|
||||
|
||||
<Image src={casaOS} alt=""/>
|
||||
|
||||
I set up CasaOS with a single command — `curl -fsSL https://get.casaos.io | sudo bash` and within a couple of minutes I was able to access its Apple-like UI where I could access the app store to continue setting up my deployment stack.
|
||||
|
||||
### Docker
|
||||
|
||||
[Docker](https://www.docker.com/) was the obvious choice for containerising my applications. It's a powerful tool that I've used before, although I had no experience with building and deploying my own images. As mentioned CasaOS comes bundled with Docker so there was no extra set up.
|
||||
|
||||
With Docker, I could easily build and run containers for my projects, isolating them from each other and making deployment a cinch.
|
||||
|
||||
## Dockerising my website
|
||||
|
||||
With a few additions to the codebase of this website, I was able to completely Dockerise and package it into a distributable image, which could be readily deployed as a container on the Raspberry Pi.
|
||||
|
||||
Next, I'll highlight some of the main changes to the codebase which made Dockerising my website possible.
|
||||
|
||||
### Dockerfile
|
||||
|
||||
A Dockerfile is required for building Docker images and should go in the root directory of the project. Next.js has tons of different examples of use cases for their framework, so I borrowed the Dockerfile from [this example](https://github.com/vercel/next.js/tree/canary/examples/with-docker). This saved me from writing a lot of boilerplate code.
|
||||
|
||||
### next.config.mjs
|
||||
|
||||
Next, I made a [small addition](https://github.com/r-freeman/portfolio/blob/main/next.config.mjs#L15) to the `next.config.mjs`. This builds the Next.js app as a standalone app inside the Docker image.
|
||||
|
||||
### .dockerignore
|
||||
|
||||
The `.dockerignore` file is used for excluding files and directories to keep the final image as lean as possible. In my case I excluded `.git`, `.next` and `node_modules` as these were not required in the final image.
|
||||
|
||||
### Building the Docker image
|
||||
|
||||
With these changes, the last thing to do was to test building the image locally with `docker built -t portfolio .` this instructs the Docker engine to build the image using the Dockerfile.
|
||||
|
||||
## Enter Gitea: The Open-Source Alternative
|
||||
|
||||
Next, I needed a Git service to host my code. Again GitHub is great and I can't compete with it but this endeavor was all about going open-source and self-hosted. [Gitea](https://about.gitea.com/) was on my radar for a while as a great, lightweight alternative to GitHub.
|
||||
|
||||
It was easy to set up thanks to CasaOS, and it offered all the core features I needed such as Gitea Actions, which would be essential for my new deployment stack.
|
||||
|
||||
## Automating deployments with Gitea Actions
|
||||
|
||||
One of my primary goals was to automate the deployment process. On Vercel, everything was automated out of the box, but I wanted to replicate that seamless workflow on the Raspberry Pi. Gitea Actions turned out to be just the tool I needed.
|
||||
|
||||
With Gitea Actions I could easily define my CI/CD pipelines directly within my repositories. Setting up Gitea Actions involved writing a [simple YAML configuration file](https://git.ryansnet.xyz/r-freeman/portfolio/src/branch/main/.gitea/workflows/publish.yml) that would trigger on every push to the main branch.
|
||||
|
||||
```yaml
|
||||
name: Build And Publish
|
||||
run-name: ${{ gitea.actor }} runs ci pipeline
|
||||
on: [ push ]
|
||||
|
||||
jobs:
|
||||
BuildAndPublish:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Checkout code
|
||||
uses: https://github.com/actions/checkout@v4
|
||||
|
||||
- name: Use Node.js
|
||||
uses: https://github.com/actions/setup-node@v3
|
||||
with:
|
||||
node-version: '18.17.0'
|
||||
|
||||
- name: Decrypt secrets
|
||||
run: ./decrypt_secrets.sh
|
||||
env:
|
||||
SECRET_PASSPHRASE: ${{ secrets.SECRET_PASSPHRASE }}
|
||||
|
||||
- name: Login to Docker Hub
|
||||
uses: docker/login-action@v3
|
||||
with:
|
||||
username: ${{secrets.DOCKER_HUB_USERNAME}}
|
||||
password: ${{secrets.DOCKER_HUB_PASSWORD}}
|
||||
|
||||
- name: Set up Docker Buildx
|
||||
uses: https://github.com/docker/setup-buildx-action@v3
|
||||
with:
|
||||
config-inline: |
|
||||
[registry."docker.io"]
|
||||
mirrors = ["mirror.gcr.io"]
|
||||
|
||||
- name: Build and push Docker image
|
||||
uses: https://github.com/docker/build-push-action@v6
|
||||
with:
|
||||
context: .
|
||||
file: ./Dockerfile
|
||||
push: true
|
||||
tags: ${{secrets.DOCKER_HUB_USERNAME}}/${{vars.REPO_NAME}}:v1
|
||||
|
||||
- name: Stop and remove old Docker container
|
||||
continue-on-error: true
|
||||
run: |
|
||||
sudo docker stop ${{vars.REPO_NAME}}
|
||||
sudo docker rm ${{vars.REPO_NAME}}
|
||||
|
||||
- name: Pull new image and start Docker container
|
||||
run: |
|
||||
sudo docker pull ${{secrets.DOCKER_HUB_USERNAME}}/${{vars.REPO_NAME}}:v1
|
||||
sudo docker run -d --restart unless-stopped --env-file ./.env --name portfolio -p ${{vars.DEPLOY_IP}}:3000:3000 ${{secrets.DOCKER_HUB_USERNAME}}/${{vars.REPO_NAME}}:v1
|
||||
|
||||
```
|
||||
|
||||
The script would pull the latest changes, build a Docker image, upload it to the registry, stop and remove the existing container, pull the new image from the registry and finally deploy the container with the new image.
|
||||
|
||||
It was very satisfying to push code and watch it go live within a couple of minutes, all running on the Raspberry Pi. However, my website was only visible on the local network at this point.
|
||||
|
||||
## Wrapping it all up
|
||||
|
||||
The final piece of the puzzle was to expose my portfolio website to the Internet, so that it could be seen outside my network. However, I didn't like the idea of port forwarding through my router and exposing my IP address to the world.
|
||||
|
||||
I had some experience with [Cloudflare Tunnels](https://www.cloudflare.com/en-gb/products/tunnel/) which allows you to use the Cloudflare network as a proxy to expose your apps to the Internet.
|
||||
|
||||
Setting this up was straightforward, I changed the nameservers on my domain name to point to Cloudflare, which allows Cloudflare to create the necessary DNS records to set up the Cloudflare Tunnel.
|
||||
|
||||
After waiting for a few minutes for the DNS records to propagate, my website was back up and running again.
|
||||
|
||||
## Conclusion
|
||||
|
||||
Migrating from Vercel to a self-hosted environment on my Raspberry Pi has been a rewarding experience. It taught me a lot about Linux, Docker, Gitea and networking. Most importantly, it's given me complete control over my development and deployment process.
|
||||
|
||||
Of course its more work than using a managed service like Vercel, but the trade-offs have been worth it. I’ve learned valuable skills, saved on hosting costs, and enjoyed the satisfaction that comes from running my own server.
|
Loading…
Reference in New Issue
Block a user