Skip to content

Commit d9d5b98

Browse files
committed
Merge remote-tracking branch 'upstream/main'
2 parents 422634e + 43d714d commit d9d5b98

30 files changed

+1778
-598
lines changed

.github/workflows/docker.yml

+41-41
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,12 @@ name: Docker
33
on:
44
push:
55
# push events will publish a new image, so only trigger on main branch or semver tags.
6-
branches: [ 'main' ]
7-
tags: [ 'v*' ]
6+
branches: ["main"]
7+
tags: ["v*"]
88
pull_request:
99
# Run the workflow on pull_request events to ensure we can still build the image.
1010
# We only publish the image on push events (see if statements in steps below).
11-
branches: [ 'main' ]
11+
branches: ["main"]
1212

1313
env:
1414
REGISTRY: ghcr.io
@@ -27,46 +27,46 @@ jobs:
2727
id-token: write
2828

2929
steps:
30-
- uses: actions/checkout@v3
30+
- uses: actions/checkout@v4
3131

32-
- name: Setup Docker buildx
33-
uses: docker/setup-buildx-action@8c0edbc76e98fa90f69d9a2c020dcb50019dc325 # v2.2.1
34-
with:
35-
# use buildx v0.9.1 (https://community.fly.io/t/10171/19)
36-
version: v0.9.1
32+
- name: Setup Docker buildx
33+
uses: docker/setup-buildx-action@f95db51fddba0c2d1ec667646a06c2ce06100226 # v3.0.0
3734

38-
- name: Log into registry ${{ env.REGISTRY }}
39-
uses: docker/login-action@f4ef78c080cd8ba55a85445d5b36e214a81df20a # v2.1.0
40-
if: github.event_name == 'push'
41-
with:
42-
registry: ${{ env.REGISTRY }}
43-
username: ${{ github.actor }}
44-
password: ${{ secrets.GITHUB_TOKEN }}
35+
- name: Log into registry ${{ env.REGISTRY }}
36+
uses: docker/login-action@343f7c4344506bcbf9b4de18042ae17996df046d # v3.0.0
37+
if: github.event_name == 'push'
38+
with:
39+
registry: ${{ env.REGISTRY }}
40+
username: ${{ github.actor }}
41+
password: ${{ secrets.GITHUB_TOKEN }}
4542

46-
- name: Extract Docker metadata
47-
id: meta
48-
uses: docker/metadata-action@57396166ad8aefe6098280995947635806a0e6ea # v4.1.1
49-
with:
50-
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
43+
- name: Extract Docker metadata
44+
id: meta
45+
uses: docker/metadata-action@96383f45573cb7f253c731d3b3ab81c87ef81934 # v5.0.0
46+
with:
47+
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
48+
tags: |
49+
type=ref,event=branch
50+
type=semver,pattern={{version}}
51+
type=semver,pattern={{major}}.{{minor}}
52+
type=semver,pattern={{major}}
5153
52-
- name: Build and push Docker image
53-
id: build-and-push
54-
uses: docker/build-push-action@c56af957549030174b10d6867f20e78cfd7debc5 # v3.2.0
55-
with:
56-
context: .
57-
push: ${{ github.event_name == 'push' }}
58-
tags: ${{ steps.meta.outputs.tags }}
59-
labels: ${{ steps.meta.outputs.labels }}
60-
platforms: linux/amd64,linux/arm64,linux/arm/v7
61-
cache-from: type=gha
62-
cache-to: type=gha,mode=max
54+
- name: Build and push Docker image
55+
id: build-and-push
56+
uses: docker/build-push-action@0565240e2d4ab88bba5387d719585280857ece09 # v5.0.0
57+
with:
58+
context: .
59+
push: ${{ github.event_name == 'push' }}
60+
tags: ${{ steps.meta.outputs.tags }}
61+
labels: ${{ steps.meta.outputs.labels }}
62+
platforms: linux/amd64,linux/arm64,linux/arm/v7
63+
cache-from: type=gha
64+
cache-to: type=gha,mode=max
6365

64-
# Sign the Docker image
65-
- name: Install cosign
66-
if: github.event_name == 'push'
67-
uses: sigstore/cosign-installer@9becc617647dfa20ae7b1151972e9b3a2c338a2b #v2.8.1
68-
- name: Sign the published Docker image
69-
if: github.event_name == 'push'
70-
env:
71-
COSIGN_EXPERIMENTAL: "true"
72-
run: cosign sign ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}@${{ steps.build-and-push.outputs.digest }}
66+
# Sign the Docker image
67+
- name: Install cosign
68+
if: github.event_name == 'push'
69+
uses: sigstore/cosign-installer@1fc5bd396d372bee37d608f955b336615edf79c8 #v3.2.0
70+
- name: Sign the published Docker image
71+
if: github.event_name == 'push'
72+
run: cosign sign --yes ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}@${{ steps.build-and-push.outputs.digest }}

.github/workflows/tests.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ on:
66
- main
77
pull_request:
88
branches:
9-
- '*'
10-
- 'release-branch/*'
9+
- "*"
10+
- "release-branch/*"
1111

1212
concurrency:
1313
group: ${{ github.workflow }}-$${{ github.head_ref || github.run_id }}

Dockerfile

+10-8
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,23 @@
1-
FROM --platform=$BUILDPLATFORM cgr.dev/chainguard/wolfi-base as build
2-
RUN apk update && apk add build-base git openssh go-1.20
1+
FROM --platform=$BUILDPLATFORM golang:1.23-alpine as build
32

43
WORKDIR /work
54

5+
# Install git so that go build populates the VCS details in build info, which
6+
# is then reported to Tailscale in the node version string.
7+
RUN apk add git
8+
69
COPY go.mod go.sum ./
710
RUN go mod download
811

912
COPY . .
1013
ARG TARGETOS TARGETARCH TARGETVARIANT
1114
RUN \
12-
if [ "${TARGETARCH}" = "arm" ] && [ -n "${TARGETVARIANT}" ]; then \
13-
export GOARM="${TARGETVARIANT#v}"; \
14-
fi; \
15-
GOOS=${TARGETOS} GOARCH=${TARGETARCH} CGO_ENABLED=0 go build -v ./cmd/golink
16-
15+
if [ "${TARGETARCH}" = "arm" ] && [ -n "${TARGETVARIANT}" ]; then \
16+
export GOARM="${TARGETVARIANT#v}"; \
17+
fi; \
18+
GOOS=${TARGETOS} GOARCH=${TARGETARCH} CGO_ENABLED=0 go build -v ./cmd/golink
1719

18-
FROM cgr.dev/chainguard/static:latest
20+
FROM gcr.io/distroless/static-debian12:nonroot
1921

2022
ENV HOME /home/nonroot
2123

README.md

+107
Original file line numberDiff line numberDiff line change
@@ -121,6 +121,100 @@ destination="/home/nonroot"
121121

122122
</details>
123123

124+
<details>
125+
<summary>Deploy on Modal</summary>
126+
127+
See the [Modal docs](https://modal.com/docs/guide/managing-deployments) for full instructions on long-lived deployments.
128+
129+
Create a `golinks.py` file:
130+
131+
```python
132+
import subprocess
133+
134+
import modal
135+
136+
app = modal.App(name="golinks")
137+
138+
vol = modal.Volume.from_name("golinks-data", create_if_missing=True)
139+
140+
image = modal.Image.from_registry(
141+
"golang:1.23.0-bookworm",
142+
add_python="3.10",
143+
).run_commands(["go install -v github.com/tailscale/golink/cmd/golink@latest"])
144+
145+
@app.cls(
146+
image=image,
147+
secrets=[modal.Secret.from_name("golinks")],
148+
volumes={"/root/.config": vol},
149+
keep_warm=1,
150+
concurrency_limit=1,
151+
)
152+
class Golinks:
153+
@modal.enter()
154+
def start_golinks(self):
155+
subprocess.Popen(
156+
[
157+
"golink",
158+
"-verbose",
159+
"--sqlitedb",
160+
"/root/.config/golink.db",
161+
]
162+
)
163+
```
164+
165+
Then create your secret and deploy with the [Modal CLI](https://github.com/modal-labs/modal-client):
166+
167+
```sh
168+
$ modal secret create golinks TS_AUTHKEY=<key>
169+
$ modal deploy golinks.py
170+
```
171+
172+
</details>
173+
174+
## Permissions
175+
176+
By default, users own the links they create and only they can update or delete those links.
177+
Ownership can be transferred to another user from the link edit page.
178+
Links whose owner is no longer part of the tailnet can be edited by any user,
179+
at which point that user will become the new owner.
180+
181+
Users can be granted admin access to edit all links using [ACL grants] in your tailnet policy file.
182+
For example, if you have your golink instance tagged with `tag:golink` and a user group named `group:golink-admins`,
183+
you can grant them admin access using:
184+
185+
```json
186+
{
187+
"grants": [{
188+
"src": ["group:golink-admins"],
189+
"dst": ["tag:golink"],
190+
"app": {
191+
"tailscale.com/cap/golink": [{
192+
"admin": true
193+
}]
194+
}
195+
}]
196+
}
197+
```
198+
199+
Or if you want to effectively disable the ownership model and allow everyone in your tailnet to edit all links,
200+
you could assign the grant to `autogroup:member`:
201+
202+
```json
203+
{
204+
"grants": [{
205+
"src": ["autogroup:member"],
206+
"dst": ["tag:golink"],
207+
"app": {
208+
"tailscale.com/cap/golink": [{
209+
"admin": true
210+
}]
211+
}
212+
}]
213+
}
214+
```
215+
216+
[ACL grants]: https://tailscale.com/kb/1324/acl-grants
217+
124218
## Backups
125219

126220
Once you have golink running, you can backup all of your links in [JSON lines] format from <http://go/.export>.
@@ -146,3 +240,16 @@ If you're using Firefox, you might want to configure two options to make it easy
146240
with a value of _true_
147241

148242
* if you use HTTPS-Only Mode, [add an exception](https://support.mozilla.org/en-US/kb/https-only-prefs#w_add-exceptions-for-http-websites-when-youre-in-https-only-mode)
243+
244+
## HTTPS
245+
246+
When golink joins your tailnet it will check to see if HTTPS is enabled and
247+
begin serving HTTPS traffic it detects that it is. When HTTPS is enabled golink
248+
will redirect all requests received by the HTTP endpoint first to their internal
249+
HTTPS equivalent before redirecting to the external link destination.
250+
251+
**NB:** If you use `curl` to interact with the API of a golink instance with HTTPS
252+
enabled over its HTTP interface you _must_ specify the `-L` flag to follow these
253+
redirects or else your request will terminate early with an empty response. We
254+
recommend the use of the `-L` flag in all deployments regardless of current
255+
HTTPS status to avoid accidental outages should it be enabled in the future.

convex.go

+10
Original file line numberDiff line numberDiff line change
@@ -171,6 +171,11 @@ func (c *ConvexDB) Save(link *Link) error {
171171
return c.mutation(&args)
172172
}
173173

174+
func (c *ConvexDB) Delete(short string) error {
175+
args := UdfExecution{"deleteLink", map[string]interface{}{"normalizedId": linkID(short)}, "json"}
176+
return c.mutation(&args)
177+
}
178+
174179
func (c *ConvexDB) LoadStats() (ClickStats, error) {
175180
args := UdfExecution{"stats:loadStats", map[string]interface{}{}, "json"}
176181
response, err := c.query(&args)
@@ -203,3 +208,8 @@ func (c *ConvexDB) SaveStats(stats ClickStats) error {
203208
args := UdfExecution{"stats:saveStats", map[string]interface{}{"stats": mungedStats}, "json"}
204209
return c.mutation(&args)
205210
}
211+
212+
func (c *ConvexDB) DeleteStats(short string) error {
213+
args := UdfExecution{"stats:deleteStats", map[string]interface{}{"normalizedId": linkID(short)}, "json"}
214+
return c.mutation(&args)
215+
}

convex_test.go

+40-9
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ func getDbUrl() string {
3030
}
3131

3232
// Test saving and loading links for SQLiteDB
33-
func Test_Convex_SaveLoadLinks(t *testing.T) {
33+
func Test_Convex_SaveLoadDeleteLinks(t *testing.T) {
3434
url := getDbUrl()
3535
db := NewConvexDB(url, "test")
3636
clear(db)
@@ -66,35 +66,51 @@ func Test_Convex_SaveLoadLinks(t *testing.T) {
6666
if !cmp.Equal(got, links, sortLinks, approximateTimeEq) {
6767
t.Errorf("db.LoadAll got %v, want %v", got, links)
6868
}
69+
70+
for _, link := range links {
71+
if err := db.Delete(link.Short); err != nil {
72+
t.Error(err)
73+
}
74+
}
75+
76+
got, err = db.LoadAll()
77+
if err != nil {
78+
t.Error(err)
79+
}
80+
want := []*Link(nil)
81+
if !cmp.Equal(got, want) {
82+
t.Errorf("db.LoadAll got %v, want %v", got, want)
83+
}
6984
}
7085

7186
// Test saving and loading stats for SQLiteDB
72-
func Test_Convex_SaveLoadStats(t *testing.T) {
87+
func Test_Convex_SaveLoadDeleteStats(t *testing.T) {
7388
url := getDbUrl()
7489
db := NewConvexDB(url, "test")
7590
clear(db)
76-
defer clear(db)
91+
// defer clear(db)
7792

7893
// preload some links
7994
links := []*Link{
8095
{Short: "a"},
81-
{Short: "b"},
96+
{Short: "B-c"},
8297
}
8398
for _, link := range links {
8499
if err := db.Save(link); err != nil {
85100
t.Error(err)
86101
}
87102
}
88103

89-
// stats to record and then retrieve
104+
// Stats to store do not need to be their canonical short name,
105+
// but returned stats always should be.
90106
stats := []ClickStats{
91107
{"a": 1},
92-
{"b": 1},
93-
{"a": 1, "b": 2},
108+
{"b-c": 1},
109+
{"a": 1, "bc": 2},
94110
}
95111
want := ClickStats{
96-
"a": 2,
97-
"b": 3,
112+
"a": 2,
113+
"B-c": 3,
98114
}
99115

100116
for _, s := range stats {
@@ -110,4 +126,19 @@ func Test_Convex_SaveLoadStats(t *testing.T) {
110126
if !cmp.Equal(got, want) {
111127
t.Errorf("db.LoadStats got %v, want %v", got, want)
112128
}
129+
130+
for k := range want {
131+
if err := db.DeleteStats(k); err != nil {
132+
t.Error(err)
133+
}
134+
}
135+
136+
got, err = db.LoadStats()
137+
if err != nil {
138+
t.Error(err)
139+
}
140+
want = ClickStats{}
141+
if !cmp.Equal(got, want) {
142+
t.Errorf("db.LoadStats got %v, want %v", got, want)
143+
}
113144
}

db.go

+2
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,8 @@ type Database interface {
3434
LoadAll() ([]*Link, error)
3535
Load(short string) (*Link, error)
3636
Save(link *Link) error
37+
Delete(short string) error
3738
LoadStats() (ClickStats, error)
3839
SaveStats(stats ClickStats) error
40+
DeleteStats(short string) error
3941
}

0 commit comments

Comments
 (0)