gowebbuild/main.go

334 lines
6.2 KiB
Go
Raw Normal View History

2021-12-17 21:39:46 +01:00
package main
import (
"encoding/json"
2022-02-01 17:15:14 +01:00
"errors"
2021-12-17 21:39:46 +01:00
"fmt"
"io/ioutil"
2022-05-11 12:35:55 +02:00
"net/http"
2021-12-17 21:39:46 +01:00
"os"
"os/signal"
"path/filepath"
"strings"
2021-12-17 21:39:46 +01:00
"syscall"
"time"
"github.com/evanw/esbuild/pkg/api"
"github.com/goyek/goyek"
"github.com/jaschaephraim/lrserver"
"github.com/otiai10/copy"
2021-12-17 21:39:46 +01:00
"github.com/radovskyb/watcher"
)
var triggerReload = make(chan struct{})
type options struct {
ESBuild api.BuildOptions
Watch struct {
2022-02-01 17:15:14 +01:00
Path string
Exclude []string
2021-12-17 21:39:46 +01:00
}
2022-05-11 12:35:55 +02:00
Serve struct {
Path string
Port int
}
Copy []struct {
Src string
Dest string
}
Replace []struct {
Pattern string
Search string
Replace string
}
Link struct {
From string
To string
}
}
func readCfg(cfgPath string) []options {
cfgContent, err := os.ReadFile(cfgPath)
if err != nil {
fmt.Printf("%+v\n", err)
os.Exit(1)
}
optsSetups := []options{}
err = json.Unmarshal(cfgContent, &optsSetups)
if err != nil {
opt := options{}
err = json.Unmarshal(cfgContent, &opt)
if err != nil {
fmt.Printf("%+v\n", err)
os.Exit(1)
}
optsSetups = append(optsSetups, opt)
}
return optsSetups
2021-12-17 21:39:46 +01:00
}
func main() {
2022-02-01 17:15:14 +01:00
flow := &goyek.Flow{}
2021-12-17 21:39:46 +01:00
2022-02-01 17:15:14 +01:00
cfgPathParam := flow.RegisterStringParam(goyek.StringParam{
Name: "c",
Usage: "Path to config file config file.",
Default: "./.gowebbuild.json",
})
2021-12-17 21:39:46 +01:00
prodParam := flow.RegisterBoolParam(goyek.BoolParam{
Name: "p",
Usage: "Use production ready build settings",
Default: false,
})
buildOnly := goyek.Task{
Name: "build",
Usage: "",
Params: goyek.Params{cfgPathParam, prodParam},
Action: func(tf *goyek.TF) {
cfgPath := cfgPathParam.Get(tf)
os.Chdir(filepath.Dir(cfgPath))
opts := readCfg(cfgPath)
for _, o := range opts {
cp(o)
if prodParam.Get(tf) {
o.ESBuild.MinifyIdentifiers = true
o.ESBuild.MinifySyntax = true
o.ESBuild.MinifyWhitespace = true
o.ESBuild.Sourcemap = api.SourceMapNone
}
api.Build(o.ESBuild)
replace(o)
}
},
}
2022-05-11 12:35:55 +02:00
watch := goyek.Task{
Name: "watch",
2022-02-01 17:15:14 +01:00
Usage: "",
Params: goyek.Params{cfgPathParam},
Action: func(tf *goyek.TF) {
cfgPath := cfgPathParam.Get(tf)
os.Chdir(filepath.Dir(cfgPath))
optsSetups := readCfg(cfgPath)
2021-12-17 21:39:46 +01:00
c := make(chan os.Signal, 1)
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
for i := range optsSetups {
opts := optsSetups[i]
2021-12-17 21:39:46 +01:00
go func(opts options) {
w := watcher.New()
w.SetMaxEvents(1)
w.FilterOps(watcher.Write, watcher.Rename, watcher.Move, watcher.Create, watcher.Remove)
2022-02-01 17:15:14 +01:00
if len(opts.Watch.Exclude) > 0 {
w.Ignore(opts.Watch.Exclude...)
}
2022-02-01 17:15:14 +01:00
if err := w.AddRecursive(opts.Watch.Path); err != nil {
fmt.Println(err.Error())
os.Exit(1)
}
2021-12-17 21:39:46 +01:00
go func() {
for {
select {
case event := <-w.Event:
fmt.Printf("File %s changed\n", event.Name())
cp(opts)
build(opts)
replace(opts)
case err := <-w.Error:
fmt.Println(err.Error())
case <-w.Closed:
return
}
2021-12-17 21:39:46 +01:00
}
}()
fmt.Printf("Watching %d elements in %s\n", len(w.WatchedFiles()), opts.Watch.Path)
cp(opts)
build(opts)
replace(opts)
if err := w.Start(time.Millisecond * 100); err != nil {
fmt.Println(err.Error())
2021-12-17 21:39:46 +01:00
}
}(opts)
if opts.Serve.Path != "" {
go func() {
port := 8888
if opts.Serve.Port != 0 {
port = opts.Serve.Port
}
2021-12-17 21:39:46 +01:00
http.Handle("/", http.FileServer(http.Dir(opts.Serve.Path)))
2021-12-17 21:39:46 +01:00
fmt.Printf("Serving contents of %s at :%d\n", opts.Serve.Path, port)
err := http.ListenAndServe(fmt.Sprintf(":%d", port), nil)
2022-02-01 17:15:14 +01:00
if err != nil {
fmt.Printf("%+v\n", err.Error())
os.Exit(1)
}
}()
2021-12-17 21:39:46 +01:00
}
if opts.Link.From != "" {
reqBuildCh := link(opts.Link.From, opts.Link.To)
go func() {
for range reqBuildCh {
cp(opts)
build(opts)
replace(opts)
}
}()
}
}
2021-12-17 21:39:46 +01:00
go func() {
fmt.Println("Starting live reload server")
2021-12-17 21:39:46 +01:00
lr := lrserver.New(lrserver.DefaultName, lrserver.DefaultPort)
go func() {
for {
<-triggerReload
lr.Reload("")
}
}()
lr.SetStatusLog(nil)
err := lr.ListenAndServe()
if err != nil {
panic(err)
}
}()
<-c
fmt.Println("\nExit")
os.Exit(0)
},
2022-02-01 17:15:14 +01:00
}
2021-12-17 21:39:46 +01:00
2022-05-11 12:35:55 +02:00
flow.DefaultTask = flow.Register(watch)
flow.Register(buildOnly)
2021-12-17 21:39:46 +01:00
flow.Main()
}
func cp(opts options) {
if len(opts.Copy) == 0 {
fmt.Println("Nothing to copy")
return
}
for _, op := range opts.Copy {
paths, err := filepath.Glob(op.Src)
if err != nil {
fmt.Printf("Invalid glob pattern: %s\n", op.Src)
continue
}
destIsDir := isDir(op.Dest)
for _, p := range paths {
d := op.Dest
if destIsDir && isFile(p) {
d = filepath.Join(d, filepath.Base(p))
}
err := copy.Copy(p, d)
fmt.Printf("Copying %s to %s\n", p, d)
if err != nil {
fmt.Printf("Failed to copy %s: %v\n", p, err)
continue
}
}
}
}
func replace(opts options) {
if len(opts.Replace) == 0 {
fmt.Println("Nothing to replace")
return
}
for _, op := range opts.Replace {
paths, err := filepath.Glob(op.Pattern)
if err != nil {
fmt.Printf("Invalid glob pattern: %s\n", op.Pattern)
continue
}
2022-05-11 12:35:55 +02:00
for _, p := range paths {
if !isFile(p) {
continue
}
read, err := ioutil.ReadFile(p)
if err != nil {
fmt.Printf("%+v\n", err)
os.Exit(1)
}
r := op.Replace
if strings.HasPrefix(op.Replace, "$") {
r = os.ExpandEnv(op.Replace)
}
2022-05-11 12:35:55 +02:00
count := strings.Count(string(read), op.Search)
if count > 0 {
fmt.Printf("Replacing %d occurrences of '%s' with '%s' in %s\n", count, op.Search, r, p)
newContents := strings.ReplaceAll(string(read), op.Search, r)
err = ioutil.WriteFile(p, []byte(newContents), 0)
2022-05-11 12:35:55 +02:00
if err != nil {
fmt.Printf("%+v\n", err)
os.Exit(1)
}
}
}
}
}
func isFile(path string) bool {
2022-02-01 17:15:14 +01:00
stat, err := os.Stat(path)
if errors.Is(err, os.ErrNotExist) {
return false
}
return !stat.IsDir()
}
func isDir(path string) bool {
2022-02-01 17:15:14 +01:00
stat, err := os.Stat(path)
if errors.Is(err, os.ErrNotExist) {
os.MkdirAll(path, 0755)
return true
}
return err == nil && stat.IsDir()
}
2021-12-17 21:39:46 +01:00
func build(opts options) {
result := api.Build(opts.ESBuild)
if len(result.Errors) == 0 {
2021-12-17 21:39:46 +01:00
triggerReload <- struct{}{}
}
}