aboutsummaryrefslogtreecommitdiffstats
path: root/journal/journal.go
blob: edb79d80f1c923cd755dbee5a6c5582702b8f4e9 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
package journal

import (
	"encoding/csv"
	"fmt"
	"io"
	"os"
	"os/user"
	"path/filepath"
	"regexp"
	"strconv"
	"strings"
	"time"

	"github.com/BurntSushi/toml"
	"github.com/mpolden/journal/record"
	"github.com/mpolden/journal/record/bulder"
	"github.com/mpolden/journal/record/dnb"
	"github.com/mpolden/journal/record/komplett"
	"github.com/mpolden/journal/record/morrow"
	"github.com/mpolden/journal/record/norwegian"
	"github.com/mpolden/journal/sql"
)

// Account represents a financial account.
type Account struct {
	Number string
	Name   string
}

// Group represents a group configuration which decides how records should be assorted into groups.
type Group struct {
	Name     string
	Account  string
	Budget   int64
	Budgets  [12]int64
	Patterns []string
	patterns []*regexp.Regexp
	IDs      []string
	Discard  bool
}

// Config represents a journal's configuration.
type Config struct {
	Database     string
	Comma        string
	DefaultGroup string
	Accounts     []Account
	Groups       []Group
}

// Journal implements a journal of financial records.
type Journal struct {
	accounts     []Account
	groups       []Group
	db           *sql.Client
	Comma        string
	DefaultGroup string
	Discarding   bool
}

// Writes represents statistics of a journal's updates.
type Writes struct {
	Account int64
	Record  int64
}

func (c *Config) load() error {
	if len(c.Database) == 0 {
		return fmt.Errorf("invalid database path: %q", c.Database)
	}
	if c.Database[0] == '~' {
		if len(c.Database) > 1 && c.Database[1] != '/' {
			return fmt.Errorf("invalid database path: %q", c.Database)
		}
		user, err := user.Current()
		if err != nil {
			return err
		}
		c.Database = filepath.Join(user.HomeDir, c.Database[1:])
	}
	for _, a := range c.Accounts {
		if len(a.Number) == 0 {
			return fmt.Errorf("invalid account number: %q", a.Number)
		}
	}
	for i, g := range c.Groups {
		if len(g.Name) == 0 {
			return fmt.Errorf("invalid group name: %q", g.Name)
		}
		for _, pattern := range g.Patterns {
			if len(pattern) == 0 {
				return fmt.Errorf("group: %q: invalid pattern: %q", g.Name, pattern)
			}
			p, err := regexp.Compile(pattern)
			if err != nil {
				return err
			}
			c.Groups[i].patterns = append(c.Groups[i].patterns, p)
		}

	}
	return nil
}

func readConfig(r io.Reader) (Config, error) {
	var conf Config
	_, err := toml.DecodeReader(r, &conf)
	return conf, err
}

func readerFrom(r io.Reader, name, filename string) (record.Reader, error) {
	var rr record.Reader
	switch name {
	case "bulder":
		rr = bulder.NewReader(r)
	case "dnb":
		rr = dnb.NewReader(r)
	case "csv":
		rr = record.NewReader(r)
	case "komplett":
		rr = komplett.NewReader(r)
	case "morrow":
		rr = morrow.NewReader(r)
	case "norwegian":
		rr = norwegian.NewReader(r)
	case "auto":
		ext := filepath.Ext(filename)
		switch ext {
		case ".csv":
			rr = record.NewReader(r)
		case ".xlsx":
			rr = norwegian.NewReader(r)
		case ".json":
			rr = komplett.NewReader(r)
		default:
			return nil, fmt.Errorf("failed to guess reader for file name: %s", filename)
		}
	default:
		return nil, fmt.Errorf("invalid reader: %q", name)
	}
	return rr, nil
}

// FromConfig creates a new journal from a configuration file located at name.
func FromConfig(name string) (*Journal, error) {
	if name == "~/.journalrc" {
		home := os.Getenv("HOME")
		name = filepath.Join(home, ".journalrc")
	}
	f, err := os.Open(name)
	if err != nil {
		return nil, err
	}
	defer f.Close()
	conf, err := readConfig(f)
	if err != nil {
		return nil, err
	}
	return New(conf)
}

// New creates a new journal from the given configuration.
func New(conf Config) (*Journal, error) {
	if err := conf.load(); err != nil {
		return nil, err
	}
	db, err := sql.New(conf.Database)
	if err != nil {
		return nil, err
	}
	comma := conf.Comma
	if comma == "" {
		comma = "."
	}
	defaultGroup := conf.DefaultGroup
	if defaultGroup == "" {
		defaultGroup = "* ungrouped *"
	}
	return &Journal{
		db:           db,
		accounts:     conf.Accounts,
		groups:       conf.Groups,
		Comma:        comma,
		DefaultGroup: defaultGroup,
		Discarding:   true,
	}, nil
}

// FormatAmount formats number n as a financial amount.
func (j *Journal) FormatAmount(n int64) string {
	i := n / 100
	f := n % 100
	var sb strings.Builder
	if f < 0 {
		f = -f
		if i == 0 {
			sb.WriteRune('-')
		}
	}
	sb.WriteString(strconv.FormatInt(i, 10))
	sb.WriteString(j.Comma)
	sb.WriteString(fmt.Sprintf("%02d", f))
	return sb.String()
}

// ReadFile uses reader to read records from file f.
func (j *Journal) ReadFile(reader string, f *os.File) ([]record.Record, error) {
	r, err := readerFrom(f, reader, f.Name())
	if err != nil {
		return nil, err
	}
	return r.Read()
}

// Export writes periods to writer w using CSV-encoding. The timeLayout defines the format of time fields.
func (j *Journal) Export(w io.Writer, periods []record.Period, timeLayout string) error {
	csv := csv.NewWriter(w)
	for _, p := range periods {
		for _, rg := range p.Groups {
			r := []string{p.Time.Format(timeLayout), rg.Name, j.FormatAmount(rg.Sum())}
			if err := csv.Write(r); err != nil {
				return err
			}
		}
	}
	csv.Flush()
	return csv.Error()
}

func (j *Journal) writeAccounts() (int64, error) {
	as := make([]sql.Account, len(j.accounts))
	for i, a := range j.accounts {
		as[i] = sql.Account{Number: a.Number, Name: a.Name}
	}
	return j.db.AddAccounts(as)
}

// Accounts returns all accounts in the journal
func (j *Journal) Accounts() ([]record.Account, error) {
	as, err := j.db.SelectAccounts("")
	if err != nil {
		return nil, err
	}
	accounts := make([]record.Account, len(as))
	for i, a := range as {
		accounts[i] = record.Account{
			Name:    a.Name,
			Number:  a.Number,
			Records: a.Records,
		}
	}
	return accounts, nil
}

// Write writes records for accountNumber into the journal.
func (j *Journal) Write(accountNumber string, records []record.Record) (Writes, error) {
	var writes Writes
	n, err := j.writeAccounts()
	if err != nil {
		return writes, err
	}
	writes.Account = n
	rs := make([]sql.Record, len(records))
	for i, r := range records {
		rs[i] = sql.Record{Time: r.Time.Unix(), Text: r.Text, Amount: r.Amount, Balance: r.Balance}
	}
	n, err = j.db.AddRecords(accountNumber, rs)
	writes.Record = n
	return writes, err
}

// Read reads records for accountNumber between the times since and until from the journal.
func (j *Journal) Read(accountNumber string, since, until time.Time) ([]record.Record, error) {
	rs, err := j.db.SelectRecordsBetween(accountNumber, since, until)
	if err != nil {
		return nil, err
	}
	records := make([]record.Record, len(rs))
	for i, r := range rs {
		records[i] = record.Record{
			Account: record.Account{Number: r.Account.Number, Name: r.Account.Name},
			Time:    time.Unix(r.Time, 0).UTC(),
			Text:    r.Text,
			Amount:  r.Amount,
		}
	}
	return records, nil
}

// Assort assorts records into groups using this journal's configuration.
func (j *Journal) Assort(records []record.Record) []record.Group {
	return record.AssortFunc(records, j.findGroup)
}

// AssortPeriod assorts record groups into time periods using timeFn.
func (j *Journal) AssortPeriod(records []record.Record, timeFn func(time.Time) time.Time) []record.Period {
	return record.AssortPeriodFunc(records, timeFn, j.findGroup)
}

func (j *Journal) findGroup(r record.Record) *record.Group {
	for _, g := range j.groups {
		if g.Account != "" && g.Account != r.Account.Number {
			continue
		}
		for _, id := range g.IDs {
			if r.ID() != id {
				continue
			}
			if j.Discarding && g.Discard {
				return nil
			}
			rg := record.NewGroup(g.Name, record.Budget{
				Default: g.Budget,
				Months:  g.Budgets,
			})
			return &rg

		}
	}
	for _, g := range j.groups {
		if g.Account != "" && g.Account != r.Account.Number {
			continue
		}
		for _, p := range g.patterns {
			if !p.MatchString(r.Text) {
				continue
			}
			if j.Discarding && g.Discard {
				return nil
			}
			rg := record.NewGroup(g.Name, record.Budget{
				Default: g.Budget,
				Months:  g.Budgets,
			})
			return &rg
		}
	}
	return &record.Group{Name: j.DefaultGroup}
}