use new internal package types and logic

This commit is contained in:
2025-11-10 14:04:51 +00:00
parent dca43fe014
commit a274718f0b

114
main.go
View File

@@ -2,14 +2,15 @@ package main
import ( import (
"container/list" "container/list"
"encoding/csv"
"errors" "errors"
"fmt" "fmt"
"io" "io"
"log/slog" "log/slog"
"math/big" "math/big"
"os" "os"
"strings"
"git.naterciomoniz.net/applications/broker2anexoj/internal"
"git.naterciomoniz.net/applications/broker2anexoj/internal/trading212"
) )
func main() { func main() {
@@ -25,7 +26,7 @@ func run() error {
return fmt.Errorf("open statement: %w", err) return fmt.Errorf("open statement: %w", err)
} }
r := NewRecordReader(f) r := trading212.NewRecordReader(f)
assets := make(map[string]*list.List) assets := make(map[string]*list.List)
for { for {
@@ -37,8 +38,8 @@ func run() error {
return fmt.Errorf("read statement record: %w", err) return fmt.Errorf("read statement record: %w", err)
} }
switch record.Direction() { switch record.Side() {
case DirectionBuy: case internal.SideBuy:
lst, ok := assets[record.Symbol()] lst, ok := assets[record.Symbol()]
if !ok { if !ok {
lst = list.New() lst = list.New()
@@ -46,7 +47,7 @@ func run() error {
} }
lst.PushBack(record) lst.PushBack(record)
case DirectionSell: case internal.SideSell:
lst, ok := assets[record.Symbol()] lst, ok := assets[record.Symbol()]
if !ok { if !ok {
return ErrSellWithoutBuy return ErrSellWithoutBuy
@@ -61,7 +62,7 @@ func run() error {
return ErrSellWithoutBuy return ErrSellWithoutBuy
} }
next, ok := front.Value.(Record) next, ok := front.Value.(internal.Record)
if !ok { if !ok {
return fmt.Errorf("unexpected record type: %T", front) return fmt.Errorf("unexpected record type: %T", front)
} }
@@ -86,7 +87,7 @@ func run() error {
} }
default: default:
return fmt.Errorf("unknown direction: %s", record.Direction()) return fmt.Errorf("unknown side: %s", record.Side())
} }
} }
@@ -96,100 +97,3 @@ func run() error {
} }
var ErrSellWithoutBuy = fmt.Errorf("found sell without bought volume") var ErrSellWithoutBuy = fmt.Errorf("found sell without bought volume")
type Record struct {
symbol string
direction Direction
quantity *big.Float
price *big.Float
}
func (r Record) Symbol() string {
return r.symbol
}
func (r Record) Direction() Direction {
return r.direction
}
func (r Record) Quantity() *big.Float {
return r.quantity
}
func (r Record) Price() *big.Float {
return r.price
}
type RecordReader struct {
reader *csv.Reader
}
func NewRecordReader(r io.Reader) *RecordReader {
return &RecordReader{
reader: csv.NewReader(r),
}
}
func (rr RecordReader) ReadRecord() (Record, error) {
for {
raw, err := rr.reader.Read()
if err != nil {
return Record{}, fmt.Errorf("read record: %w", err)
}
var dir Direction
switch strings.ToLower(raw[0]) {
case "market buy":
dir = DirectionBuy
case "market sell":
dir = DirectionSell
case "action", "stock split open", "stock split close":
continue
default:
return Record{}, fmt.Errorf("unhandled record: %s", raw[0])
}
qant, _, err := big.ParseFloat(raw[6], 10, 20, big.ToZero)
if err != nil {
return Record{}, fmt.Errorf("parse quantity: %w", err)
}
price, _, err := big.ParseFloat(raw[7], 10, 20, big.ToZero)
if err != nil {
return Record{}, fmt.Errorf("parse price: %w", err)
}
return Record{
symbol: raw[2],
direction: dir,
quantity: qant,
price: price,
}, nil
}
}
type Direction uint
const (
DirectionUnknown Direction = 0
DirectionBuy = 1
DirectionSell = 2
)
func (d Direction) String() string {
switch d {
case 1:
return "buy"
case 2:
return "sell"
default:
return "unknown"
}
}
func (d Direction) IsBuy() bool {
return d == DirectionBuy
}
func (d Direction) IsSell() bool {
return d == DirectionSell
}