Compare commits

...

3 Commits

Author SHA1 Message Date
6622ea0e1c
Improve formatting of durations
All checks were successful
Alpine 3.20 Success
Since "m" could stand for both "minute" and "month",
and months vary in length, let's stop at days.
2025-01-02 00:36:03 +01:00
a492b3b668
Clean up 2024-12-28 00:27:46 +01:00
280114a5d3
Unify our usage of the local shell 2024-12-27 02:16:14 +01:00
2 changed files with 39 additions and 28 deletions

50
acid.go
View File

@ -30,7 +30,6 @@ import (
"syscall" "syscall"
ttemplate "text/template" ttemplate "text/template"
"time" "time"
"unicode"
_ "github.com/mattn/go-sqlite3" _ "github.com/mattn/go-sqlite3"
"github.com/pkg/sftp" "github.com/pkg/sftp"
@ -154,6 +153,14 @@ var shellFuncs = ttemplate.FuncMap{
// --- Utilities --------------------------------------------------------------- // --- Utilities ---------------------------------------------------------------
func localShell() string {
if shell := os.Getenv("SHELL"); shell != "" {
return shell
}
// The os/user package doesn't store the parsed out shell field.
return "/bin/sh"
}
func giteaSign(b []byte) string { func giteaSign(b []byte) string {
payloadHmac := hmac.New(sha256.New, []byte(getConfig().Secret)) payloadHmac := hmac.New(sha256.New, []byte(getConfig().Secret))
payloadHmac.Write(b) payloadHmac.Write(b)
@ -306,9 +313,7 @@ function get(id) {
return document.getElementById(id) return document.getElementById(id)
} }
function getLog(id) { function getLog(id) {
const header = document.getElementById(id) const header = get(id), log = get(id + 'log'), text = log.textContent
const log = document.getElementById(id + 'log')
const text = log.textContent
// lines[-1] is an implementation detail of terminalWriter.Serialize, // lines[-1] is an implementation detail of terminalWriter.Serialize,
// lines[-2] is the actual last line. // lines[-2] is the actual last line.
const last = Math.max(0, text.split('\n').length - 2) const last = Math.max(0, text.split('\n').length - 2)
@ -326,8 +331,7 @@ function refreshLog(log, top, changed) {
log.log.hidden = empty log.log.hidden = empty
} }
let refresher = setInterval(() => { let refresher = setInterval(() => {
let run = getLog('run'), task = getLog('task'), deploy = getLog('deploy') const run = getLog('run'), task = getLog('task'), deploy = getLog('deploy')
const url = new URL(window.location.href) const url = new URL(window.location.href)
url.search = '' url.search = ''
url.searchParams.set('json', '') url.searchParams.set('json', '')
@ -366,8 +370,8 @@ let refresher = setInterval(() => {
if (!data.IsRunning) if (!data.IsRunning)
clearInterval(refresher) clearInterval(refresher)
}).catch(error => { }).catch(error => {
alert(error)
clearInterval(refresher) clearInterval(refresher)
alert(error)
}) })
}, 1000 /* For faster updates than this, we should use WebSockets. */) }, 1000 /* For faster updates than this, we should use WebSockets. */)
</script> </script>
@ -842,7 +846,7 @@ func notifierRunCommand(ctx context.Context, task Task) {
return return
} }
cmd := exec.CommandContext(ctx, "sh") cmd := exec.CommandContext(ctx, localShell())
cmd.Stdin = script cmd.Stdin = script
cmd.Stdout = os.Stdout cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr cmd.Stderr = os.Stderr
@ -1166,14 +1170,6 @@ func executorDownload(client *ssh.Client, remoteRoot, localRoot string) error {
return nil return nil
} }
func executorLocalShell() string {
if shell := os.Getenv("SHELL"); shell != "" {
return shell
}
// The os/user package doesn't store the parsed out shell field.
return "/bin/sh"
}
func executorTmpDir(fallback string) string { func executorTmpDir(fallback string) string {
// See also: https://systemd.io/TEMPORARY_DIRECTORIES/ // See also: https://systemd.io/TEMPORARY_DIRECTORIES/
if tmp := os.Getenv("TMPDIR"); tmp != "" { if tmp := os.Getenv("TMPDIR"); tmp != "" {
@ -1209,9 +1205,10 @@ func executorDeploy(
return err return err
} }
cmd := exec.CommandContext(ctx, executorLocalShell(), "-c", script.String()) cmd := exec.CommandContext(ctx, localShell())
cmd.Env = rt.localEnv() cmd.Env = rt.localEnv()
cmd.Dir = dir cmd.Dir = dir
cmd.Stdin = script
cmd.Stdout = &rt.DeployLog cmd.Stdout = &rt.DeployLog
cmd.Stderr = &rt.DeployLog cmd.Stderr = &rt.DeployLog
return cmd.Run() return cmd.Run()
@ -1605,18 +1602,15 @@ func (t *Task) CloneURL() string {
} }
func shortDurationString(d time.Duration) string { func shortDurationString(d time.Duration) string {
rs := []rune(d.Truncate(time.Second).String()) if d.Abs() >= 24*time.Hour {
for i, r := range rs { return strconv.FormatInt(int64(d/time.Hour/24), 10) + "d"
if !unicode.IsLetter(r) { } else if d.Abs() >= time.Hour {
continue return strconv.FormatInt(int64(d/time.Hour), 10) + "h"
} } else if d.Abs() >= time.Minute {
i++ return strconv.FormatInt(int64(d/time.Minute), 10) + "m"
for i < len(rs) && unicode.IsLetter(rs[i]) { } else {
i++ return strconv.FormatInt(int64(d/time.Second), 10) + "s"
}
return string(rs[:i])
} }
return string(rs)
} }
func (t *Task) Created() *time.Time { func (t *Task) Created() *time.Time {

View File

@ -4,6 +4,7 @@ import (
"bytes" "bytes"
"testing" "testing"
ttemplate "text/template" ttemplate "text/template"
"time"
) )
func TestTemplateQuote(t *testing.T) { func TestTemplateQuote(t *testing.T) {
@ -30,3 +31,19 @@ func TestTemplateQuote(t *testing.T) {
} }
} }
} }
func TestShortDurationString(t *testing.T) {
for _, test := range []struct {
d time.Duration
expect string
}{
{72 * time.Hour, "3d"},
{-3 * time.Hour, "-3h"},
{12 * time.Minute, "12m"},
{time.Millisecond, "0s"},
} {
if sd := shortDurationString(test.d); sd != test.expect {
t.Errorf("%s = %s; want %s\n", test.d, sd, test.expect)
}
}
}