monobank-firefly3-bot/main.go

262 lines
8.4 KiB
Go
Raw Normal View History

2024-03-25 11:55:01 +02:00
package main
import (
"context"
"encoding/json"
"fmt"
"github.com/antihax/optional"
"github.com/joho/godotenv"
"io"
"log"
"main/firefly3"
"main/monobank/api/webhook/models"
2024-03-26 11:26:35 +02:00
"math"
2024-03-25 11:55:01 +02:00
"net/http"
"os"
2024-03-26 12:39:04 +02:00
"slices"
2024-03-25 11:55:01 +02:00
"strconv"
"time"
)
// https://api.monobank.ua/docs/index.html#tag/Kliyentski-personalni-dani/paths/~1personal~1statement~1{account}~1{from}~1{to}/get
// https://api-docs.firefly-iii.org/#/accounts/listAccount
2024-03-26 19:06:31 +02:00
// curl -X POST https://api.monobank.ua/personal/webhook -H 'Content-TransactionType: application/json' -H 'X-Token: ' -d '{"webHookUrl":"https://monobank-firefly3.stuzer.link/webhook"}'
2024-03-25 11:55:01 +02:00
2024-03-26 19:06:31 +02:00
// curl -X POST https://monobank-firefly3.stuzer.link/webhook -H 'Content-TransactionType: application/json' -d '{"test":123}'
2024-03-25 11:55:01 +02:00
2024-03-26 19:06:31 +02:00
var ShopConfig []ShopConfigItem
2024-03-26 11:26:35 +02:00
2024-03-25 11:55:01 +02:00
func handleWebhook(w http.ResponseWriter, r *http.Request) {
2024-03-26 19:06:31 +02:00
LogString("-----------------\nwebhook received!")
2024-03-25 11:55:01 +02:00
// read body bytes
body, err := io.ReadAll(r.Body)
if err != nil {
2024-03-26 19:06:31 +02:00
LogString(err.Error())
2024-03-25 11:55:01 +02:00
w.WriteHeader(http.StatusOK)
return
}
//fmt.Println(string(body))
//w.WriteHeader(http.StatusOK)
//return
//body = []byte("{\"type\":\"StatementItem\",\"data\":{\"account\":\"4723djMLsLOCzhoeYjxqRw\",\"statementItem\":{\"id\":\"5_NQ0arGAmp2pyNzvA\",\"time\":1711544958,\"description\":\"Ілля Ш.\",\"mcc\":4829,\"originalMcc\":4829,\"amount\":-572000,\"operationAmount\":-572000,\"currencyCode\":980,\"commissionRate\":22000,\"cashbackAmount\":0,\"balance\":8101246,\"hold\":true,\"receiptId\":\"EMXC-P266-90PC-EB8C\"}}}")
2024-03-26 19:06:31 +02:00
LogString(string(body))
2024-03-25 11:55:01 +02:00
2024-03-26 12:39:04 +02:00
if len(string(body)) == 0 {
2024-03-26 19:06:31 +02:00
LogString("empty body")
2024-03-26 12:39:04 +02:00
w.WriteHeader(http.StatusOK)
return
}
2024-03-25 11:55:01 +02:00
// parse body
var transaction models.Transaction
err = json.Unmarshal(body, &transaction)
if err != nil {
2024-03-26 19:06:31 +02:00
LogString(err.Error())
2024-03-25 11:55:01 +02:00
w.WriteHeader(http.StatusOK)
return
}
//statement, err := requests.Statement(models2.StatementRequest{
// Account: transaction.Data.Account,
// From: transaction.Data.StatementItem.Time,
// To: transaction.Data.StatementItem.Time,
//})
//if err != nil {
// fmt.Printf("%+v", err.Error())
// w.WriteHeader(http.StatusOK)
// return
//}
//fmt.Printf("%+v", statement)
2024-03-25 11:55:01 +02:00
// init firefly3 client
clientConf := firefly3.NewConfiguration()
2024-03-26 11:26:35 +02:00
clientConf.BasePath = os.Getenv("FIREFLY3_API_URL")
2024-03-25 11:55:01 +02:00
clientConf.AddDefaultHeader("Authorization", "Bearer "+os.Getenv("FIREFLY3_TOKEN"))
client := firefly3.NewAPIClient(clientConf)
ctx := context.Background()
2024-03-26 11:26:35 +02:00
// get firefly3 account
2024-03-25 11:55:01 +02:00
listOpts := firefly3.AccountsApiListAccountOpts{
Type_: optional.NewInterface("asset"),
}
accounts, _, err := client.AccountsApi.ListAccount(ctx, &listOpts)
if err != nil {
2024-03-26 19:06:31 +02:00
LogString(err.Error())
2024-03-26 11:26:35 +02:00
w.WriteHeader(http.StatusOK)
return
2024-03-25 11:55:01 +02:00
}
var account firefly3.AccountRead
for _, row := range accounts.Data {
if row.Attributes.Notes == transaction.Data.Account {
account = row
}
}
if len(account.Id) == 0 {
2024-03-26 19:06:31 +02:00
LogString("unable to find account " + transaction.Data.Account + " in firefly3")
2024-03-25 11:55:01 +02:00
w.WriteHeader(http.StatusOK)
return
}
2024-03-26 12:39:04 +02:00
// create transaction
2024-03-26 19:36:29 +02:00
var firefly3Transactions []firefly3.TransactionSplitStore
2024-03-26 19:06:31 +02:00
transactionTypeWithdrawal := firefly3.WITHDRAWAL_TransactionTypeProperty
transactionTypeTransfer := firefly3.TRANSFER_TransactionTypeProperty
2024-03-26 12:40:23 +02:00
firefly3Transaction := firefly3.TransactionSplitStore{
2024-03-26 11:26:35 +02:00
Type_: &transactionTypeWithdrawal,
2024-03-26 12:39:04 +02:00
Date: time.Unix(int64(transaction.Data.StatementItem.Time), 0).Add(time.Hour * 2),
2024-03-26 11:26:35 +02:00
Notes: string(body),
Amount: strconv.Itoa(int(math.Abs(math.Round(float64(transaction.Data.StatementItem.Amount/100)))) - int(math.Abs(math.Round(float64(transaction.Data.StatementItem.CommissionRate/100))))),
2024-03-26 11:26:35 +02:00
SourceId: account.Id,
}
2024-03-26 12:39:04 +02:00
if slices.Contains([]string{"З чорної картки"}, transaction.Data.StatementItem.Description) {
2024-03-26 12:40:23 +02:00
firefly3Transaction.Type_ = &transactionTypeTransfer
firefly3Transaction.Description = "Transfer between accounts"
firefly3Transaction.DestinationId = "60"
2024-03-26 19:36:29 +02:00
firefly3Transactions = append(firefly3Transactions, firefly3Transaction)
2024-03-26 12:39:04 +02:00
} else if slices.Contains([]string{"З білої картки"}, transaction.Data.StatementItem.Description) {
2024-03-26 12:40:23 +02:00
firefly3Transaction.Type_ = &transactionTypeTransfer
firefly3Transaction.Description = "Transfer between accounts"
firefly3Transaction.DestinationId = "1"
2024-03-26 19:36:29 +02:00
firefly3Transactions = append(firefly3Transactions, firefly3Transaction)
2024-03-26 19:06:31 +02:00
} else {
for _, row := range ShopConfig {
if slices.Contains(row.Names, transaction.Data.StatementItem.Description) || slices.Contains(row.MCCCodes, transaction.Data.StatementItem.Mcc) {
firefly3Transaction.Description = row.TransactionDescription
firefly3Transaction.DestinationName = row.TransactionDestination
firefly3Transaction.CategoryName = row.TransactionCategory
2024-03-26 19:36:29 +02:00
firefly3Transactions = append(firefly3Transactions, firefly3Transaction)
2024-03-26 19:06:31 +02:00
break
}
}
2024-03-25 11:55:01 +02:00
}
2024-03-26 19:36:29 +02:00
// record transfer fee
if transaction.Data.StatementItem.CommissionRate > 0 {
firefly3Transactions = append(firefly3Transactions, firefly3.TransactionSplitStore{
Type_: &transactionTypeWithdrawal,
Date: time.Unix(int64(transaction.Data.StatementItem.Time), 0).Add(time.Hour * 2),
Notes: string(body),
Description: "Transfer fee",
Amount: strconv.Itoa(int(math.Abs(math.Round(float64(transaction.Data.StatementItem.CommissionRate / 100))))),
SourceId: account.Id,
})
}
if len(firefly3Transactions) > 0 {
2024-03-25 11:55:01 +02:00
transactionOpts := firefly3.TransactionsApiStoreTransactionOpts{}
for _, transaction := range firefly3Transactions {
_, _, err = client.TransactionsApi.StoreTransaction(ctx, firefly3.TransactionStore{
ApplyRules: true,
Transactions: []firefly3.TransactionSplitStore{transaction},
}, &transactionOpts)
if err != nil {
LogString(err.Error())
fmt.Println(err.Error())
w.WriteHeader(http.StatusOK)
return
}
2024-03-25 11:55:01 +02:00
}
}
w.WriteHeader(http.StatusOK)
}
func main() {
err := godotenv.Load(".env")
if err != nil {
log.Fatalf("Error loading .env file")
}
2024-03-26 19:06:31 +02:00
/**
* Bills
*/
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"Hetzner"},
TransactionDescription: "Hetzner: vps2",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"YouTube"},
TransactionDescription: "YouTube membership: Latte ASMR",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"Київстар +380672463500"},
TransactionDescription: "Kyivstar: +380672463500",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"Lifecell +380732463500"},
TransactionDescription: "Lifecell: +380732463500",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"JetBrains"},
TransactionDescription: "JetBrains: GoLand",
})
/**
* People
*/
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"Ілля Ш."},
TransactionDescription: "Legal services: Alva Privacy Law Firm",
TransactionDestination: "Legal: Alva Privacy Law Firm",
TransactionCategory: "Legal services",
})
2024-03-26 19:06:31 +02:00
/**
* Other
*/
ShopConfig = append(ShopConfig, ShopConfigItem{
MCCCodes: []int{5411, 5499, 5451, 5422, 5412, 5921},
Names: []string{"АТБ", "Велмарт", "Novus", "Glovo", "zakaz.ua", "Мегамаркет", "Сільпо"},
TransactionDescription: "Groceries",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"Аптека Доброго Дня", "Аптека оптових цін", "Аптека Копійка", "Аптека Гала", "Аптека АНЦ", "APTEKA 7", "vidshkod ekv apt12"},
TransactionDescription: "Medications",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
MCCCodes: []int{4131, 4111, 4112},
Names: []string{"Київ Цифровий", "Київпастранс"},
TransactionDescription: "Medications",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"McDonalds"},
TransactionDescription: "McDonalds",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"LeoCafe"},
TransactionDescription: "Cafe",
})
ShopConfig = append(ShopConfig, ShopConfigItem{
Names: []string{"Lumberjack Barberhouse"},
TransactionDescription: "Lumberjack: haircut",
})
2024-03-25 11:55:01 +02:00
http.HandleFunc("/webhook", handleWebhook)
2024-03-26 19:06:31 +02:00
2024-03-25 11:55:01 +02:00
fmt.Println("Webhook server listening on :3021")
2024-03-26 19:06:31 +02:00
err = http.ListenAndServe(":3021", nil)
if err != nil {
panic(err.Error())
}
2024-03-25 11:55:01 +02:00
}