-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwells.go
104 lines (97 loc) · 2.82 KB
/
wells.go
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
package goldmachine
import (
"bytes"
"encoding/csv"
"errors"
"io"
"io/ioutil"
"strconv"
"time"
"github.com/Rhymond/go-money"
)
// converts Wells Fargo transaction data to Ledger CLi data
func ParseCheckingCSV(in string) ([]JournalEntry, error) {
return convertToJournalEntries(in, func(in []string) (JournalEntry, error) {
je := &JournalEntry{
AccountEntries: []AccountEntry{},
}
// "06/28/2019","0.41","*","","INTEREST PAYMENT"
effective, err := time.Parse("01/02/2006", in[0])
if err != nil {
return *je, err
}
je.EffectiveAt = effective
amount, err := strconv.ParseFloat(in[1], 64)
if err != nil {
return *je, err
}
moneyAmount := money.New(int64(amount*100), "USD")
memo := in[4]
if amount < 0 {
je.AddEntry(NewDebitEntry(moneyAmount.Absolute(), accountFromPatterns(memo, checkingPatterns), memo))
je.AddEntry(NewCreditEntry(moneyAmount.Absolute(), CHECKING_ACCOUNT, memo))
} else {
je.AddEntry(NewCreditEntry(moneyAmount, accountFromPatterns(memo, checkingPatterns), memo))
je.AddEntry(NewDebitEntry(moneyAmount, CHECKING_ACCOUNT, memo))
}
je.Identifier = memo
return *je, nil
})
}
func ParseCreditCardCSV(in string) ([]JournalEntry, error) {
return convertToJournalEntries(in, func(in []string) (JournalEntry, error) {
// "01/09/2019","-28.13","*","","4505 BURGERS AND B SAN FRANCISCOCA"
// actually identical to checking accounts but we keep them split in case there's future diversions
je := &JournalEntry{
AccountEntries: []AccountEntry{},
}
effective, err := time.Parse("01/02/2006", in[0])
if err != nil {
return *je, err
}
je.EffectiveAt = effective
amount, err := strconv.ParseFloat(in[1], 64)
if err != nil {
return *je, err
}
moneyAmount := money.New(int64(amount*100), "USD")
memo := in[4]
if amount < 0 {
je.AddEntry(NewDebitEntry(moneyAmount.Absolute(), accountFromPatterns(memo, creditCardPatterns), memo))
je.AddEntry(NewCreditEntry(moneyAmount.Absolute(), CREDIT_CARD, memo))
} else {
je.AddEntry(NewCreditEntry(moneyAmount, accountFromPatterns(memo, creditCardPatterns), memo))
je.AddEntry(NewDebitEntry(moneyAmount, CREDIT_CARD, memo))
}
je.Identifier = memo
return *je, nil
})
}
var Skip error = errors.New("Skip this")
func convertToJournalEntries(in string, convert func([]string) (JournalEntry, error)) ([]JournalEntry, error) {
byteContent, err := ioutil.ReadFile(in)
if err != nil {
return nil, err
}
ret := []JournalEntry{}
r := csv.NewReader(bytes.NewBuffer(byteContent))
for {
record, err := r.Read()
if err == io.EOF {
break
} else if err != nil {
return nil, err
}
je, err := convert(record)
if err != nil && err != Skip {
return nil, err
} else if err == Skip {
continue
}
if err = je.Validate(); err != nil {
return nil, err
}
ret = append(ret, je)
}
return ret, nil
}