bel 2023-10-27 09:16:54 -06:00
parent 037d0a5efc
commit 2dbe5fa0fe
1 changed files with 107 additions and 76 deletions

View File

@ -64,6 +64,21 @@ func main() {
like = append(like, ledger.LikeAfter(*likeAfter)) like = append(like, ledger.LikeAfter(*likeAfter))
foo := func(w http.ResponseWriter, r *http.Request) { foo := func(w http.ResponseWriter, r *http.Request) {
switch r.URL.Path {
case "/bal", "/reg":
case "/ui":
f, err := os.Open("./index.html")
if err != nil {
panic(err)
}
defer f.Close()
io.Copy(w, f)
return
default:
http.NotFound(w, r)
return
}
foolike := make(ledger.Likes, 0) foolike := make(ledger.Likes, 0)
for _, v := range r.URL.Query()["likeName"] { for _, v := range r.URL.Query()["likeName"] {
foolike = append(foolike, ledger.LikeName(v)) foolike = append(foolike, ledger.LikeName(v))
@ -79,6 +94,7 @@ func main() {
} }
register := deltas.Like(foolike...).Register() register := deltas.Like(foolike...).Register()
predicted := make(ledger.Register)
bpis := maps.Clone(bpis) bpis := maps.Clone(bpis)
// MODIFIERS // MODIFIERS
@ -103,8 +119,8 @@ func main() {
panic(k) panic(k)
} }
} }
predicted := prediction.Predict(register, window) predicted = prediction.Predict(register, window)
register.PushAll(predicted) // TODO draw line separately
for _, currencyRate := range r.URL.Query()["predictFixedGrowth"] { for _, currencyRate := range r.URL.Query()["predictFixedGrowth"] {
currency := strings.Split(currencyRate, "=")[0] currency := strings.Split(currencyRate, "=")[0]
rate, err := strconv.ParseFloat(strings.Split(currencyRate, "=")[1], 64) rate, err := strconv.ParseFloat(strings.Split(currencyRate, "=")[1], 64)
@ -117,94 +133,92 @@ func main() {
} }
} }
} }
if r.URL.Query().Get("bpi") != "" { if r.URL.Query().Get("bpi") != "" {
register = register.WithBPIs(bpis) register = register.WithBPIs(bpis)
predicted = predicted.WithBPIs(bpis)
} }
if zoomStart, err := time.ParseInLocation("2006-01", r.URL.Query().Get("zoomStart"), time.Local); err == nil { if zoomStart, err := time.ParseInLocation("2006-01", r.URL.Query().Get("zoomStart"), time.Local); err == nil {
register = register.Between(zoomStart, time.Now().Add(time.Hour*24*365*100)) register = register.Between(zoomStart, time.Now().Add(time.Hour*24*365*100))
predicted = predicted.Between(zoomStart, time.Now().Add(time.Hour*24*365*100))
} }
// /MODIFIERS // /MODIFIERS
nameCurrencyDateValue := map[string]map[ledger.Currency]map[string]float64{} toChart := func(cumulative bool, display string, register ledger.Register) Chart {
for date, balances := range register { nameCurrencyDateValue := map[string]map[ledger.Currency]map[string]float64{}
for name, balance := range balances { for date, balances := range register {
for currency, value := range balance { for name, balance := range balances {
if _, ok := nameCurrencyDateValue[name]; !ok { for currency, value := range balance {
nameCurrencyDateValue[name] = make(map[ledger.Currency]map[string]float64) if _, ok := nameCurrencyDateValue[name]; !ok {
} nameCurrencyDateValue[name] = make(map[ledger.Currency]map[string]float64)
if _, ok := nameCurrencyDateValue[name][currency]; !ok {
nameCurrencyDateValue[name][currency] = make(map[string]float64)
}
nameCurrencyDateValue[name][currency][date] += value
}
}
}
chart := NewChart("line")
if v := r.URL.Query().Get("chart"); v != "" {
chart = NewChart(v)
}
dates := register.Dates()
names := register.Names()
chart.AddX(dates)
switch r.URL.Path {
default:
http.NotFound(w, r)
return
case "/ui":
f, err := os.Open("./index.html")
if err != nil {
panic(err)
}
defer f.Close()
io.Copy(w, f)
return
case "/bal":
for _, name := range names {
currencyDateValue := nameCurrencyDateValue[name]
for currency, dateValue := range currencyDateValue {
series := make([]int, len(dates))
for i := range dates {
var lastValue float64
for j := range dates[:i+1] {
if newLastValue, ok := dateValue[dates[j]]; ok {
lastValue = newLastValue
}
} }
series[i] = int(lastValue) if _, ok := nameCurrencyDateValue[name][currency]; !ok {
} nameCurrencyDateValue[name][currency] = make(map[string]float64)
key := fmt.Sprintf("%s (%s)", name, currency)
if slices.Min(series) != 0 || slices.Max(series) != 0 {
chart.AddY(key, series)
}
}
}
case "/reg":
for _, name := range names {
currencyDateValue := nameCurrencyDateValue[name]
for currency, dateValue := range currencyDateValue {
series := make([]int, len(dates))
for i := range dates {
var prevValue float64
var lastValue float64
for j := range dates[:i+1] {
if newLastValue, ok := dateValue[dates[j]]; ok {
prevValue = lastValue
lastValue = newLastValue
}
} }
series[i] = int(lastValue - prevValue) nameCurrencyDateValue[name][currency][date] += value
}
key := fmt.Sprintf("%s (%s)", name, currency)
if slices.Min(series) != 0 || slices.Max(series) != 0 {
chart.AddY(key, series)
} }
} }
} }
chart := NewChart("line")
if v := display; v != "" {
chart = NewChart(v)
}
dates := register.Dates()
names := register.Names()
chart.AddX(dates)
if cumulative {
for _, name := range names {
currencyDateValue := nameCurrencyDateValue[name]
for currency, dateValue := range currencyDateValue {
series := make([]int, len(dates))
for i := range dates {
var lastValue float64
for j := range dates[:i+1] {
if newLastValue, ok := dateValue[dates[j]]; ok {
lastValue = newLastValue
}
}
series[i] = int(lastValue)
}
key := fmt.Sprintf("%s (%s)", name, currency)
if slices.Min(series) != 0 || slices.Max(series) != 0 {
chart.AddY(key, series)
}
}
}
} else {
for _, name := range names {
currencyDateValue := nameCurrencyDateValue[name]
for currency, dateValue := range currencyDateValue {
series := make([]int, len(dates))
for i := range dates {
var prevValue float64
var lastValue float64
for j := range dates[:i+1] {
if newLastValue, ok := dateValue[dates[j]]; ok {
prevValue = lastValue
lastValue = newLastValue
}
}
series[i] = int(lastValue - prevValue)
}
key := fmt.Sprintf("%s (%s)", name, currency)
if slices.Min(series) != 0 || slices.Max(series) != 0 {
chart.AddY(key, series)
}
}
}
}
return chart
} }
if err := chart.Render(w); err != nil { primary := toChart(r.URL.Path == "/bal", r.URL.Query().Get("chart"), register)
if len(predicted) > 0 {
primary.Overlap(toChart(r.URL.Path == "/bal", "line", predicted))
}
if err := primary.Render(w); err != nil {
panic(err) panic(err)
} }
} }
@ -277,6 +291,7 @@ type Chart interface {
AddX(interface{}) AddX(interface{})
AddY(string, []int) AddY(string, []int)
Render(io.Writer) error Render(io.Writer) error
Overlap(Chart)
} }
func NewChart(name string) Chart { func NewChart(name string) Chart {
@ -312,6 +327,14 @@ func (line Line) AddY(name string, v []int) {
line.AddSeries(name, y) line.AddSeries(name, y)
} }
func (line Line) Overlap(other Chart) {
overlapper, ok := other.(charts.Overlaper)
if !ok {
panic(fmt.Sprintf("cannot overlap %T", other))
}
line.Line.Overlap(overlapper)
}
type Bar struct { type Bar struct {
*charts.Bar *charts.Bar
} }
@ -332,6 +355,14 @@ func (bar Bar) AddY(name string, v []int) {
bar.AddSeries(name, y) bar.AddSeries(name, y)
} }
func (bar Bar) Overlap(other Chart) {
overlapper, ok := other.(charts.Overlaper)
if !ok {
panic(fmt.Sprintf("cannot overlap %T", other))
}
bar.Bar.Overlap(overlapper)
}
type Stack struct { type Stack struct {
Bar Bar
} }