Define favorite scripts in user dir

This commit is contained in:
Wancat
2022-11-17 16:54:30 +08:00
parent 3aab4827fe
commit 3ef4dfaa48
7 changed files with 171 additions and 126 deletions

View File

@@ -5,3 +5,7 @@ var SCRIPTS = map[string]string{
"register": "r --tail 10", "register": "r --tail 10",
"balance this month": "b -b \"this month\"", "balance this month": "b -b \"this month\"",
} }
const SCRIPTS_FILE = "queries.txt"
const HTPASSWD_FILE = ".htpasswd"
const DEFAULT_JOURNAL = "ledger.txt"

View File

@@ -19,9 +19,6 @@ var HOST string
var store auth.AuthStore var store auth.AuthStore
const HTPASSWD_FILE = ".htpasswd"
const DEFAULT_JOURNAL = "ledger.txt"
func init() { func init() {
ledgerTpl = template.Must(template.ParseGlob("tx/*")) ledgerTpl = template.Must(template.ParseGlob("tx/*"))
flag.StringVar(&DATA_DIR, "d", "data", "data folder") flag.StringVar(&DATA_DIR, "d", "data", "data folder")

148
models.go Normal file
View File

@@ -0,0 +1,148 @@
package main
import (
"bufio"
"bytes"
"fmt"
"io"
"log"
"os"
"os/exec"
"path"
"strings"
"time"
)
type User struct {
IsLogin bool
Email string `form:"email" binding:"required"`
Password string `form:"password" binding:"required"`
}
func (u *User) Dir() string {
dir := path.Join(DATA_DIR, u.Email)
if err := os.MkdirAll(dir, 0755); err != nil {
panic(err)
}
return dir
}
func (u *User) FilePath(name string) string {
return path.Join(u.Dir(), name)
}
func (u *User) File(name string, mode int) (*os.File, error) {
return os.OpenFile(u.FilePath(name), mode, 0644)
}
func (u *User) AppendFile(name string) (*os.File, error) {
return u.File(name, os.O_WRONLY|os.O_CREATE|os.O_APPEND)
}
func (u *User) ReadFile(name string) (*os.File, error) {
return u.File(name, os.O_RDONLY|os.O_CREATE)
}
func (u *User) WriteFile(name string) (*os.File, error) {
return u.File(name, os.O_WRONLY|os.O_CREATE|os.O_TRUNC)
}
func (u *User) List() ([]string, error) {
files, err := os.ReadDir(u.Dir())
if err != nil {
panic(err)
}
result := make([]string, len(files))
for i, v := range files {
result[i] = v.Name()
}
return result, nil
}
func (u *User) appendToFile(tx string) (err error) {
f, err := u.AppendFile(DEFAULT_JOURNAL)
if err != nil {
return err
}
defer f.Close()
buf := strings.NewReader(strings.ReplaceAll(tx, "\r", "")) // Remove CR generated from browser
_, err = io.Copy(f, buf)
return err
}
func (u *User) overwriteFile(filename string, tx string) (err error) {
f, err := u.WriteFile(filename)
if err != nil {
return err
}
defer f.Close()
buf := strings.NewReader(strings.ReplaceAll(tx, "\r", "")) // Remove CR generated from browser
_, err = io.Copy(f, buf)
return err
}
func (u *User) query(query string) (result string, err error) {
var buf bytes.Buffer
cmd := exec.Command("ledger", "--file", DEFAULT_JOURNAL)
cmd.Dir = u.Dir()
cmd.Stdin = strings.NewReader(query)
cmd.Stdout = &buf
cmd.Stderr = &buf
err = cmd.Run()
return buf.String(), err
}
func (u *User) scripts() (scripts map[string]string, err error) {
f, err := u.ReadFile(SCRIPTS_FILE)
if err != nil {
panic(err)
}
defer f.Close()
fileScanner := bufio.NewScanner(f)
fileScanner.Split(bufio.ScanLines)
scripts = make(map[string]string)
for fileScanner.Scan() {
arr := strings.SplitN(fileScanner.Text(), ":", 2)
if len(arr) < 2 {
err = fmt.Errorf("invalid data %s", arr)
return
}
scripts[arr[0]] = arr[1]
}
return
}
func (u *User) templates() (templates []string, err error) {
files, err := u.List()
if err != nil {
return
}
for _, v := range files {
if strings.HasSuffix(v, ".tpl") {
templates = append(templates, v)
}
}
log.Println(templates)
return
}
type TxData struct {
Action string `form:"action" binding:"required"`
Name string `form:"name"`
Date string
Amount string `form:"amount" binding:"required"`
Destination string `form:"dest"`
Source string `form:"src"`
}
func newTx(data TxData) (result string, err error) {
data.Date = time.Now().Format("2006/01/02")
var buf bytes.Buffer
err = ledgerTpl.ExecuteTemplate(&buf, data.Action, data)
return buf.String(), nil
}

View File

@@ -3,7 +3,6 @@ package main
import ( import (
"io/ioutil" "io/ioutil"
"log" "log"
"text/template"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
) )
@@ -31,12 +30,18 @@ func router() *gin.Engine {
authZone := r.Group("", authenticate) authZone := r.Group("", authenticate)
authZone.GET("/dashboard", func(c *gin.Context) { authZone.GET("/dashboard", func(c *gin.Context) {
HTML(c, 200, "dashboard.html", struct { user := getUser(c)
Templates []*template.Template scripts, err := user.scripts()
Scripts map[string]string if err != nil {
}{ panic(err)
ledgerTpl.Templates(), }
SCRIPTS, templates, err := user.templates()
if err != nil {
panic(err)
}
HTML(c, 200, "dashboard.html", gin.H{
"Scripts": scripts,
"Templates": templates,
}) })
}) })
@@ -118,14 +123,18 @@ func router() *gin.Engine {
}) })
authZone.GET("/query", func(c *gin.Context) { authZone.GET("/query", func(c *gin.Context) {
user := getUser(c)
response := struct { response := struct {
Query string Query string
Result string Result string
Scripts map[string]string Scripts map[string]string
}{Scripts: SCRIPTS} }{}
user := getUser(c)
var ok bool var ok bool
var err error var err error
response.Scripts, err = user.scripts()
if err != nil {
panic(err)
}
response.Query, ok = c.GetQuery("query") response.Query, ok = c.GetQuery("query")
if ok && response.Query != "" { if ok && response.Query != "" {
response.Result, err = user.query(response.Query) response.Result, err = user.query(response.Query)

View File

@@ -3,7 +3,7 @@
<form action="/new" method="POST"> <form action="/new" method="POST">
<label>選擇模板: <label>選擇模板:
{{ range .Templates }} {{ range .Templates }}
<input type="radio" name="action" value="{{ .Name }}">{{ .Name }}</option> <input type="radio" name="action" value="{{ . }}">{{ . }}</option>
{{ end }} {{ end }}
</label><br> </label><br>
<label>金額:<input name="amount" type="number"></label><br> <label>金額:<input name="amount" type="number"></label><br>

61
tx.go
View File

@@ -1,61 +0,0 @@
package main
import (
"bytes"
"io"
"os/exec"
"strings"
"time"
)
type TxData struct {
Action string `form:"action" binding:"required"`
Name string `form:"name"`
Date string
Amount string `form:"amount" binding:"required"`
Destination string `form:"dest"`
Source string `form:"src"`
}
func newTx(data TxData) (result string, err error) {
data.Date = time.Now().Format("2006/01/02")
var buf bytes.Buffer
err = ledgerTpl.ExecuteTemplate(&buf, data.Action, data)
return buf.String(), nil
}
func (u *User) appendToFile(tx string) (err error) {
f, err := u.AppendFile(DEFAULT_JOURNAL)
if err != nil {
return err
}
defer f.Close()
buf := strings.NewReader(strings.ReplaceAll(tx, "\r", "")) // Remove CR generated from browser
_, err = io.Copy(f, buf)
return err
}
func (u *User) overwriteFile(filename string, tx string) (err error) {
f, err := u.WriteFile(filename)
if err != nil {
return err
}
defer f.Close()
buf := strings.NewReader(strings.ReplaceAll(tx, "\r", "")) // Remove CR generated from browser
_, err = io.Copy(f, buf)
return err
}
func (u *User) query(query string) (result string, err error) {
var buf bytes.Buffer
cmd := exec.Command("ledger", "--file", DEFAULT_JOURNAL)
cmd.Dir = u.Dir()
cmd.Stdin = strings.NewReader(query)
cmd.Stdout = &buf
cmd.Stderr = &buf
err = cmd.Run()
return buf.String(), err
}

52
user.go
View File

@@ -1,52 +0,0 @@
package main
import (
"os"
"path"
)
type User struct {
IsLogin bool
Email string `form:"email" binding:"required"`
Password string `form:"password" binding:"required"`
}
func (u *User) Dir() string {
dir := path.Join(DATA_DIR, u.Email)
if err := os.MkdirAll(dir, 0755); err != nil {
panic(err)
}
return dir
}
func (u *User) FilePath(name string) string {
return path.Join(u.Dir(), name)
}
func (u *User) File(name string, mode int) (*os.File, error) {
return os.OpenFile(u.FilePath(name), mode, 0644)
}
func (u *User) AppendFile(name string) (*os.File, error) {
return u.File(name, os.O_WRONLY|os.O_CREATE|os.O_APPEND)
}
func (u *User) ReadFile(name string) (*os.File, error) {
return u.File(name, os.O_RDONLY|os.O_CREATE)
}
func (u *User) WriteFile(name string) (*os.File, error) {
return u.File(name, os.O_WRONLY|os.O_CREATE|os.O_TRUNC)
}
func (u *User) List() ([]string, error) {
files, err := os.ReadDir(u.Dir())
if err != nil {
panic(err)
}
result := make([]string, len(files))
for i, v := range files {
result[i] = v.Name()
}
return result, nil
}