package cli import ( "flag" "fmt" "io" "os" "slices" "strings" "gogs.inhome.blapointe.com/ana-ledger/src/ana" "gogs.inhome.blapointe.com/ana-ledger/src/ledger" ) func Main() { var config Config fs := flag.NewFlagSet(os.Args[0], flag.ContinueOnError) fs.Var(&config.Files, "f", "paths to files") fs.Var(&config.Query.Period, "period", "period") fs.StringVar(&config.Query.Sort, "S", "", "sort ie date") fs.BoolVar(&config.Query.NoRounding, "no-rounding", false, "no rounding") fs.IntVar(&config.Query.Depth, "depth", 0, "depth grouping") fs.BoolVar(&config.Query.Reverse, "r", false, "reverse printed accounts") fs.BoolVar(&config.Query.Normalize, "n", false, "normalize with default normalizer") fs.BoolVar(&config.Query.NoExchanging, "no-exchanging", true, "omit currency exchanges") if err := fs.Parse(os.Args[1:]); err != nil { panic(err) } files := config.Files.Strings() if len(files) == 0 { panic("must specify at least one file") } ledgerFiles, err := ledger.NewFiles(files[0], files[1:]...) if err != nil { panic(err) } positional := fs.Args() if len(positional) == 0 || len(positional[0]) < 3 { panic("positional arguments required, ie bal|reg PATTERN MATCHING") } cmd := positional[0] q, err := BuildQuery(positional[1:]) if err != nil { panic(err) } deltas, err := ledgerFiles.Deltas() if err != nil { panic(err) } if period := config.Query.Period; !period.Empty() { after := period.Start.Format("2006-01-02") before := period.Stop.Format("2006-01-02") deltas = deltas.Like( ledger.LikeAfter(after), ledger.LikeBefore(before), ) } pattern := ".*" if depth := config.Query.Depth; depth > 0 { pattern = "" for i := 0; i < depth; i++ { pattern += "[^:]*:" } pattern = strings.Trim(pattern, ":") } group := ledger.GroupName(pattern) if config.Query.Normalize { deltas = ana.NewDefaultNormalizer().Normalize(deltas) } switch cmd[:3] { case "bal": /* balances := deltas.Balances() if likePattern != "" { balances = balances.Like(likePattern) } if notLikePattern != "" { balances = balances.NotLike(notLikePattern) } FPrintBalances(os.Stdout, "", balances, nil) */ case "reg": transactions := deltas.Transactions() for i, transaction := range transactions { balances := ledger.Deltas(transaction).Like(q).Group(group).Balances() shouldPrint := false shouldPrint = shouldPrint || len(balances) > 2 if config.Query.NoExchanging { shouldPrint = shouldPrint || len(balances) > 1 for _, v := range balances { shouldPrint = shouldPrint || len(v) == 1 } } else { shouldPrint = shouldPrint || len(balances) > 0 } if shouldPrint { fmt.Printf("%s\t%s\n", transaction[0].Date, transaction[0].Description) FPrintBalances(os.Stdout, "\t\t", balances, transactions[:i+1].Deltas().Like(q).Group(group).Balances()) } } default: panic("unknown command " + positional[0]) } } func FPrintBalances(w io.Writer, linePrefix string, balances, cumulatives ledger.Balances) { keys := []string{} for k := range balances { keys = append(keys, k) } slices.Sort(keys) max := 0 for _, k := range keys { if n := len(k); n > max { max = n } } format := fmt.Sprintf("%s%%-%ds\t%%s%%.2f (%%s%%.2f)\n", linePrefix, max) for _, key := range keys { currencies := []ledger.Currency{} for currency := range balances[key] { currencies = append(currencies, currency) } slices.Sort(currencies) for _, currency := range currencies { printableCurrency := currency if printableCurrency != "$" { printableCurrency += " " } cumulative := balances[key][currency] if balance, ok := cumulatives[key]; !ok { } else if value, ok := balance[currency]; !ok { } else { cumulative = value } fmt.Fprintf(w, format, key, printableCurrency, balances[key][currency], printableCurrency, cumulative) } } }