Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Match environment input to mirror Gitea's #6

Merged
merged 5 commits into from
Feb 27, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,2 +1,14 @@
# bids-hook
Tiny CI server to run bids-validator using Gitea webhooks


## Deployment

Results are placed in `%(GITEA_CUSTOM)/public/bids-validator/`;
this folder needs to be writable (and ideally created first and owned)
by the user running this daemon.

It assumes the URL `%(ROOT_URL)s/static/assets/` loads from
Gitea's `%(GITEA_CUSTOM)/public/`; it is **not** compatible
with configuring Gitea's `%(STATIC_URL_PREFIX)` so that
static files are hosted on a different server or CDN.
76 changes: 27 additions & 49 deletions bids-hook.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,6 @@ import (
"net/url"
"os"
"os/exec"
"path"
"path/filepath"
"regexp"
"strconv"
Expand All @@ -43,27 +42,21 @@ var (
// this should be entered as-in in Gitea to configure the webhook
bidsHookSecret []byte

// the base URL to reach Gitea's API
// read from environment variable GITEA_API_URL
// should end with "/api/v1"
giteaApiUrl *url.URL
// the base URL to reach Gitea
// read from environment variable GITEA_ROOT_URL
// should match Gitea's app.ini's [server].ROOT_URL
giteaRootUrl *url.URL

// secret used to authenticate api calls from bids-hook to Gitea
// read from environment variable GITEA_API_SECRET
// read from environment variable GITEA_TOKEN
// can be generated from a gitea admin account under "Settings" -> "Applications"
giteaApiSecret []byte
giteaToken []byte

// the base URL for writing links to Gitea's static assets
// read from environment variable GITEA_PUBLIC_URL
// should end with "/assets"
giteaPublicUrl *url.URL

// the path to Gitea's static assets directory
// read from environment variable GITEA_PUBLIC_PATH
// the path to Gitea's custom/ directory
// read from environment variable GITEA_CUSTOM
// used to save job result pages
// it should already exist
// should end with "/custom/public"
giteaPublicPath string
// see https://docs.gitea.io/en-us/config-cheat-sheet/#default-configuration-non-appini-configuration
giteaCustom string

// executable run by the worker for each accepted job
// read from environment variable WORKER_SCRIPT
Expand Down Expand Up @@ -286,15 +279,13 @@ type job struct {
// web link to the results page for this job
// see also j.resultPath()
func (j job) resultUrl() string {
url := *giteaPublicUrl
url.Path = path.Join(url.Path, fmt.Sprintf("%s.html", j.uuid))
return url.String()
return giteaRootUrl.JoinPath("assets", fmt.Sprintf("%s.html", j.uuid)).String()
}

// file path to the results page for this job
// see also j.resultUrl()
func (j job) resultPath() string {
return filepath.Join(giteaPublicPath, fmt.Sprintf("%s.html", j.uuid))
return filepath.Join(giteaCustom, "public", fmt.Sprintf("%s.html", j.uuid))
}

// file path to the log file for this job
Expand All @@ -305,8 +296,7 @@ func (j job) logPath() string {
// postStatus posts a commit status to Gitea
// 'state' should be one of the constants defined at the top of this module
func (j job) postStatus(ctx context.Context, state string) error {
url := *giteaApiUrl
url.Path = path.Join(url.Path, "repos", j.user, j.repo, "statuses", j.commit)
url := giteaRootUrl.JoinPath("api", "v1", "repos", j.user, j.repo, "statuses", j.commit)

var description, targetUrl string
switch state {
Expand Down Expand Up @@ -341,7 +331,7 @@ func (j job) postStatus(ctx context.Context, state string) error {
if err != nil {
return err
}
req.Header.Add("Authorization", fmt.Sprintf("token %s", giteaApiSecret))
req.Header.Add("Authorization", fmt.Sprintf("token %s", giteaToken))
req.Header.Add("Content-Type", "application/json")

resp, err := http.DefaultClient.Do(req)
Expand Down Expand Up @@ -451,44 +441,32 @@ func readConfig() {
}
bidsHookSecret = []byte(val)

val, ok = os.LookupEnv("GITEA_API_URL")
val, ok = os.LookupEnv("GITEA_ROOT_URL")
if !ok {
log.Fatal("missing environment variable GITEA_API_URL")
log.Fatal("missing environment variable GITEA_ROOT_URL")
}
giteaApiUrl, err = url.Parse(val)
giteaRootUrl, err = url.Parse(val)
if err != nil {
log.Fatalf("error parsing GITEA_API_URL: %v", err)
log.Fatalf("error parsing GITEA_ROOT_URL: %v", err)
}

val, ok = os.LookupEnv("GITEA_API_SECRET")
val, ok = os.LookupEnv("GITEA_TOKEN")
if !ok {
log.Fatal("missing environment variable GITEA_API_SECRET")
log.Fatal("missing environment variable GITEA_TOKEN")
}
giteaApiSecret = []byte(val)
giteaToken = []byte(val)

val, ok = os.LookupEnv("GITEA_PUBLIC_URL")
val, ok = os.LookupEnv("GITEA_CUSTOM")
if !ok {
log.Fatal("missing environment variable GITEA_PUBLIC_URL")
log.Fatal("missing environment variable GITEA_CUSTOM")
}
giteaPublicUrl, err = url.Parse(val)
giteaCustom, err = filepath.Abs(val)
if err != nil {
log.Fatalf("error parsing GITEA_PUBLIC_URL: %v", err)
}

val, ok = os.LookupEnv("GITEA_PUBLIC_PATH")
if !ok {
log.Fatal("missing environment variable GITEA_PUBLIC_PATH")
log.Fatalf("invalid GITEA_CUSTOM: %v", err)
}
giteaPublicPath, err = filepath.Abs(val)
err = os.MkdirAll(filepath.Join(giteaCustom, "public"), 0750)
if err != nil {
log.Fatalf("invalid GITEA_PUBLIC_PATH: %v", err)
}
info, err = os.Stat(giteaPublicPath)
if err != nil {
log.Fatalf("error opening GITEA_PUBLIC_PATH: %v", err)
}
if !info.IsDir() {
log.Fatal("GITEA_PUBLIC_PATH is not a directory")
log.Fatalf("error creating output folder: %v", err)
}

val, ok = os.LookupEnv("WORKER_SCRIPT")
Expand Down
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
module github.com/neuropoly/bids-hook

go 1.18
go 1.19
mguaypaq marked this conversation as resolved.
Show resolved Hide resolved
22 changes: 17 additions & 5 deletions start
Original file line number Diff line number Diff line change
@@ -1,14 +1,26 @@
#!/bin/bash

set -e

# this replicates the default location logic from https://docs.gitea.io/en-us/config-cheat-sheet/
# any setting can be overridden just by setting its variable before calling this script
: ${GITEA_APP_PATH:=../gitea/gitea}
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is awkward!

In our deployments, this will be /srv/gitea/gitea. On my dev machine I have ~/src/gitea/ and ~/src/bids-hook.

I don't know. Maybe this should be a mandatory input? But then again, if you on a server where you're explicitly setting GITEA_CUSTOM for gitea, you can also set it for bids-hook and then GITEA_APP_PATH is irrelevant.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I realize now that I didn't mention this before, but: the start script is meant to disappear from the repo at some point! It's just scaffolding for manual, local testing on my own machine (or yours!) while we develop a v1. Hence all the hardcoded values and dummy secret and token.

So actual deployment situations are out of scope for this file.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That makes sense. In that case, the part that mirrors Gitea's defaults logic should probably make its way into Go instead of shell. I could probably copy it straight out of their codebase.

And the environment variables that remain would migrate to a systemd .service file? Is it weird to put credentials in a .service file?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It looks like systemd has a whole system in place for credentials. According to System and Service Credentials:

Within unit files, there are four settings to configure service credentials.

  1. LoadCredential= may be used to load a credential from disk, from an AF_UNIX socket, or propagate them from a system credential.
  2. SetCredential= may be used to set a credential to a literal string encoded in the unit file. Because unit files are world-readable (both on disk and via D-Bus), this should only be used for credentials that aren’t sensitive, i.e. public keys/certificates – but not private keys.
  3. LoadCredentialEncrypted= is similar to LoadCredential= but will load an encrypted credential, and decrypt it before passing it to the service. For details on credential encryption, see below.
  4. SetCredentialEncrypted= is similar to SetCredential= but expects an encrypted credential to be specified literally. Unlike SetCredential= it is thus safe to be used even for sensitive information, because even though unit files are world readable, the ciphertext included in them cannot be decoded unless access to TPM2/encryption key is available.

So, systemd wants to make credentials available as files, and currently bids-hook wants to accept credentials as environment variables. I guess I can change the environment variable to contain a path to the secret, rather than the secret itself, and that should be sufficiently flexible for manual testing and also for systemd-controlled execution.

: ${GITEA_WORK_DIR:="$(dirname "$GITEA_APP_PATH")"}

: ${GITEA_CUSTOM:="$GITEA_WORK_DIR/custom"}

: ${GITEA_APP_DATA_PATH:="$GITEA_WORK_DIR/data"}
: ${GITEA_REPOSITORY_ROOT:="$GITEA_APP_DATA_PATH/gitea-repositories"}

export GITEA_REPOSITORY_ROOT
export GITEA_CUSTOM

# 127.0.0.1 is localhost, and 2845 is 0xB1D
export BIDS_HOOK_URL='http://127.0.0.1:2845/bids-hook'
export BIDS_HOOK_SECRET='blabla'

export GITEA_API_URL='http://127.0.0.1:3000/api/v1'
export GITEA_API_SECRET='69e45fa9cfa75a7497633c6be8dd2347226e2f62'

export GITEA_PUBLIC_URL='http://127.0.0.1:3000/assets'
export GITEA_PUBLIC_PATH='./custom/public'
export GITEA_ROOT_URL='http://127.0.0.1:3000'
export GITEA_TOKEN='69e45fa9cfa75a7497633c6be8dd2347226e2f62'

export WORKER_SCRIPT='./worker'
export WORKER_LOG_PATH='./log'
Expand Down