mirror of
https://github.com/jbowdre/runtimeterror.git
synced 2025-03-01 04:52:14 +00:00
91 lines
6.8 KiB
Markdown
91 lines
6.8 KiB
Markdown
---
|
|
series: Projects
|
|
date: "2021-08-20T00:00:00Z"
|
|
lastmod: 2022-02-03
|
|
usePageBundles: true
|
|
tags:
|
|
- gcp
|
|
- cloud
|
|
- serverless
|
|
title: Free serverless URL shortener on Google Cloud Run
|
|
---
|
|
### Intro
|
|
I've been [using short.io with a custom domain](https://twitter.com/johndotbowdre/status/1370125198196887556) to keep track of and share messy links for a few months now. That approach has worked very well, but it's also seriously overkill for my needs. I don't need (nor want) tracking metrics to know anything about when those links get clicked, and short.io doesn't provide an easy way to turn that off. I was casually looking for a lighter self-hosted alternative today when I stumbled upon a *serverless* alternative: **[sheets-url-shortener](https://github.com/ahmetb/sheets-url-shortener)**. This uses [Google Cloud Run](https://cloud.google.com/run/) to run an ultralight application container which receives an incoming web request, looks for the path in a Google Sheet, and redirects the client to the appropriate URL. It supports connecting with a custom domain, and should run happily within the [Cloud Run Free Tier limits](https://cloud.google.com/run/pricing).
|
|
|
|
The Github instructions were pretty straight-forward but I did have to fumble through a few additional steps to get everything up and running. Here we go:
|
|
|
|
### Shortcut mapping
|
|
Since the setup uses a simple Google Sheets document to map the shortcuts to the original long-form URLs, I started by going to [https://sheets.new](https://sheets.new) to create a new Sheet. I then just copied in the shortcuts and URLs I was already using in short.io. By the way, I learned on a previous attempt that this solution only works with lowercase shortcuts so I made sure to convert my `MixedCase` ones as I went.
|
|
data:image/s3,"s3://crabby-images/0a77a/0a77a632536a49c9f5f188158c724f953f68451c" alt="Creating a new sheet"
|
|
|
|
I then made a note of the Sheet ID from the URL; that's the bit that looks like `1SMeoyesCaGHRlYdGj9VyqD-qhXtab1jrcgHZ0irvNDs`. That will be needed later on.
|
|
|
|
### Create a new GCP project
|
|
I created a new project in my GCP account by going to [https://console.cloud.google.com/projectcreate](https://console.cloud.google.com/projectcreate) and entering a descriptive name.
|
|
data:image/s3,"s3://crabby-images/728ec/728ecc20c5f4dda619bda383bd04c27b2a0c03bd" alt="Creating a new GCP project"
|
|
|
|
### Deploy to GCP
|
|
At this point, I was ready to actually kick off the deployment. Ahmet made this part exceptionally easy: just hit the **Run on Google Cloud** button from the [Github project page](https://github.com/ahmetb/sheets-url-shortener#setup). That opens up a Google Cloud Shell instance which prompts for authorization before it starts the deployment script.
|
|
data:image/s3,"s3://crabby-images/9ed4f/9ed4f1f6bb375729b7fd9fd0acdb717a17c0f9a2" alt="Open in Cloud Shell prompt"
|
|
|
|
data:image/s3,"s3://crabby-images/a77f2/a77f237360d66e2acd341de49731bd2d73c385a5" alt="Authorize Cloud Shell prompt"
|
|
|
|
The script prompted me to select a project and a region, and then asked for the Sheet ID that I copied earlier.
|
|
data:image/s3,"s3://crabby-images/321f4/321f4db203fbca7e96f45a0c2189b91e88b33f7d" alt="Cloud Shell deployment"
|
|
|
|
### Grant access to the Sheet
|
|
In order for the Cloud Run service to be able to see the URL mappings in the Sheet I needed to share the Sheet with the service account. That service account is found by going to [https://console.cloud.google.com/run](https://console.cloud.google.com/run), clicking on the new `sheets-url-shortener` service, and then viewing the **Permissions** tab. I'm interested in the one that's `############-computer@developer.gserviceaccount.com`.
|
|
data:image/s3,"s3://crabby-images/d63ac/d63ac541249922a9d31b26cfafc4b068c1cd5780" alt="Finding the service account"
|
|
|
|
I then went back to the Sheet, hit the big **Share** button at the top, and shared the Sheet to the service account with *Viewer* access.
|
|
data:image/s3,"s3://crabby-images/dfde7/dfde714b6aedd78d8411110dfa6a676f7adc5b69" alt="Sharing to the service account"
|
|
|
|
### Quick test
|
|
Back in GCP land, the details page for the `sheets-url-shortener` Cloud Run service shows a gross-looking URL near the top: `https://sheets-url-shortener-vrw7x6wdzq-uc.a.run.app`. That doesn't do much for *shortening* my links, but it'll do just fine for a quick test. First, I pointed my browser straight to that listed URL:
|
|
data:image/s3,"s3://crabby-images/08ed4/08ed498256669b11fa0817bc59396970997e4131" alt="Testing the web server"
|
|
|
|
This at least tells me that the web server portion is working. Now to see if I can redirect to my [project car posts on Polywork](https://john.bowdre.net/?badges%5B%5D=Car+Nerd):
|
|
data:image/s3,"s3://crabby-images/6b8b2/6b8b26401605170b98f5ec6ea5e665c1d19eb5f5" alt="Testing a redirect"
|
|
|
|
Hmm, not quite. Luckily the error tells me exactly what I need to do...
|
|
|
|
### Enable Sheets API
|
|
I just needed to visit `https://console.developers.google.com/apis/api/sheets.googleapis.com/overview?project=############` to enable the Google Sheets API.
|
|
data:image/s3,"s3://crabby-images/49207/49207f0cfcfaebe3b19c4ded801c4cf3249f35fd" alt="Enabling Sheets API"
|
|
|
|
Once that's done, I can try my redirect again - and, after a brief moment, it successfully sends me on to Polywork!
|
|
data:image/s3,"s3://crabby-images/5db4a/5db4a4cf32df5aae6d0289ea433d4d6cc7ea1356" alt="Successful redirect"
|
|
|
|
### Link custom domain
|
|
The whole point of this project is to *shorten* URLs, but I haven't done that yet. I'll want to link in my `go.bowdre.net` domain to use that in place of the rather unwieldy `https://sheets-url-shortener-vrw7x6wdzq-uc.a.run.app`. I do that by going back to the [Cloud Run console](https://console.cloud.google.com/run) and selecting the option at the top to **Manage Custom Domains**.
|
|
data:image/s3,"s3://crabby-images/b665b/b665bd44bd5051b90addf8ad796d234028bf1773" alt="Manage custom domains"
|
|
|
|
I can then use the **Add Mapping** button, select my `sheets-url-shortener` service, choose one of my verified domains (which I *think* are already verified since they're registered through Google Domains with the same account), and then specify the desired subdomain.
|
|
data:image/s3,"s3://crabby-images/097fd/097fd977fe1e743561c443c4691b1684d12407c3" alt="Adding a domain mapping"
|
|
|
|
The wizard then tells me exactly what record I need to create/update with my domain host:
|
|
data:image/s3,"s3://crabby-images/6b1e2/6b1e264f6d7ab4785649cebb1e0e5ff04f83ef87" alt="CNAME details"
|
|
|
|
It took a while for the domain mapping to go live once I've updated the record.
|
|
data:image/s3,"s3://crabby-images/63413/6341360022cc475425e4a3523303120baa6fb424" alt="Processing mapping..."
|
|
|
|
### Final tests
|
|
Once it did finally update, I was able to hit `https://go.bowdre.net` to get the error/landing page, complete with a valid SSL cert:
|
|
data:image/s3,"s3://crabby-images/7756a/7756a9b26b01316a5877d0e50c87844b0e6b0590" alt="Successful error!"
|
|
|
|
And testing [go.bowdre.net/ghia](https://go.bowdre.net/ghia) works as well!
|
|
|
|
### Outro
|
|
I'm very pleased with how this quick little project turned out. Managing my shortened links with a Google Sheet is quite convenient, and I really like the complete lack of tracking or analytics. Plus I'm a sucker for an excuse to use a cloud technology I haven't played a lot with yet.
|
|
|
|
And now I can hand out handy-dandy short links!
|
|
|
|
| Link | Description|
|
|
| --- | --- |
|
|
| [go.bowdre.net/ghia](https://go.bowdre.net/ghia) | 1974 VW Karmann Ghia project |
|
|
| [go.bowdre.net/conedoge](https://go.bowdre.net/conedoge) | 2014 Subaru BRZ autocross videos |
|
|
| [go.bowdre.net/matrix](https://go.bowdre.net/matrix) | Chat with me on Matrix |
|
|
| [go.bowdre.net/twits](https://go.bowdre.net/twits) | Follow me on Twitter |
|
|
| [go.bowdre.net/stadia](https://go.bowdre.net/stadia) | Game with me on Stadia |
|
|
| [go.bowdre.net/shorterer](https://go.bowdre.net/shorterer) | This post! |
|
|
|