parent
19aa5f82fb
commit
fc95242c94
|
|
@ -17,11 +17,13 @@ type Uploader int
|
||||||
const (
|
const (
|
||||||
UploaderTodo = Uploader(iota)
|
UploaderTodo = Uploader(iota)
|
||||||
UploaderLedger
|
UploaderLedger
|
||||||
|
UploaderPTTodo
|
||||||
)
|
)
|
||||||
|
|
||||||
var uploaders = map[string]Uploader{
|
var uploaders = map[string]Uploader{
|
||||||
"todo": UploaderTodo,
|
"todo": UploaderTodo,
|
||||||
"ledger": UploaderLedger,
|
"ledger": UploaderLedger,
|
||||||
|
"pttodo": UploaderPTTodo,
|
||||||
}
|
}
|
||||||
|
|
||||||
type Config struct {
|
type Config struct {
|
||||||
|
|
@ -48,7 +50,7 @@ func NewConfig() Config {
|
||||||
as.Append(args.STRING, "emailpass", "email password", "diblloewfncwssof")
|
as.Append(args.STRING, "emailpass", "email password", "diblloewfncwssof")
|
||||||
as.Append(args.STRING, "emailimap", "email imap", "imap.gmail.com:993")
|
as.Append(args.STRING, "emailimap", "email imap", "imap.gmail.com:993")
|
||||||
|
|
||||||
as.Append(args.STRING, "uploader", "todo, ledger", "todo")
|
as.Append(args.STRING, "uploader", "todo, ledger, pttodo", "todo")
|
||||||
|
|
||||||
as.Append(args.STRING, "todoaddr", "todo addr", "https://todo-server.remote.blapointe.com")
|
as.Append(args.STRING, "todoaddr", "todo addr", "https://todo-server.remote.blapointe.com")
|
||||||
as.Append(args.STRING, "todopass", "todo pass", "gJtEXbbLHLf54yS9EdujtVN2n6Y")
|
as.Append(args.STRING, "todopass", "todo pass", "gJtEXbbLHLf54yS9EdujtVN2n6Y")
|
||||||
|
|
|
||||||
43
upload.go
43
upload.go
|
|
@ -1,6 +1,7 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
|
@ -8,10 +9,13 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
|
"path"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
yaml "gopkg.in/yaml.v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
func Upload(config Config, transaction *Transaction) error {
|
func Upload(config Config, transaction *Transaction) error {
|
||||||
|
|
@ -20,6 +24,8 @@ func Upload(config Config, transaction *Transaction) error {
|
||||||
return uploadTodo(config, transaction)
|
return uploadTodo(config, transaction)
|
||||||
case UploaderLedger:
|
case UploaderLedger:
|
||||||
return uploadLedger(config, transaction)
|
return uploadLedger(config, transaction)
|
||||||
|
case UploaderPTTodo:
|
||||||
|
return uploadPTTodo(config, transaction)
|
||||||
default:
|
default:
|
||||||
return errors.New("not impl: uploader")
|
return errors.New("not impl: uploader")
|
||||||
}
|
}
|
||||||
|
|
@ -49,6 +55,43 @@ func uploadTodo(config Config, transaction *Transaction) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func uploadPTTodo(config Config, transaction *Transaction) error {
|
||||||
|
b, err := ioutil.ReadFile(config.TodoAddr)
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
b = []byte("todo:\n")
|
||||||
|
} else if err != nil {
|
||||||
|
return err
|
||||||
|
} else if len(b) == 0 {
|
||||||
|
b = []byte("todo:\n")
|
||||||
|
}
|
||||||
|
f, err := ioutil.TempFile(os.TempDir(), path.Base(config.TodoAddr))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
sep := []byte{'\n'}
|
||||||
|
seek := []byte("todo:")
|
||||||
|
for len(b) > 0 {
|
||||||
|
idx := bytes.Index(b, sep)
|
||||||
|
if idx == -1 {
|
||||||
|
idx = len(b) - 1
|
||||||
|
}
|
||||||
|
fmt.Fprintf(f, "%s\n", b[:idx])
|
||||||
|
if bytes.Equal(bytes.TrimSpace(b[:idx]), seek) {
|
||||||
|
fmt.Fprintf(f, `- {"todo":%q, "tags":%q}%s`, transaction.Format(), config.TodoTag, "\n")
|
||||||
|
}
|
||||||
|
b = b[idx+1:]
|
||||||
|
}
|
||||||
|
f.Close()
|
||||||
|
var v interface{}
|
||||||
|
if b, err := ioutil.ReadFile(f.Name()); err != nil {
|
||||||
|
return err
|
||||||
|
} else if err := yaml.Unmarshal(b, &v); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return os.Rename(f.Name(), config.TodoAddr)
|
||||||
|
}
|
||||||
|
|
||||||
func uploadLedger(config Config, transaction *Transaction) error {
|
func uploadLedger(config Config, transaction *Transaction) error {
|
||||||
f, err := os.OpenFile(config.TodoAddr, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
|
f, err := os.OpenFile(config.TodoAddr, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
||||||
|
|
@ -4,12 +4,92 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"local/storage"
|
"local/storage"
|
||||||
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func TestUploadPTTodo(t *testing.T) {
|
||||||
|
addr := path.Join(t.TempDir(), "test.upload.pttodo")
|
||||||
|
config := Config{TodoAddr: addr, TodoTag: "expense"}
|
||||||
|
xaction := func() *Transaction {
|
||||||
|
return &Transaction{
|
||||||
|
ID: "id",
|
||||||
|
Bank: UCCU,
|
||||||
|
Amount: "1.23",
|
||||||
|
Vendor: "vendor vendor",
|
||||||
|
Date: "today",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
t.Run("full file", func(t *testing.T) {
|
||||||
|
if err := ioutil.WriteFile(addr, []byte(`
|
||||||
|
todo:
|
||||||
|
- first
|
||||||
|
- todo: second
|
||||||
|
scheduled: []
|
||||||
|
done: []
|
||||||
|
`), os.ModePerm); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
err := uploadPTTodo(config, xaction())
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
b, err := ioutil.ReadFile(addr)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
if bytes.Compare(bytes.TrimSpace(b), bytes.TrimSpace([]byte(`
|
||||||
|
todo:
|
||||||
|
- {"todo":"(today) /UCCU: 1.23 @ vendor vendor", "tags":"expense"}
|
||||||
|
- first
|
||||||
|
- todo: second
|
||||||
|
scheduled: []
|
||||||
|
done: []
|
||||||
|
`))) != 0 {
|
||||||
|
t.Errorf("full file came out wrong: got %s", b)
|
||||||
|
}
|
||||||
|
if !bytes.Contains(b, []byte(xaction().Format())) {
|
||||||
|
t.Errorf("full file didnt get target: %s", string(b))
|
||||||
|
}
|
||||||
|
t.Logf("%s", b)
|
||||||
|
})
|
||||||
|
t.Run("no file", func(t *testing.T) {
|
||||||
|
os.Remove(addr)
|
||||||
|
err := uploadPTTodo(config, xaction())
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
b, err := ioutil.ReadFile(addr)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
if !bytes.Contains(b, []byte(xaction().Format())) {
|
||||||
|
t.Errorf("no file didnt get target: %s", string(b))
|
||||||
|
}
|
||||||
|
t.Logf("%s", b)
|
||||||
|
})
|
||||||
|
t.Run("empty file", func(t *testing.T) {
|
||||||
|
if err := ioutil.WriteFile(addr, []byte{}, os.ModePerm); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
err := uploadPTTodo(config, xaction())
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
b, err := ioutil.ReadFile(addr)
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
if !bytes.Contains(b, []byte(xaction().Format())) {
|
||||||
|
t.Errorf("empty file didnt get target: %s", string(b))
|
||||||
|
}
|
||||||
|
t.Logf("%s", b)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func TestUploadLedger(t *testing.T) {
|
func TestUploadLedger(t *testing.T) {
|
||||||
cases := map[string]struct {
|
cases := map[string]struct {
|
||||||
transaction Transaction
|
transaction Transaction
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue