A good time to finally release a stable version
Signed-off-by: Izuru Yakumo <yakumo.izuru@chaotic.ninja> git-svn-id: file:///srv/svn/repo/aya/trunk@80 cec141ff-132a-4243-88a5-ce187bd62f94
This commit is contained in:
4
Makefile
4
Makefile
@@ -1,6 +1,8 @@
|
|||||||
DESTDIR ?=
|
DESTDIR ?=
|
||||||
GOFLAGS ?= -v -buildvcs=false -mod=vendor -buildmode=exe
|
GOFLAGS ?= -v -buildvcs=false -mod=vendor -buildmode=exe -ldflags "-w -X `go list`.Date=${DATE} -X `go list`.Vendor=${GOOS} -X `go list`.Version=${VERSION}"
|
||||||
PREFIX ?= /usr/local
|
PREFIX ?= /usr/local
|
||||||
|
DATE ?= `date -u +%F`
|
||||||
|
GOOS ?= `go env GOOS`
|
||||||
VERSION ?= `git describe --tags`
|
VERSION ?= `git describe --tags`
|
||||||
|
|
||||||
build:
|
build:
|
||||||
|
|||||||
29
README.md
29
README.md
@@ -17,28 +17,33 @@ Named after [Aya Shameimaru](https://en.touhouwiki.net/wiki/Aya_Shameimaru) from
|
|||||||
|
|
||||||
Build it manually assuming you have Go (>=1.17) installed:
|
Build it manually assuming you have Go (>=1.17) installed:
|
||||||
|
|
||||||
$ go install marisa.chaotic.ninja/aya/cmd/aya@latest
|
$ go install marisa.chaotic.ninja/aya/cmd/aya@latest (1)
|
||||||
--- or ---
|
--- or ---
|
||||||
$ git clone https://git.chaotic.ninja/yakumo.izuru/aya
|
$ git clone https://git.chaotic.ninja/yakumo.izuru/aya
|
||||||
$ cd aya
|
$ cd aya
|
||||||
$ make
|
$ make
|
||||||
# make install
|
# make install
|
||||||
|
|
||||||
|
(1) If you use this method, the `aya version` subcommand may print the wrong string,
|
||||||
|
but it should not be a problem unless you use it on a page.
|
||||||
|
|
||||||
## Ideology
|
## Ideology
|
||||||
|
|
||||||
Keep your texts in markdown, or HTML format right in the main directory
|
Keep your texts in markdown, [amber](https://github.com/eknkc/amber), or html format right in the main directory
|
||||||
of your blog/site.
|
of your blog/site.
|
||||||
|
|
||||||
Keep all service files (extensions, layout pages, deployment scripts etc)
|
Keep all service files (extensions, layout pages, deployment scripts etc)
|
||||||
in the `.aya` subdirectory.
|
in the `.aya` subdirectory.
|
||||||
|
|
||||||
Define variables in the header of the content files using [YAML]:
|
Define variables in the header of the content files using [YAML](https://www.yaml.io) :
|
||||||
|
|
||||||
title: My web site
|
```markdown
|
||||||
keywords: best website, hello, world
|
title: My web site
|
||||||
---
|
keywords: best website, hello, world
|
||||||
|
---
|
||||||
|
|
||||||
Markdown text goes after a header *separator*
|
Markdown text goes after a header *separator*
|
||||||
|
```
|
||||||
|
|
||||||
Use placeholders for variables and plugins in your markdown or html
|
Use placeholders for variables and plugins in your markdown or html
|
||||||
files, e.g. `{{ title }}` or `{{ command arg1 arg2 }}.
|
files, e.g. `{{ title }}` or `{{ command arg1 arg2 }}.
|
||||||
@@ -46,10 +51,11 @@ files, e.g. `{{ title }}` or `{{ command arg1 arg2 }}.
|
|||||||
Write extensions in any language you like and put them into the `.aya`
|
Write extensions in any language you like and put them into the `.aya`
|
||||||
subdiretory.
|
subdiretory.
|
||||||
|
|
||||||
Everything the extensions prints to stdout becomes the value of the
|
Everything the extensions prints to [stdout](https://man.freebsd.org/cgi/man.cgi?fd) becomes the value of the
|
||||||
placeholder.
|
placeholder.
|
||||||
|
|
||||||
Every variable from the content header will be passed via environment variables like `title` becomes `$AYA_TITLE` and so on. There are some special variables:
|
Every variable from the content header will be passed via environment variables like `title` becomes `$AYA\_TITLE` and so on.
|
||||||
|
There are some special variables:
|
||||||
|
|
||||||
* `$AYA` - a path to the `aya` executable
|
* `$AYA` - a path to the `aya` executable
|
||||||
* `$AYA\_OUTDIR` - a path to the directory with generated files
|
* `$AYA\_OUTDIR` - a path to the directory with generated files
|
||||||
@@ -93,10 +99,7 @@ content generation, or additional commands, like LESS to CSS conversion:
|
|||||||
lessc < $AYA_OUTDIR/styles.less > $AYA_OUTDIR/styles.css
|
lessc < $AYA_OUTDIR/styles.less > $AYA_OUTDIR/styles.css
|
||||||
rm -f $AYA_OUTDIR/styles.css
|
rm -f $AYA_OUTDIR/styles.css
|
||||||
|
|
||||||
## Extras
|
Note, you can also place `.gcss` files for [gcss](https://github.com/yosssi/gcss) to process instead
|
||||||
|
|
||||||
`aya` also supports generating `.html` and `.css` by means of using `.amber`
|
|
||||||
and `.gcss` files. See more at [eknkc/amber](https://github.com/eknkc/amber) [yosssi/gcss](https://github.com/yosssi/gcss)
|
|
||||||
|
|
||||||
## Command line usage
|
## Command line usage
|
||||||
|
|
||||||
|
|||||||
23
cmd/aya/build.go
Normal file
23
cmd/aya/build.go
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
// This function passes the files to build to their corresponding functions
|
||||||
|
// As far as I'm aware, Markdown has three possible filename extensions,
|
||||||
|
// but .md is the most common one known.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"path/filepath"
|
||||||
|
)
|
||||||
|
func build(path string, w io.Writer, vars Vars) error {
|
||||||
|
ext := filepath.Ext(path)
|
||||||
|
if ext == ".md" || ext == ".mkd" || ext == ".markdown" {
|
||||||
|
return buildMarkdown(path, w, vars)
|
||||||
|
} else if ext == ".htm" || ext == ".html" || ext == ".xht" || ext == ".xhtml" {
|
||||||
|
return buildHTML(path, w, vars)
|
||||||
|
} else if ext == ".amber" {
|
||||||
|
return buildAmber(path, w, vars)
|
||||||
|
} else if ext == ".gcss" {
|
||||||
|
return buildGCSS(path, w)
|
||||||
|
} else {
|
||||||
|
return buildRaw(path, w)
|
||||||
|
}
|
||||||
|
}
|
||||||
57
cmd/aya/buildall.go
Normal file
57
cmd/aya/buildall.go
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
// Build everything and store it on PUBDIR
|
||||||
|
// If boolean watch is true, it keeps on going
|
||||||
|
// every time you modify something.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func buildAll(watch bool) {
|
||||||
|
lastModified := time.Unix(0, 0)
|
||||||
|
modified := false
|
||||||
|
|
||||||
|
vars := globals()
|
||||||
|
for {
|
||||||
|
os.Mkdir(PUBDIR, 0755)
|
||||||
|
filepath.Walk(".", func(path string, info os.FileInfo, err error) error {
|
||||||
|
// ignore hidden files and directories
|
||||||
|
if filepath.Base(path)[0] == '.' || strings.HasPrefix(path, ".") {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// inform user about fs walk errors, but continue iteration
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("error:", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if info.IsDir() {
|
||||||
|
os.Mkdir(filepath.Join(PUBDIR, path), 0755)
|
||||||
|
return nil
|
||||||
|
} else if info.ModTime().After(lastModified) {
|
||||||
|
if !modified {
|
||||||
|
// First file in this build cycle is about to be modified
|
||||||
|
run(vars, "prehook")
|
||||||
|
modified = true
|
||||||
|
}
|
||||||
|
fmt.Println("build:", path)
|
||||||
|
return build(path, nil, vars)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
if modified {
|
||||||
|
// At least one file in this build cycle has been modified
|
||||||
|
run(vars, "posthook")
|
||||||
|
modified = false
|
||||||
|
}
|
||||||
|
if !watch {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
lastModified = time.Now()
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
}
|
||||||
|
}
|
||||||
64
cmd/aya/getvars.go
Normal file
64
cmd/aya/getvars.go
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
// getVars returns list of variables defined in a text file and actual file
|
||||||
|
// content following the variables declaration. Header is separated from
|
||||||
|
// content by an empty line. Header is in YAML
|
||||||
|
// If no empty newline is found - file is treated as content-only.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"gopkg.in/yaml.v3"
|
||||||
|
)
|
||||||
|
|
||||||
|
func getVars(path string, globals Vars) (Vars, string, error) {
|
||||||
|
b, err := os.ReadFile(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, "", err
|
||||||
|
}
|
||||||
|
s := string(b)
|
||||||
|
|
||||||
|
// Pick some default values for content-dependent variables
|
||||||
|
v := Vars{}
|
||||||
|
title := strings.Replace(strings.Replace(path, "_", " ", -1), "-", " ", -1)
|
||||||
|
v["title"] = strings.ToTitle(title)
|
||||||
|
v["description"] = ""
|
||||||
|
v["file"] = path
|
||||||
|
v["url"] = path[:len(path)-len(filepath.Ext(path))] + ".html"
|
||||||
|
v["output"] = filepath.Join(PUBDIR, v["url"])
|
||||||
|
|
||||||
|
// Override default values with globals
|
||||||
|
for name, value := range globals {
|
||||||
|
v[name] = value
|
||||||
|
}
|
||||||
|
// Add a layout if none is specified
|
||||||
|
// For this to work, the layout must be available
|
||||||
|
// in AYADIR
|
||||||
|
if _, ok := v["layout"]; !ok {
|
||||||
|
v["layout"] = "layout.html"
|
||||||
|
}
|
||||||
|
|
||||||
|
delim := "\n---\n"
|
||||||
|
if sep := strings.Index(s, delim); sep == -1 {
|
||||||
|
return v, s, nil
|
||||||
|
} else {
|
||||||
|
header := s[:sep]
|
||||||
|
body := s[sep+len(delim):1]
|
||||||
|
|
||||||
|
vars := Vars{}
|
||||||
|
if err := yaml.Unmarshal([]byte(header), &vars); err != nil {
|
||||||
|
fmt.Println("ERROR: Failed to parse header", err)
|
||||||
|
return nil, "", err
|
||||||
|
} else {
|
||||||
|
// Override default values and globals with the ones defined in the file
|
||||||
|
for key, value := range vars {
|
||||||
|
v[key] = value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(v["url"], "./") {
|
||||||
|
v["url"] = v["url"][2:]
|
||||||
|
}
|
||||||
|
return v, body, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
19
cmd/aya/globals.go
Normal file
19
cmd/aya/globals.go
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
// globals returns list of global OS environment variables that start
|
||||||
|
// with AYA_ prefix as Vars, so the values can be used inside templates
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
func globals() Vars {
|
||||||
|
vars := Vars{}
|
||||||
|
for _, e := range os.Environ() {
|
||||||
|
pair := strings.Split(e, "=")
|
||||||
|
if strings.HasPrefix(pair[0], "AYA_") {
|
||||||
|
vars[strings.ToLower(pair[0][3:])] = pair[1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return vars
|
||||||
|
}
|
||||||
223
cmd/aya/main.go
223
cmd/aya/main.go
@@ -1,17 +1,11 @@
|
|||||||
// $TheSupernovaDuo: marisa.chaotic.ninja/aya/cmd/aya, v0.7.0 2023-12-11 17:22:51+0000, yakumo_izuru Exp $
|
// $TheSupernovaDuo: marisa.chaotic.ninja/aya/cmd/aya, v1.0.0 2023-12-12 13:44:23+0000, yakumo_izuru Exp $
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
|
||||||
|
|
||||||
"gopkg.in/yaml.v3"
|
|
||||||
"marisa.chaotic.ninja/aya"
|
"marisa.chaotic.ninja/aya"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -23,219 +17,6 @@ const (
|
|||||||
|
|
||||||
type Vars map[string]string
|
type Vars map[string]string
|
||||||
|
|
||||||
// renameExt renames extension (if any) from oldext to newext
|
|
||||||
// If oldext is an empty string - extension is extracted automatically.
|
|
||||||
// If path has no extension - new extension is appended
|
|
||||||
func renameExt(path, oldext, newext string) string {
|
|
||||||
if oldext == "" {
|
|
||||||
oldext = filepath.Ext(path)
|
|
||||||
}
|
|
||||||
if oldext == "" || strings.HasSuffix(path, oldext) {
|
|
||||||
return strings.TrimSuffix(path, oldext) + newext
|
|
||||||
}
|
|
||||||
return path
|
|
||||||
}
|
|
||||||
|
|
||||||
// globals returns list of global OS environment variables that start
|
|
||||||
// with AYA_ prefix as Vars, so the values can be used inside templates
|
|
||||||
func globals() Vars {
|
|
||||||
vars := Vars{}
|
|
||||||
for _, e := range os.Environ() {
|
|
||||||
pair := strings.Split(e, "=")
|
|
||||||
if strings.HasPrefix(pair[0], "AYA_") {
|
|
||||||
vars[strings.ToLower(pair[0][3:])] = pair[1]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return vars
|
|
||||||
}
|
|
||||||
|
|
||||||
// run executes a command or a script. Vars define the command environment,
|
|
||||||
// each aya var is converted into OS environemnt variable with AYA_ prefix
|
|
||||||
// prepended. Additional variable $AYA contains path to the aya binary. Command
|
|
||||||
// stderr is printed to aya stderr, command output is returned as a string.
|
|
||||||
func run(vars Vars, cmd string, args ...string) (string, error) {
|
|
||||||
// First check if partial exists (.html)
|
|
||||||
if b, err := os.ReadFile(filepath.Join(AYADIR, cmd+".html")); err == nil {
|
|
||||||
return string(b), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
var errbuf, outbuf bytes.Buffer
|
|
||||||
c := exec.Command(cmd, args...)
|
|
||||||
env := []string{"AYA=" + os.Args[0], "AYA_OUTDIR=" + PUBDIR}
|
|
||||||
env = append(env, os.Environ()...)
|
|
||||||
for k, v := range vars {
|
|
||||||
env = append(env, "AYA_"+strings.ToUpper(k)+"="+v)
|
|
||||||
}
|
|
||||||
c.Env = env
|
|
||||||
c.Stdout = &outbuf
|
|
||||||
c.Stderr = &errbuf
|
|
||||||
|
|
||||||
err := c.Run()
|
|
||||||
|
|
||||||
if errbuf.Len() > 0 {
|
|
||||||
fmt.Println("ERROR:", errbuf.String())
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return string(outbuf.Bytes()), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// getVars returns list of variables defined in a text file and actual file
|
|
||||||
// content following the variables declaration. Header is separated from
|
|
||||||
// content by an empty line. Header can be either YAML or JSON.
|
|
||||||
// If no empty newline is found - file is treated as content-only.
|
|
||||||
func getVars(path string, globals Vars) (Vars, string, error) {
|
|
||||||
b, err := os.ReadFile(path)
|
|
||||||
if err != nil {
|
|
||||||
return nil, "", err
|
|
||||||
}
|
|
||||||
s := string(b)
|
|
||||||
|
|
||||||
// Pick some default values for content-dependent variables
|
|
||||||
v := Vars{}
|
|
||||||
title := strings.Replace(strings.Replace(path, "_", " ", -1), "-", " ", -1)
|
|
||||||
v["title"] = strings.ToTitle(title)
|
|
||||||
v["description"] = ""
|
|
||||||
v["file"] = path
|
|
||||||
v["url"] = path[:len(path)-len(filepath.Ext(path))] + ".html"
|
|
||||||
v["output"] = filepath.Join(PUBDIR, v["url"])
|
|
||||||
|
|
||||||
// Override default values with globals
|
|
||||||
for name, value := range globals {
|
|
||||||
v[name] = value
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add layout if none is specified
|
|
||||||
if _, ok := v["layout"]; !ok {
|
|
||||||
v["layout"] = "layout.html"
|
|
||||||
}
|
|
||||||
|
|
||||||
delim := "\n---\n"
|
|
||||||
if sep := strings.Index(s, delim); sep == -1 {
|
|
||||||
return v, s, nil
|
|
||||||
} else {
|
|
||||||
header := s[:sep]
|
|
||||||
body := s[sep+len(delim):]
|
|
||||||
|
|
||||||
vars := Vars{}
|
|
||||||
if err := yaml.Unmarshal([]byte(header), &vars); err != nil {
|
|
||||||
fmt.Println("ERROR: failed to parse header", err)
|
|
||||||
return nil, "", err
|
|
||||||
} else {
|
|
||||||
// Override default values + globals with the ones defines in the file
|
|
||||||
for key, value := range vars {
|
|
||||||
v[key] = value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if strings.HasPrefix(v["url"], "./") {
|
|
||||||
v["url"] = v["url"][2:]
|
|
||||||
}
|
|
||||||
return v, body, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Render expanding aya plugins and variables
|
|
||||||
func render(s string, vars Vars) (string, error) {
|
|
||||||
delimOpen := "{{"
|
|
||||||
delimClose := "}}"
|
|
||||||
|
|
||||||
out := &bytes.Buffer{}
|
|
||||||
for {
|
|
||||||
if from := strings.Index(s, delimOpen); from == -1 {
|
|
||||||
out.WriteString(s)
|
|
||||||
return out.String(), nil
|
|
||||||
} else {
|
|
||||||
if to := strings.Index(s, delimClose); to == -1 {
|
|
||||||
return "", fmt.Errorf("Closing delimiter not found")
|
|
||||||
} else {
|
|
||||||
out.WriteString(s[:from])
|
|
||||||
cmd := s[from+len(delimOpen) : to]
|
|
||||||
s = s[to+len(delimClose):]
|
|
||||||
m := strings.Fields(cmd)
|
|
||||||
if len(m) == 1 {
|
|
||||||
if v, ok := vars[m[0]]; ok {
|
|
||||||
out.WriteString(v)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if res, err := run(vars, m[0], m[1:]...); err == nil {
|
|
||||||
out.WriteString(res)
|
|
||||||
} else {
|
|
||||||
fmt.Println(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// This function passes the files to build to their corresponding functions
|
|
||||||
// As far as I'm aware, Markdown has three possible filename extensions,
|
|
||||||
// but .md is the most common one known.
|
|
||||||
func build(path string, w io.Writer, vars Vars) error {
|
|
||||||
ext := filepath.Ext(path)
|
|
||||||
if ext == ".md" || ext == ".mkd" || ext == ".markdown" {
|
|
||||||
return buildMarkdown(path, w, vars)
|
|
||||||
} else if ext == ".htm" || ext == ".html" || ext == ".xht" || ext == ".xhtml" {
|
|
||||||
return buildHTML(path, w, vars)
|
|
||||||
} else if ext == ".amber" {
|
|
||||||
return buildAmber(path, w, vars)
|
|
||||||
} else if ext == ".gcss" {
|
|
||||||
return buildGCSS(path, w)
|
|
||||||
} else {
|
|
||||||
return buildRaw(path, w)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build everything and store it on PUBDIR
|
|
||||||
// If boolean watch is true, it keeps on going
|
|
||||||
// every time you modify something.
|
|
||||||
func buildAll(watch bool) {
|
|
||||||
lastModified := time.Unix(0, 0)
|
|
||||||
modified := false
|
|
||||||
|
|
||||||
vars := globals()
|
|
||||||
for {
|
|
||||||
os.Mkdir(PUBDIR, 0755)
|
|
||||||
filepath.Walk(".", func(path string, info os.FileInfo, err error) error {
|
|
||||||
// ignore hidden files and directories
|
|
||||||
if filepath.Base(path)[0] == '.' || strings.HasPrefix(path, ".") {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
// inform user about fs walk errors, but continue iteration
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("error:", err)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if info.IsDir() {
|
|
||||||
os.Mkdir(filepath.Join(PUBDIR, path), 0755)
|
|
||||||
return nil
|
|
||||||
} else if info.ModTime().After(lastModified) {
|
|
||||||
if !modified {
|
|
||||||
// First file in this build cycle is about to be modified
|
|
||||||
run(vars, "prehook")
|
|
||||||
modified = true
|
|
||||||
}
|
|
||||||
fmt.Println("build:", path)
|
|
||||||
return build(path, nil, vars)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
if modified {
|
|
||||||
// At least one file in this build cycle has been modified
|
|
||||||
run(vars, "posthook")
|
|
||||||
modified = false
|
|
||||||
}
|
|
||||||
if !watch {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
lastModified = time.Now()
|
|
||||||
time.Sleep(1 * time.Second)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize the environment
|
// Initialize the environment
|
||||||
func init() {
|
func init() {
|
||||||
// prepend .aya to $PATH, so plugins will be found before OS commands
|
// prepend .aya to $PATH, so plugins will be found before OS commands
|
||||||
@@ -294,7 +75,7 @@ func main() {
|
|||||||
fmt.Println(strings.TrimSpace(s))
|
fmt.Println(strings.TrimSpace(s))
|
||||||
}
|
}
|
||||||
case "version":
|
case "version":
|
||||||
fmt.Printf("%v\n", aya.FullVersion())
|
fmt.Printf("%v\n", aya.PrintVersion())
|
||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
case "watch":
|
case "watch":
|
||||||
buildAll(true)
|
buildAll(true)
|
||||||
|
|||||||
19
cmd/aya/renameext.go
Normal file
19
cmd/aya/renameext.go
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
// renameExt renames extension (if any) from oldext to newext
|
||||||
|
// If oldext is an empty string - extension is extracted automatically.
|
||||||
|
// If path has no extension - new extension is appended
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
func renameExt(path, oldext, newext string) string {
|
||||||
|
if oldext == "" {
|
||||||
|
oldext = filepath.Ext(path)
|
||||||
|
}
|
||||||
|
if oldext == "" || strings.HasSuffix(path, oldext) {
|
||||||
|
return strings.TrimSuffix(path, oldext) + newext
|
||||||
|
}
|
||||||
|
return path
|
||||||
|
}
|
||||||
42
cmd/aya/render.go
Normal file
42
cmd/aya/render.go
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
// Render expanding aya plugins and variables
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
func render(s string, vars Vars) (string, error) {
|
||||||
|
delimOpen := "{{"
|
||||||
|
delimClose := "}}"
|
||||||
|
|
||||||
|
out := &bytes.Buffer{}
|
||||||
|
for {
|
||||||
|
if from := strings.Index(s, delimOpen); from == -1 {
|
||||||
|
out.WriteString(s)
|
||||||
|
return out.String(), nil
|
||||||
|
} else {
|
||||||
|
if to := strings.Index(s, delimClose); to == -1 {
|
||||||
|
return "", fmt.Errorf("Closing delimiter not found")
|
||||||
|
} else {
|
||||||
|
out.WriteString(s[:from])
|
||||||
|
cmd := s[from+len(delimOpen) : to]
|
||||||
|
s = s[to+len(delimClose):]
|
||||||
|
m := strings.Fields(cmd)
|
||||||
|
if len(m) == 1 {
|
||||||
|
if v, ok := vars[m[0]]; ok {
|
||||||
|
out.WriteString(v)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if res, err := run(vars, m[0], m[1:]...); err == nil {
|
||||||
|
out.WriteString(res)
|
||||||
|
} else {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
42
cmd/aya/run.go
Normal file
42
cmd/aya/run.go
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
// run() executes a command or a script.
|
||||||
|
// Vars define the command environment, each aya var is converted into OS environment variable with AYA_ prefix prepended.
|
||||||
|
// Additional variable $AYA contains path to the aya binary.
|
||||||
|
// Command stderr(4) is printed to aya stderr(4), command stdout(4) is returned as a string
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
func run(vars Vars, cmd string, args ...string) (string, error) {
|
||||||
|
// First check if partial exists (.html)
|
||||||
|
if b, err := os.ReadFile(filepath.Join(AYADIR, cmd+".html")); err == nil {
|
||||||
|
return string(b), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var errbuf, outbuf bytes.Buffer
|
||||||
|
c := exec.Command(cmd, args...)
|
||||||
|
env := []string{"AYA=" + os.Args[0], "AYA_OUTDIR=" + PUBDIR}
|
||||||
|
env = append(env, os.Environ()...)
|
||||||
|
for k, v := range vars {
|
||||||
|
env = append(env, "AYA_"+strings.ToUpper(k)+"="+v)
|
||||||
|
}
|
||||||
|
c.Env = env
|
||||||
|
c.Stdout = &outbuf
|
||||||
|
c.Stderr = &errbuf
|
||||||
|
|
||||||
|
err := c.Run()
|
||||||
|
|
||||||
|
if errbuf.Len() > 0 {
|
||||||
|
fmt.Println("ERROR:", errbuf.String())
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return string(outbuf.Bytes()), nil
|
||||||
|
}
|
||||||
2
usage.go
2
usage.go
@@ -6,7 +6,7 @@ import (
|
|||||||
|
|
||||||
// This function is called by the `aya help` subcommand
|
// This function is called by the `aya help` subcommand
|
||||||
func PrintUsage() {
|
func PrintUsage() {
|
||||||
fmt.Printf("aya/%v\n", FullVersion())
|
fmt.Printf("aya/%v\n", PrintFullVersion())
|
||||||
fmt.Println("Homepage: https://aya.chaotic.ninja")
|
fmt.Println("Homepage: https://aya.chaotic.ninja")
|
||||||
fmt.Println("Repository: https://git.chaotic.ninja/yakumo.izuru/aya")
|
fmt.Println("Repository: https://git.chaotic.ninja/yakumo.izuru/aya")
|
||||||
fmt.Println("==")
|
fmt.Println("==")
|
||||||
|
|||||||
22
version.go
22
version.go
@@ -3,19 +3,21 @@ package aya
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
// Set to current tag
|
// Variables set at build-time
|
||||||
Version = "v0.7.0"
|
Date string
|
||||||
Time = time.Now()
|
Vendor string
|
||||||
|
Version string
|
||||||
)
|
)
|
||||||
|
|
||||||
// FullVersion display the full version and build
|
// PrintVersion only displays the obvious
|
||||||
func FullVersion() string {
|
func PrintVersion() string {
|
||||||
d := Time.Day()
|
return fmt.Sprintf("%s", Version)
|
||||||
m := Time.Month()
|
}
|
||||||
y := Time.Year()
|
|
||||||
return fmt.Sprintf("%v || %d.%d.%d", Version, y, m, d)
|
// PrintFullVersion display the full version and build
|
||||||
|
func PrintFullVersion() string {
|
||||||
|
return fmt.Sprintf("%s, built at %s, on %s", Version, Date, Vendor)
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user