feat(gts): add background image for GTS

Add a background image for the custom CSS for GoToSocial.
This commit is contained in:
Dan Anglin 2023-07-29 01:05:33 +01:00
parent 94db6cbe19
commit a4819c61c8
Signed by: dananglin
GPG key ID: 0C1D44CFBEE68638
9 changed files with 395 additions and 334 deletions

BIN
assets/gotosocial/images/background.jpg (Stored with Git LFS) Normal file

Binary file not shown.

31
magefiles/clean.go Normal file
View file

@ -0,0 +1,31 @@
//go:build mage
package main
import (
"os"
"github.com/magefile/mage/sh"
)
// Clean cleans the workspace.
func Clean() error {
buildDir := "./build"
objects, err := os.ReadDir(buildDir)
if err != nil {
return err
}
for i := range objects {
name := objects[i].Name()
if name != ".gitkeep" {
if err := sh.Rm(buildDir + "/" + name); err != nil {
return err
}
}
}
return nil
}

View file

@ -13,7 +13,7 @@ import (
// Deploy deploys the services to the Flow Platform.
func Deploy(name string) error {
mg.Deps(
mg.F(Render, name),
mg.F(Prepare, name),
)
cfg, err := newConfig(configFile)

280
magefiles/download.go Normal file
View file

@ -0,0 +1,280 @@
//go:build mage
package main
import (
"encoding/json"
"fmt"
"io"
"net/http"
"os"
"path/filepath"
"github.com/magefile/mage/sh"
)
const (
configFile string = "./config/services.json"
rootBuildDir string = "./build"
templateExtension string = ".gotmpl"
rootTemplatesDir string = "./templates"
rootAssetsDir string = "./assets"
)
// Download downloads the binaries for a given service.
func Download(name string) error {
cfg, err := newConfig(configFile)
if err != nil {
return fmt.Errorf("unable to load the configuration; %v", err)
}
switch name {
case "forgejo":
if err := downloadForgejo(cfg.Forgejo.Version); err != nil {
return fmt.Errorf("an error occurred whilst getting the forgejo binary; %w", err)
}
case "gotosocial":
if err := downloadGoToSocial(cfg.GoToSocial.Version); err != nil {
return fmt.Errorf("an error occurred whilst getting the packages for GoToSocial; %w", err)
}
default:
fmt.Printf("'%s' has no files to download.\n", name)
}
return nil
}
// Download Forgejo
type forgejoDownload struct {
Downloads map[string]forgejoFiles `json:"downloads"`
}
type forgejoFiles struct {
Binary string `json:"binary"`
Signature string `json:"signature"`
Digest string `json:"digest"`
}
const (
forgejoBinaryFileFormat string = "forgejo-%s-linux-amd64"
forgejoDigestExtension string = ".sha256"
forgejoSignatureExtension string = ".asc"
forgejoDownloadJson string = "./magefiles/forgejo/download.json"
)
func downloadForgejo(version string) error {
downloadFolder := filepath.Join(rootBuildDir, "forgejo")
if err := os.MkdirAll(downloadFolder, 0o750); err != nil {
return fmt.Errorf("unable to make %s; %w", downloadFolder, err)
}
binaryPath := filepath.Join(
downloadFolder,
fmt.Sprintf(forgejoBinaryFileFormat, version),
)
signaturePath := binaryPath + forgejoSignatureExtension
digestPath := binaryPath + forgejoDigestExtension
_, err := os.Stat(binaryPath)
if err == nil {
fmt.Printf("Forgejo %s is already downloaded.\n", version)
return nil
}
data, err := newForgejoDownloadData()
if err != nil {
return err
}
downloads := []struct {
url string
path string
}{
{
url: data.Downloads[version].Binary,
path: binaryPath,
},
{
url: data.Downloads[version].Signature,
path: signaturePath,
},
{
url: data.Downloads[version].Digest,
path: digestPath,
},
}
for _, v := range downloads {
if err := func() error {
download, err := os.Create(v.path)
if err != nil {
return fmt.Errorf("unable to create %s; %w", v.path, err)
}
defer download.Close()
client := http.Client{
CheckRedirect: func(r *http.Request, _ []*http.Request) error {
r.URL.Opaque = r.URL.Path
return nil
},
}
resp, err := client.Get(v.url)
if err != nil {
return err
}
defer resp.Body.Close()
size, err := io.Copy(download, resp.Body)
if err != nil {
return err
}
fmt.Printf("Downloaded %s with size %d.\n", v.path, size)
return nil
}(); err != nil {
return err
}
}
if err = sh.Run("gpg", "--verify", signaturePath, binaryPath); err != nil {
return fmt.Errorf("GPG verification failed; %w", err)
}
if err := os.Chdir(downloadFolder); err != nil {
return err
}
if err := sh.Run("sha256sum", "--check", fmt.Sprintf(forgejoBinaryFileFormat+forgejoDigestExtension, version)); err != nil {
return err
}
if err := os.Chdir("../.."); err != nil {
return err
}
return nil
}
func newForgejoDownloadData() (forgejoDownload, error) {
var data forgejoDownload
f, err := os.Open(forgejoDownloadJson)
if err != nil {
return data, err
}
defer f.Close()
decoder := json.NewDecoder(f)
if err = decoder.Decode(&data); err != nil {
return data, err
}
return data, nil
}
// Download GTS
func downloadGoToSocial(version string) error {
downloadFolder := filepath.Join(rootBuildDir, "gotosocial")
if err := os.MkdirAll(downloadFolder, 0o750); err != nil {
return fmt.Errorf("unable to make %s; %w", downloadFolder, err)
}
binaryTarUrl := fmt.Sprintf(
"https://github.com/superseriousbusiness/gotosocial/releases/download/v%s/gotosocial_%s_linux_amd64.tar.gz",
version,
version,
)
binaryTarFilepath := filepath.Join(downloadFolder, fmt.Sprintf("gotosocial_%s_linux_amd64.tar.gz", version))
webAssetsTarUrl := fmt.Sprintf(
"https://github.com/superseriousbusiness/gotosocial/releases/download/v%s/gotosocial_%s_web-assets.tar.gz",
version,
version,
)
webAssetsFilepath := filepath.Join(downloadFolder, fmt.Sprintf("gotosocial_%s_web-assets.tar.gz", version))
checksumUrl := fmt.Sprintf(
"https://github.com/superseriousbusiness/gotosocial/releases/download/v%s/checksums.txt",
version,
)
checksumFilePath := filepath.Join(downloadFolder, fmt.Sprintf("gotosocial_%s_checksums.txt", version))
_, err := os.Stat(binaryTarFilepath)
if err == nil {
fmt.Printf("GoToSocial %s is already downloaded.\n", version)
return nil
}
downloads := []struct {
url string
path string
}{
{
url: binaryTarUrl,
path: binaryTarFilepath,
},
{
url: webAssetsTarUrl,
path: webAssetsFilepath,
},
{
url: checksumUrl,
path: checksumFilePath,
},
}
for _, v := range downloads {
if err := func() error {
download, err := os.Create(v.path)
if err != nil {
return fmt.Errorf("unable to create %s; %w", v.path, err)
}
defer download.Close()
client := http.Client{
CheckRedirect: func(r *http.Request, _ []*http.Request) error {
r.URL.Opaque = r.URL.Path
return nil
},
}
resp, err := client.Get(v.url)
if err != nil {
return err
}
defer resp.Body.Close()
size, err := io.Copy(download, resp.Body)
if err != nil {
return err
}
fmt.Printf("Downloaded %s with size %d.\n", v.path, size)
return nil
}(); err != nil {
return err
}
}
if err := os.Chdir(downloadFolder); err != nil {
return err
}
if err := sh.Run("sha256sum", "--check", "--ignore-missing", fmt.Sprintf("gotosocial_%s_checksums.txt", version)); err != nil {
return err
}
if err := os.Chdir("../.."); err != nil {
return err
}
return nil
}

View file

@ -1,145 +0,0 @@
//go:build mage
package main
import (
"encoding/json"
"fmt"
"io"
"net/http"
"os"
"path/filepath"
"github.com/magefile/mage/sh"
)
type forgejoDownload struct {
Downloads map[string]forgejoFiles `json:"downloads"`
}
type forgejoFiles struct {
Binary string `json:"binary"`
Signature string `json:"signature"`
Digest string `json:"digest"`
}
const (
forgejoBinaryFileFormat string = "forgejo-%s-linux-amd64"
forgejoDigestExtension string = ".sha256"
forgejoSignatureExtension string = ".asc"
forgejoDownloadJson string = "./magefiles/forgejo/download.json"
)
func downloadForgejo(version string) error {
downloadFolder := filepath.Join(rootBuildDir, "forgejo")
if err := os.MkdirAll(downloadFolder, 0o750); err != nil {
return fmt.Errorf("unable to make %s; %w", downloadFolder, err)
}
binaryPath := filepath.Join(
downloadFolder,
fmt.Sprintf(forgejoBinaryFileFormat, version),
)
signaturePath := binaryPath + forgejoSignatureExtension
digestPath := binaryPath + forgejoDigestExtension
_, err := os.Stat(binaryPath)
if err == nil {
fmt.Printf("Forgejo %s is already downloaded.\n", version)
return nil
}
data, err := newForgejoDownloadData()
if err != nil {
return err
}
downloads := []struct {
url string
path string
}{
{
url: data.Downloads[version].Binary,
path: binaryPath,
},
{
url: data.Downloads[version].Signature,
path: signaturePath,
},
{
url: data.Downloads[version].Digest,
path: digestPath,
},
}
for _, v := range downloads {
if err := func() error {
download, err := os.Create(v.path)
if err != nil {
return fmt.Errorf("unable to create %s; %w", v.path, err)
}
defer download.Close()
client := http.Client{
CheckRedirect: func(r *http.Request, _ []*http.Request) error {
r.URL.Opaque = r.URL.Path
return nil
},
}
resp, err := client.Get(v.url)
if err != nil {
return err
}
defer resp.Body.Close()
size, err := io.Copy(download, resp.Body)
if err != nil {
return err
}
fmt.Printf("Downloaded %s with size %d.\n", v.path, size)
return nil
}(); err != nil {
return err
}
}
if err = sh.Run("gpg", "--verify", signaturePath, binaryPath); err != nil {
return fmt.Errorf("GPG verification failed; %w", err)
}
if err := os.Chdir(downloadFolder); err != nil {
return err
}
if err := sh.Run("sha256sum", "--check", fmt.Sprintf(forgejoBinaryFileFormat+forgejoDigestExtension, version)); err != nil {
return err
}
if err := os.Chdir("../.."); err != nil {
return err
}
return nil
}
func newForgejoDownloadData() (forgejoDownload, error) {
var data forgejoDownload
f, err := os.Open(forgejoDownloadJson)
if err != nil {
return data, err
}
defer f.Close()
decoder := json.NewDecoder(f)
if err = decoder.Decode(&data); err != nil {
return data, err
}
return data, nil
}

View file

@ -1,113 +0,0 @@
//go:build mage
package main
import (
"fmt"
"io"
"net/http"
"os"
"path/filepath"
"github.com/magefile/mage/sh"
)
func downloadGoToSocial(version string) error {
downloadFolder := filepath.Join(rootBuildDir, "gotosocial")
if err := os.MkdirAll(downloadFolder, 0o750); err != nil {
return fmt.Errorf("unable to make %s; %w", downloadFolder, err)
}
binaryTarUrl := fmt.Sprintf(
"https://github.com/superseriousbusiness/gotosocial/releases/download/v%s/gotosocial_%s_linux_amd64.tar.gz",
version,
version,
)
binaryTarFilepath := filepath.Join(downloadFolder, fmt.Sprintf("gotosocial_%s_linux_amd64.tar.gz", version))
webAssetsTarUrl := fmt.Sprintf(
"https://github.com/superseriousbusiness/gotosocial/releases/download/v%s/gotosocial_%s_web-assets.tar.gz",
version,
version,
)
webAssetsFilepath := filepath.Join(downloadFolder, fmt.Sprintf("gotosocial_%s_web-assets.tar.gz", version))
checksumUrl := fmt.Sprintf(
"https://github.com/superseriousbusiness/gotosocial/releases/download/v%s/checksums.txt",
version,
)
checksumFilePath := filepath.Join(downloadFolder, fmt.Sprintf("gotosocial_%s_checksums.txt", version))
_, err := os.Stat(binaryTarFilepath)
if err == nil {
fmt.Printf("GoToSocial %s is already downloaded.\n", version)
return nil
}
downloads := []struct {
url string
path string
}{
{
url: binaryTarUrl,
path: binaryTarFilepath,
},
{
url: webAssetsTarUrl,
path: webAssetsFilepath,
},
{
url: checksumUrl,
path: checksumFilePath,
},
}
for _, v := range downloads {
if err := func() error {
download, err := os.Create(v.path)
if err != nil {
return fmt.Errorf("unable to create %s; %w", v.path, err)
}
defer download.Close()
client := http.Client{
CheckRedirect: func(r *http.Request, _ []*http.Request) error {
r.URL.Opaque = r.URL.Path
return nil
},
}
resp, err := client.Get(v.url)
if err != nil {
return err
}
defer resp.Body.Close()
size, err := io.Copy(download, resp.Body)
if err != nil {
return err
}
fmt.Printf("Downloaded %s with size %d.\n", v.path, size)
return nil
}(); err != nil {
return err
}
}
if err := os.Chdir(downloadFolder); err != nil {
return err
}
if err := sh.Run("sha256sum", "--check", "--ignore-missing", fmt.Sprintf("gotosocial_%s_checksums.txt", version)); err != nil {
return err
}
if err := os.Chdir("../.."); err != nil {
return err
}
return nil
}

View file

@ -1,62 +0,0 @@
//go:build mage
package main
import (
"fmt"
"os"
"github.com/magefile/mage/sh"
)
const (
configFile string = "./config/services.json"
rootBuildDir string = "./build"
templateExtension string = ".gotmpl"
rootTemplatesDir string = "./templates"
)
// Clean cleans the workspace.
func Clean() error {
buildDir := "./build"
objects, err := os.ReadDir(buildDir)
if err != nil {
return err
}
for i := range objects {
name := objects[i].Name()
if name != ".gitkeep" {
if err := sh.Rm(buildDir + "/" + name); err != nil {
return err
}
}
}
return nil
}
// Download downloads the binaries for a given service.
func Download(name string) error {
cfg, err := newConfig(configFile)
if err != nil {
return fmt.Errorf("unable to load the configuration; %v", err)
}
switch name {
case "forgejo":
if err := downloadForgejo(cfg.Forgejo.Version); err != nil {
return fmt.Errorf("an error occurred whilst getting the forgejo binary; %w", err)
}
case "gotosocial":
if err := downloadGoToSocial(cfg.GoToSocial.Version); err != nil {
return fmt.Errorf("an error occurred whilst getting the packages for GoToSocial; %w", err)
}
default:
fmt.Printf("'%s' has no files to download.\n", name)
}
return nil
}

View file

@ -4,6 +4,8 @@ package main
import (
"fmt"
"io"
"io/fs"
"log"
"os"
"path/filepath"
@ -13,14 +15,14 @@ import (
"github.com/magefile/mage/mg"
)
// Render renders the template files.
func Render(name string) error {
// Prepare prepares the service's build directory.
func Prepare(service string) error {
cfg, err := newConfig(configFile)
if err != nil {
return fmt.Errorf("unable to load the configuration; %v", err)
}
if name == "all" {
if service == "all" {
objects, err := os.ReadDir(rootTemplatesDir)
if err != nil {
return fmt.Errorf("unable to read the templates directory; %w", err)
@ -31,30 +33,39 @@ func Render(name string) error {
continue
}
dirName := o.Name()
service := o.Name()
if dirName != "compose" {
if service != "compose" {
mg.Deps(
mg.F(Download, dirName),
mg.F(Download, service),
)
}
log.Printf("Rendering templates for %s.\n", dirName)
if err := render(cfg, o.Name()); err != nil {
return fmt.Errorf("unable to render templates for %s; %w", dirName, err)
log.Printf("Rendering templates for %s.\n", service)
if err := render(cfg, service); err != nil {
return fmt.Errorf("unable to render templates for %s; %w", service, err)
}
log.Printf("Copying assets for %s.\n", service)
if err := copyAssets(service); err != nil {
return fmt.Errorf("unable to copy the assets for %s; %w", service, err)
}
}
} else {
if name != "compose" {
if service != "compose" {
mg.Deps(
mg.F(Download, name),
mg.F(Render, "compose"),
mg.F(Download, service),
mg.F(Prepare, "compose"),
)
}
if err := render(cfg, name); err != nil {
if err := render(cfg, service); err != nil {
return fmt.Errorf("an error occurred whilst rendering the templates; %w", err)
}
if err := copyAssets(service); err != nil {
return fmt.Errorf("unable to copy the assets for %s; %w", service, err)
}
}
return nil
@ -109,3 +120,58 @@ func render(cfg config, component string) error {
return nil
}
func copyAssets(service string) error {
assetsDirName := filepath.Join(rootAssetsDir, service)
if _, err := os.Stat(assetsDirName); err != nil {
log.Printf("There's no assets directory for %s.\n", service)
return nil
}
buildDirName := filepath.Join(rootBuildDir, service, "assets")
walkDirFunc := func(path string, d fs.DirEntry, err error) error {
if err != nil {
return err
}
if path == assetsDirName {
return nil
}
buildAssetPath := filepath.Join(buildDirName, strings.TrimPrefix(path, assetsDirName))
if d.IsDir() {
if err := os.MkdirAll(buildAssetPath, 0o750); err != nil {
return fmt.Errorf("unable to make %s; %w", path, err)
}
return nil
}
source, err := os.Open(path)
if err != nil {
return err
}
defer source.Close()
dest, err := os.Create(buildAssetPath)
if err != nil {
return err
}
defer dest.Close()
if _, err := io.Copy(dest, source); err != nil {
return err
}
return nil
}
err := filepath.WalkDir(assetsDirName, walkDirFunc)
if err != nil {
return fmt.Errorf("error walking the path '%s'; %w", assetsDirName, err)
}
return nil
}

View file

@ -17,6 +17,7 @@ RUN --mount=type=bind,source=.,target=/packages \
COPY --chown={{ .GoToSocial.LinuxUID }}:{{ .GoToSocial.LinuxUID }} entrypoint.sh /usr/local/bin/entrypoint
COPY --chown={{ .GoToSocial.LinuxUID }}:{{ .GoToSocial.LinuxUID }} config.yaml /flow/gts/config/config.yaml
COPY --chown={{ .GoToSocial.LinuxUID }}:{{ .GoToSocial.LinuxUID }} assets/images/background.jpg /flow/gts/web/assets/background.jpg
COPY --chown={{ .GoToSocial.LinuxUID }}:{{ .FlowGID }} traefik_gotosocial.yaml /flow/gts/tmp/traefik_gotosocial.yaml
RUN chmod a+x /usr/local/bin/entrypoint