1
0
mirror of https://github.com/hrfee/jfa-go.git synced 2025-01-01 05:50:12 +00:00

add external/internal data options

"make all" will build with internal data, whereas "make debug"/"make
all-external" will make an external "data/" directory.
This commit is contained in:
Harvey Tindall 2021-02-01 20:25:20 +00:00
parent e2a68809fe
commit 27530bff0a
Signed by: hrfee
GPG Key ID: BBC65952848FB1A2
8 changed files with 91 additions and 44 deletions

View File

@ -23,13 +23,13 @@ steps:
event: tag event: tag
--- ---
name: jfa-go-git name: jfa-go-1.16-git
kind: pipeline kind: pipeline
type: docker type: docker
steps: steps:
- name: build - name: build
image: golang:latest image: golang:1.16rc
commands: commands:
- apt update -y - apt update -y
- apt install build-essential python3-pip curl software-properties-common sed upx -y - apt install build-essential python3-pip curl software-properties-common sed upx -y

1
.gitignore vendored
View File

@ -4,6 +4,7 @@ dist/
build/ build/
data/ data/
version.go version.go
embed.go
notes notes
docs/* docs/*
lang/langtostruct.py lang/langtostruct.py

View File

@ -4,31 +4,31 @@ npm:
configuration: configuration:
$(info Fixing config-base) $(info Fixing config-base)
-mkdir -p build/data -mkdir -p data
python3 config/fixconfig.py -i config/config-base.json -o build/data/config-base.json python3 config/fixconfig.py -i config/config-base.json -o data/config-base.json
$(info Generating config-default.ini) $(info Generating config-default.ini)
python3 config/generate_ini.py -i config/config-base.json -o build/data/config-default.ini python3 config/generate_ini.py -i config/config-base.json -o data/config-default.ini
email: email:
$(info Generating email html) $(info Generating email html)
python3 mail/generate.py -o build/data/ python3 mail/generate.py -o data/
typescript: typescript:
$(info compiling typescript) $(info compiling typescript)
-mkdir -p build/data/web/js -mkdir -p data/web/js
-npx esbuild --bundle ts/admin.ts --outfile=./build/data/web/js/admin.js --minify -npx esbuild --bundle ts/admin.ts --outfile=./data/web/js/admin.js --minify
-npx esbuild --bundle ts/form.ts --outfile=./build/data/web/js/form.js --minify -npx esbuild --bundle ts/form.ts --outfile=./data/web/js/form.js --minify
-npx esbuild --bundle ts/setup.ts --outfile=./build/data/web/js/setup.js --minify -npx esbuild --bundle ts/setup.ts --outfile=./data/web/js/setup.js --minify
ts-debug: ts-debug:
$(info compiling typescript w/ sourcemaps) $(info compiling typescript w/ sourcemaps)
-mkdir -p build/data/web/js -mkdir -p data/web/js
-npx esbuild --bundle ts/admin.ts --sourcemap --outfile=./build/data/web/js/admin.js -npx esbuild --bundle ts/admin.ts --sourcemap --outfile=./data/web/js/admin.js
-npx esbuild --bundle ts/form.ts --sourcemap --outfile=./build/data/web/js/form.js -npx esbuild --bundle ts/form.ts --sourcemap --outfile=./data/web/js/form.js
-npx esbuild --bundle ts/setup.ts --sourcemap --outfile=./build/data/web/js/setup.js -npx esbuild --bundle ts/setup.ts --sourcemap --outfile=./data/web/js/setup.js
-rm -r build/data/web/js/ts -rm -r data/web/js/ts
$(info copying typescript) $(info copying typescript)
cp -r ts build/data/web/js cp -r ts data/web/js
swagger: swagger:
go1.16rc1 get github.com/swaggo/swag/cmd/swag go1.16rc1 get github.com/swaggo/swag/cmd/swag
@ -48,24 +48,33 @@ compress:
upx --lzma build/jfa-go upx --lzma build/jfa-go
bundle-css: bundle-css:
-mkdir -p build/data/web/css -mkdir -p data/web/css
$(info bundling css) $(info bundling css)
npx esbuild --bundle css/base.css --outfile=build/data/web/css/bundle.css --external:remixicon.css --minify npx esbuild --bundle css/base.css --outfile=data/web/css/bundle.css --external:remixicon.css --minify
copy: copy:
$(info copying fonts) $(info copying fonts)
cp -r node_modules/remixicon/fonts/remixicon.css node_modules/remixicon/fonts/remixicon.woff2 build/data/web/css/ cp -r node_modules/remixicon/fonts/remixicon.css node_modules/remixicon/fonts/remixicon.woff2 data/web/css/
$(info copying html) $(info copying html)
cp -r html build/data/ cp -r html data/
$(info copying static data) $(info copying static data)
-mkdir -p build/data/web -mkdir -p data/web
cp -r static/* build/data/web/ cp -r static/* data/web/
$(info copying language files) $(info copying language files)
cp -r lang build/data/ cp -r lang data/
embed:
python embed.py internal
noembed:
python embed.py external
-mkdir -p build
$(info copying internal data into build/)
cp -r data build/
install: install:
cp -r build $(DESTDIR)/jfa-go cp -r build $(DESTDIR)/jfa-go
all: configuration npm email version typescript bundle-css swagger copy compile all: configuration npm email version typescript bundle-css swagger copy embed compile
debug: configuration npm email version ts-debug bundle-css swagger copy compile all-external: configuration npm email version ts-debug bundle-css swagger copy noembed compile
debug: configuration npm email version ts-debug bundle-css swagger copy noembed compile

View File

@ -15,7 +15,7 @@ var emailEnabled = false
func (app *appContext) GetPath(sect, key string) (fs.FS, string) { func (app *appContext) GetPath(sect, key string) (fs.FS, string) {
val := app.config.Section(sect).Key(key).MustString("") val := app.config.Section(sect).Key(key).MustString("")
if strings.HasPrefix(val, "jfa-go:") { if strings.HasPrefix(val, "jfa-go:") {
return localFS, "build/data/" + strings.TrimPrefix(val, "jfa-go:") return localFS, "data/" + strings.TrimPrefix(val, "jfa-go:")
} }
return app.systemFS, val return app.systemFS, val
} }
@ -47,7 +47,6 @@ func (app *appContext) loadConfig() error {
app.config.Section("invite_emails").Key("email_text").SetValue(app.config.Section("invite_emails").Key("email_text").MustString("jfa-go:" + "invite-email.txt")) app.config.Section("invite_emails").Key("email_text").SetValue(app.config.Section("invite_emails").Key("email_text").MustString("jfa-go:" + "invite-email.txt"))
app.config.Section("email_confirmation").Key("email_html").SetValue(app.config.Section("email_confirmation").Key("email_html").MustString("jfa-go:" + "confirmation.html")) app.config.Section("email_confirmation").Key("email_html").SetValue(app.config.Section("email_confirmation").Key("email_html").MustString("jfa-go:" + "confirmation.html"))
fmt.Println(app.config.Section("email_confirmation").Key("email_html").String())
app.config.Section("email_confirmation").Key("email_text").SetValue(app.config.Section("email_confirmation").Key("email_text").MustString("jfa-go:" + "confirmation.txt")) app.config.Section("email_confirmation").Key("email_text").SetValue(app.config.Section("email_confirmation").Key("email_text").MustString("jfa-go:" + "confirmation.txt"))
app.config.Section("notifications").Key("expiry_html").SetValue(app.config.Section("notifications").Key("expiry_html").MustString("jfa-go:" + "expired.html")) app.config.Section("notifications").Key("expiry_html").SetValue(app.config.Section("notifications").Key("expiry_html").MustString("jfa-go:" + "expired.html"))

46
embed.py Executable file
View File

@ -0,0 +1,46 @@
#!/usr/bin/python
import sys
import argparse
parser = argparse.ArgumentParser()
parser.add_argument("embed", metavar="<true/false>|<internal/external>|<yes/no>", type=str)
trues = ["true", "internal", "yes", "y"]
falses = ["false", "external", "no", "n"]
EMBED = parser.parse_args().embed
with open("embed.go", "w") as f:
if EMBED in trues:
f.write("""package main
import (
"embed"
"log"
)
//go:embed data data/html data/web data/web/css data/web/js
var localFS embed.FS
//go:embed lang/common lang/admin lang/email lang/form lang/setup
var langFS embed.FS
func loadLocalFS() {
log.Println("Using internal storage")
}""")
elif EMBED in falses:
f.write("""package main
import (
"io/fs"
"os"
"log"
"path/filepath"
)
var localFS fs.FS
var langFS fs.FS
func loadLocalFS() {
log.Println("Using external storage")
executable, _ := os.Executable()
localFS = os.DirFS(filepath.Dir(executable))
langFS = os.DirFS(filepath.Join(filepath.Dir(executable), "data"))
}""")

18
main.go
View File

@ -3,7 +3,6 @@ package main
import ( import (
"context" "context"
"crypto/rand" "crypto/rand"
"embed"
"encoding/base64" "encoding/base64"
"encoding/json" "encoding/json"
"flag" "flag"
@ -51,13 +50,7 @@ type User struct {
Password string `json:"password"` Password string `json:"password"`
} }
//go:embed build/data build/data/html build/data/web build/data/web/css build/data/web/js build/data/web/js/ts // contains (almost) everything the application needs, essentially. This was a dumb design decision imo.
var localFS embed.FS
//go:embed lang/common lang/admin lang/email lang/form lang/setup
var langFS embed.FS
// contains everything the application needs, essentially. Wouldn't do this in the future.
type appContext struct { type appContext struct {
// defaults *Config // defaults *Config
config *ini.File config *ini.File
@ -90,7 +83,7 @@ type appContext struct {
func (app *appContext) loadHTML(router *gin.Engine) { func (app *appContext) loadHTML(router *gin.Engine) {
customPath := app.config.Section("files").Key("html_templates").MustString("") customPath := app.config.Section("files").Key("html_templates").MustString("")
templatePath := "build/data/html" templatePath := "data/html"
htmlFiles, err := fs.ReadDir(localFS, templatePath) htmlFiles, err := fs.ReadDir(localFS, templatePath)
if err != nil { if err != nil {
app.err.Fatalf("Couldn't access template directory: \"%s\"", templatePath) app.err.Fatalf("Couldn't access template directory: \"%s\"", templatePath)
@ -200,7 +193,7 @@ func start(asDaemon, firstCall bool) {
/* /*
set default config, data and local paths set default config, data and local paths
also, confusing naming here. data_path is not the internal 'data' directory, rather the users .config/jfa-go folder. also, confusing naming here. data_path is not the internal 'data' directory, rather the users .config/jfa-go folder.
local_path is the internal 'data' directory. localFS/data is the internal 'data' directory.
*/ */
userConfigDir, _ := os.UserConfigDir() userConfigDir, _ := os.UserConfigDir()
app.dataPath = filepath.Join(userConfigDir, "jfa-go") app.dataPath = filepath.Join(userConfigDir, "jfa-go")
@ -270,7 +263,7 @@ func start(asDaemon, firstCall bool) {
} }
if _, err := os.Stat(app.configPath); os.IsNotExist(err) { if _, err := os.Stat(app.configPath); os.IsNotExist(err) {
firstRun = true firstRun = true
dConfig, err := fs.ReadFile(localFS, "build/data/config-default.ini") dConfig, err := fs.ReadFile(localFS, "data/config-default.ini")
if err != nil { if err != nil {
app.err.Fatalf("Couldn't find default config file") app.err.Fatalf("Couldn't find default config file")
} }
@ -434,7 +427,7 @@ func start(asDaemon, firstCall bool) {
} }
app.configBasePath = "build/data/config-base.json" app.configBasePath = "data/config-base.json"
configBase, _ := fs.ReadFile(localFS, app.configBasePath) configBase, _ := fs.ReadFile(localFS, app.configBasePath)
json.Unmarshal(configBase, &app.configBase) json.Unmarshal(configBase, &app.configBase)
@ -759,6 +752,7 @@ func main() {
if flagPassed("test") { if flagPassed("test") {
TEST = true TEST = true
} }
loadLocalFS()
if flagPassed("start") { if flagPassed("start") {
args := []string{} args := []string{}
for i, f := range os.Args { for i, f := range os.Args {

View File

@ -1,7 +1,6 @@
package main package main
import ( import (
"fmt"
"io/fs" "io/fs"
"net/http" "net/http"
"strings" "strings"
@ -15,13 +14,12 @@ type httpFS struct {
} }
func (f httpFS) Open(name string) (http.File, error) { func (f httpFS) Open(name string) (http.File, error) {
fmt.Println("build/data/web" + name) return f.hfs.Open("data/web" + name)
return f.hfs.Open("build/data/web" + name)
} }
func (f httpFS) Exists(prefix string, filepath string) bool { func (f httpFS) Exists(prefix string, filepath string) bool {
if p := strings.TrimPrefix(filepath, prefix); len(p) < len(filepath) { if p := strings.TrimPrefix(filepath, prefix); len(p) < len(filepath) {
stats, err := fs.Stat(f.fs, "build/data/web/"+p) stats, err := fs.Stat(f.fs, "data/web/"+p)
if err != nil { if err != nil {
return false return false
} }

View File

@ -1,6 +1,6 @@
import subprocess import subprocess
import sys import sys
import os
try: try:
version = sys.argv[1].replace('v', '') version = sys.argv[1].replace('v', '')
except IndexError: except IndexError: