1
0
mirror of https://github.com/aclindsa/moneygo.git synced 2024-12-26 23:42:29 -05:00
moneygo/internal/integration/common_test.go
Aaron Lindsay a357d38eee Store currency/security values/prices using big.Rat natively
This adds 'shadow' types used only by the store/db internal package whch
handle converting these types to their DB-equivalent values. This change
should allow reports to be generated significantly faster since it
allows a large portion of the computation to be shifted to the database
engines.
2017-12-12 19:50:38 -05:00

286 lines
5.7 KiB
Go

package integration_test
import (
"bytes"
"encoding/json"
"github.com/aclindsa/moneygo/internal/config"
"github.com/aclindsa/moneygo/internal/handlers"
"github.com/aclindsa/moneygo/internal/models"
"github.com/aclindsa/moneygo/internal/store/db"
"io"
"io/ioutil"
"log"
"mime/multipart"
"net/http"
"net/http/httptest"
"os"
"testing"
)
var server *httptest.Server
func Delete(client *http.Client, url string) (*http.Response, error) {
request, err := http.NewRequest(http.MethodDelete, url, nil)
if err != nil {
return nil, err
}
return client.Do(request)
}
func Put(client *http.Client, url string, contentType string, body io.Reader) (*http.Response, error) {
request, err := http.NewRequest(http.MethodPut, url, body)
if err != nil {
return nil, err
}
request.Header.Set("Content-Type", contentType)
return client.Do(request)
}
type TransactType interface {
Read(string) error
}
func create(client *http.Client, input, output TransactType, urlsuffix string) error {
obj, err := json.MarshalIndent(input, "", " ")
if err != nil {
return err
}
response, err := client.Post(server.URL+urlsuffix, "application/json", bytes.NewReader(obj))
if err != nil {
return err
}
body, err := ioutil.ReadAll(response.Body)
response.Body.Close()
if err != nil {
return err
}
var e handlers.Error
err = (&e).Read(string(body))
if err != nil {
return err
}
if e.ErrorId != 0 || len(e.ErrorString) != 0 {
return &e
}
err = output.Read(string(body))
if err != nil {
return err
}
return nil
}
func read(client *http.Client, output TransactType, urlsuffix string) error {
response, err := client.Get(server.URL + urlsuffix)
if err != nil {
return err
}
body, err := ioutil.ReadAll(response.Body)
response.Body.Close()
if err != nil {
return err
}
var e handlers.Error
err = (&e).Read(string(body))
if err != nil {
return err
}
if e.ErrorId != 0 || len(e.ErrorString) != 0 {
return &e
}
err = output.Read(string(body))
if err != nil {
return err
}
return nil
}
func update(client *http.Client, input, output TransactType, urlsuffix string) error {
obj, err := json.MarshalIndent(input, "", " ")
if err != nil {
return err
}
response, err := Put(client, server.URL+urlsuffix, "application/json", bytes.NewReader(obj))
if err != nil {
return err
}
body, err := ioutil.ReadAll(response.Body)
response.Body.Close()
if err != nil {
return err
}
var e handlers.Error
err = (&e).Read(string(body))
if err != nil {
return err
}
if e.ErrorId != 0 || len(e.ErrorString) != 0 {
return &e
}
err = output.Read(string(body))
if err != nil {
return err
}
return nil
}
func remove(client *http.Client, urlsuffix string) error {
response, err := Delete(client, server.URL+urlsuffix)
if err != nil {
return err
}
body, err := ioutil.ReadAll(response.Body)
response.Body.Close()
if err != nil {
return err
}
var e handlers.Error
err = (&e).Read(string(body))
if err != nil {
return err
}
if e.ErrorId != 0 || len(e.ErrorString) != 0 {
return &e
}
return nil
}
func uploadFile(client *http.Client, filename, urlsuffix string) error {
var buf bytes.Buffer
mw := multipart.NewWriter(&buf)
file, err := os.Open(filename)
if err != nil {
return err
}
defer file.Close()
filewriter, err := mw.CreateFormFile("file", filename)
if err != nil {
return err
}
if _, err := io.Copy(filewriter, file); err != nil {
return err
}
mw.Close()
response, err := client.Post(server.URL+urlsuffix, mw.FormDataContentType(), &buf)
if err != nil {
return err
}
body, err := ioutil.ReadAll(response.Body)
response.Body.Close()
if err != nil {
return err
}
var e handlers.Error
err = (&e).Read(string(body))
if err != nil {
return err
}
if e.ErrorId != 0 || len(e.ErrorString) != 0 {
return &e
}
return nil
}
func NewAmount(amt string) models.Amount {
var a models.Amount
if _, ok := a.SetString(amt); !ok {
panic("Unable to call Amount.SetString()")
}
return a
}
func amountsMatch(a models.Amount, amt string) bool {
cmp := NewAmount(amt)
return a.Cmp(&cmp.Rat) == 0
}
func accountBalanceHelper(t *testing.T, client *http.Client, account *models.Account, balance string) {
t.Helper()
transactions, err := getAccountTransactions(client, account.AccountId, 0, 0, "")
if err != nil {
t.Fatalf("Couldn't fetch account transactions for '%s': %s\n", account.Name, err)
}
if !amountsMatch(transactions.EndingBalance, balance) {
t.Errorf("Expected ending balance for '%s' to be '%s', but found %s\n", account.Name, balance, transactions.EndingBalance)
}
}
func RunWith(t *testing.T, d *TestData, fn TestDataFunc) {
testdata, err := d.Initialize()
if err != nil {
t.Fatalf("Failed to initialize test data: %s", err)
}
defer func() {
err := testdata.Teardown()
if err != nil {
t.Fatal(err)
}
}()
fn(t, testdata)
}
func RunTests(m *testing.M) int {
envDbType := os.Getenv("MONEYGO_TEST_DB")
var dbType config.DbType
var dsn string
switch envDbType {
case "", "sqlite", "sqlite3":
dbType = config.SQLite
dsn = ":memory:"
case "mariadb", "mysql":
dbType = config.MySQL
dsn = "root@127.0.0.1/moneygo_test&parseTime=true"
case "postgres", "postgresql":
dbType = config.Postgres
dsn = "postgres://postgres@localhost/moneygo_test"
default:
log.Fatalf("Invalid value for $MONEYGO_TEST_DB: %s\n", envDbType)
}
if envDSN := os.Getenv("MONEYGO_TEST_DSN"); len(envDSN) > 0 {
dsn = envDSN
}
db, err := db.GetStore(dbType, dsn)
if err != nil {
log.Fatal(err)
}
defer db.Close()
db.Empty() // clear the DB tables
server = httptest.NewTLSServer(&handlers.APIHandler{Store: db})
defer server.Close()
return m.Run()
}
func TestMain(m *testing.M) {
log.SetFlags(log.Ldate | log.Ltime | log.Lshortfile)
os.Exit(RunTests(m))
}