2017-10-04 19:35:59 -04:00
package handlers
2015-06-25 22:36:58 -04:00
2015-06-27 17:46:06 -04:00
import (
"encoding/json"
"errors"
"log"
"net/http"
2015-07-11 08:58:36 -04:00
"regexp"
2015-06-27 17:46:06 -04:00
"strings"
)
2017-02-08 05:40:51 -05:00
type AccountType int64
2015-06-25 22:36:58 -04:00
const (
2017-02-08 05:40:51 -05:00
Bank AccountType = 1 // start at 1 so that the default (0) is invalid
Cash = 2
Asset = 3
Liability = 4
Investment = 5
Income = 6
Expense = 7
Trading = 8
Equity = 9
Receivable = 10
Payable = 11
2015-06-25 22:36:58 -04:00
)
2017-02-08 05:40:51 -05:00
var AccountTypes = [ ] AccountType {
Bank ,
Cash ,
Asset ,
Liability ,
Investment ,
Income ,
Expense ,
Trading ,
Equity ,
Receivable ,
Payable ,
}
func ( t AccountType ) String ( ) string {
switch t {
case Bank :
return "Bank"
case Cash :
return "Cash"
case Asset :
return "Asset"
case Liability :
return "Liability"
case Investment :
return "Investment"
case Income :
return "Income"
case Expense :
return "Expense"
case Trading :
return "Trading"
case Equity :
return "Equity"
case Receivable :
return "Receivable"
case Payable :
return "Payable"
}
return ""
}
2015-06-25 22:36:58 -04:00
type Account struct {
2016-02-02 21:46:27 -05:00
AccountId int64
ExternalAccountId string
UserId int64
SecurityId int64
ParentAccountId int64 // -1 if this account is at the root
2017-02-08 05:40:51 -05:00
Type AccountType
2016-02-02 21:46:27 -05:00
Name string
2015-07-11 08:58:36 -04:00
// monotonically-increasing account transaction version number. Used for
// allowing a client to ensure they have a consistent version when paging
// through transactions.
2015-08-05 21:25:25 -04:00
AccountVersion int64 ` json:"Version" `
2017-05-31 21:04:01 -04:00
// Optional fields specifying how to fetch transactions from a bank via OFX
OFXURL string
OFXORG string
OFXFID string
OFXUser string
OFXBankID string // OFX BankID (BrokerID if AcctType == Investment)
OFXAcctID string
OFXAcctType string // ofxgo.acctType
OFXClientUID string
OFXAppID string
OFXAppVer string
OFXVersion string
OFXNoIndent bool
2015-06-27 17:46:06 -04:00
}
type AccountList struct {
Accounts * [ ] Account ` json:"accounts" `
}
2015-07-11 08:58:36 -04:00
var accountTransactionsRE * regexp . Regexp
2016-02-02 21:46:27 -05:00
var accountImportRE * regexp . Regexp
2015-07-11 08:58:36 -04:00
func init ( ) {
2017-11-11 08:05:09 -05:00
accountTransactionsRE = regexp . MustCompile ( ` ^/v1/accounts/[0-9]+/transactions/?$ ` )
accountImportRE = regexp . MustCompile ( ` ^/v1/accounts/[0-9]+/imports/[a-z]+/?$ ` )
2015-07-11 08:58:36 -04:00
}
2015-06-27 17:46:06 -04:00
func ( a * Account ) Write ( w http . ResponseWriter ) error {
enc := json . NewEncoder ( w )
return enc . Encode ( a )
}
func ( a * Account ) Read ( json_str string ) error {
dec := json . NewDecoder ( strings . NewReader ( json_str ) )
return dec . Decode ( a )
}
func ( al * AccountList ) Write ( w http . ResponseWriter ) error {
enc := json . NewEncoder ( w )
return enc . Encode ( al )
}
2017-10-12 22:19:53 -04:00
func ( al * AccountList ) Read ( json_str string ) error {
dec := json . NewDecoder ( strings . NewReader ( json_str ) )
return dec . Decode ( al )
}
2017-10-14 14:20:50 -04:00
func GetAccount ( tx * Tx , accountid int64 , userid int64 ) ( * Account , error ) {
2015-06-27 17:46:06 -04:00
var a Account
2017-10-14 14:20:50 -04:00
err := tx . SelectOne ( & a , "SELECT * from accounts where UserId=? AND AccountId=?" , userid , accountid )
2015-06-27 17:46:06 -04:00
if err != nil {
return nil , err
}
return & a , nil
}
2017-10-14 14:20:50 -04:00
func GetAccounts ( tx * Tx , userid int64 ) ( * [ ] Account , error ) {
2015-06-27 17:46:06 -04:00
var accounts [ ] Account
2017-10-14 14:20:50 -04:00
_ , err := tx . Select ( & accounts , "SELECT * from accounts where UserId=?" , userid )
2015-06-27 17:46:06 -04:00
if err != nil {
return nil , err
}
return & accounts , nil
}
2016-02-15 11:28:44 -05:00
// Get (and attempt to create if it doesn't exist). Matches on UserId,
// SecurityId, Type, Name, and ParentAccountId
2017-10-14 19:41:13 -04:00
func GetCreateAccount ( tx * Tx , a Account ) ( * Account , error ) {
2016-02-15 11:28:44 -05:00
var accounts [ ] Account
2016-02-10 18:36:11 -05:00
var account Account
// Try to find the top-level trading account
2017-10-14 19:41:13 -04:00
_ , err := tx . Select ( & accounts , "SELECT * from accounts where UserId=? AND SecurityId=? AND Type=? AND Name=? AND ParentAccountId=? ORDER BY AccountId ASC LIMIT 1" , a . UserId , a . SecurityId , a . Type , a . Name , a . ParentAccountId )
2016-02-10 18:36:11 -05:00
if err != nil {
return nil , err
}
if len ( accounts ) == 1 {
account = accounts [ 0 ]
} else {
2016-02-15 11:28:44 -05:00
account . UserId = a . UserId
account . SecurityId = a . SecurityId
account . Type = a . Type
account . Name = a . Name
account . ParentAccountId = a . ParentAccountId
2016-02-10 18:36:11 -05:00
2017-10-14 19:41:13 -04:00
err = tx . Insert ( & account )
2016-02-11 05:53:44 -05:00
if err != nil {
return nil , err
}
}
return & account , nil
}
// Get (and attempt to create if it doesn't exist) the security/currency
2016-02-15 11:28:44 -05:00
// trading account for the supplied security/currency
2017-10-14 19:41:13 -04:00
func GetTradingAccount ( tx * Tx , userid int64 , securityid int64 ) ( * Account , error ) {
2016-02-15 11:28:44 -05:00
var tradingAccount Account
2016-02-11 05:53:44 -05:00
var account Account
2017-10-14 19:41:13 -04:00
user , err := GetUser ( tx , userid )
2017-09-21 21:00:30 -04:00
if err != nil {
return nil , err
}
2016-02-15 11:28:44 -05:00
tradingAccount . UserId = userid
tradingAccount . Type = Trading
tradingAccount . Name = "Trading"
2017-09-21 21:00:30 -04:00
tradingAccount . SecurityId = user . DefaultCurrency
2016-02-15 11:28:44 -05:00
tradingAccount . ParentAccountId = - 1
// Find/create the top-level trading account
2017-10-14 19:41:13 -04:00
ta , err := GetCreateAccount ( tx , tradingAccount )
2016-02-11 05:53:44 -05:00
if err != nil {
return nil , err
}
2017-10-14 19:41:13 -04:00
security , err := GetSecurity ( tx , securityid , userid )
2016-10-16 08:19:11 -04:00
if err != nil {
return nil , err
}
2016-02-15 11:28:44 -05:00
account . UserId = userid
account . Name = security . Name
account . ParentAccountId = ta . AccountId
account . SecurityId = securityid
account . Type = Trading
2017-10-14 19:41:13 -04:00
a , err := GetCreateAccount ( tx , account )
2016-02-11 05:53:44 -05:00
if err != nil {
return nil , err
}
2016-02-15 11:28:44 -05:00
return a , nil
}
// Get (and attempt to create if it doesn't exist) the security/currency
// imbalance account for the supplied security/currency
2017-10-14 19:41:13 -04:00
func GetImbalanceAccount ( tx * Tx , userid int64 , securityid int64 ) ( * Account , error ) {
2016-02-15 11:28:44 -05:00
var imbalanceAccount Account
var account Account
2017-06-04 16:01:42 -04:00
xxxtemplate := FindSecurityTemplate ( "XXX" , Currency )
if xxxtemplate == nil {
return nil , errors . New ( "Couldn't find XXX security template" )
}
2017-10-14 19:41:13 -04:00
xxxsecurity , err := ImportGetCreateSecurity ( tx , userid , xxxtemplate )
2017-06-04 16:01:42 -04:00
if err != nil {
return nil , errors . New ( "Couldn't create XXX security" )
}
2016-02-11 05:53:44 -05:00
2016-02-15 11:28:44 -05:00
imbalanceAccount . UserId = userid
imbalanceAccount . Name = "Imbalances"
imbalanceAccount . ParentAccountId = - 1
2017-06-04 16:01:42 -04:00
imbalanceAccount . SecurityId = xxxsecurity . SecurityId
2016-02-15 11:28:44 -05:00
imbalanceAccount . Type = Bank
// Find/create the top-level trading account
2017-10-14 19:41:13 -04:00
ia , err := GetCreateAccount ( tx , imbalanceAccount )
2016-02-11 05:53:44 -05:00
if err != nil {
return nil , err
}
2017-10-14 19:41:13 -04:00
security , err := GetSecurity ( tx , securityid , userid )
2016-10-16 08:19:11 -04:00
if err != nil {
return nil , err
}
2016-02-15 11:28:44 -05:00
account . UserId = userid
account . Name = security . Name
account . ParentAccountId = ia . AccountId
account . SecurityId = securityid
account . Type = Bank
2016-02-10 18:36:11 -05:00
2017-10-14 19:41:13 -04:00
a , err := GetCreateAccount ( tx , account )
2016-02-10 18:36:11 -05:00
if err != nil {
return nil , err
}
2016-02-15 11:28:44 -05:00
return a , nil
2016-02-10 18:36:11 -05:00
}
2015-06-27 17:46:06 -04:00
type ParentAccountMissingError struct { }
func ( pame ParentAccountMissingError ) Error ( ) string {
return "Parent account missing"
}
2017-10-12 21:20:13 -04:00
type TooMuchNestingError struct { }
func ( tmne TooMuchNestingError ) Error ( ) string {
return "Too much nesting"
}
type CircularAccountsError struct { }
func ( cae CircularAccountsError ) Error ( ) string {
return "Would result in circular account relationship"
}
2017-10-14 14:20:50 -04:00
func insertUpdateAccount ( tx * Tx , a * Account , insert bool ) error {
2017-10-12 21:20:13 -04:00
found := make ( map [ int64 ] bool )
if ! insert {
found [ a . AccountId ] = true
}
parentid := a . ParentAccountId
depth := 0
for parentid != - 1 {
depth += 1
if depth > 100 {
return TooMuchNestingError { }
2015-06-27 17:46:06 -04:00
}
2017-10-12 21:20:13 -04:00
var a Account
2017-10-14 14:20:50 -04:00
err := tx . SelectOne ( & a , "SELECT * from accounts where AccountId=?" , parentid )
2017-10-12 21:20:13 -04:00
if err != nil {
2015-06-27 17:46:06 -04:00
return ParentAccountMissingError { }
}
2017-10-12 21:20:13 -04:00
// Insertion by itself can never result in circular dependencies
if insert {
break
}
found [ parentid ] = true
parentid = a . ParentAccountId
if _ , ok := found [ parentid ] ; ok {
return CircularAccountsError { }
}
2015-06-27 17:46:06 -04:00
}
if insert {
2017-10-14 14:20:50 -04:00
err := tx . Insert ( a )
2015-06-27 17:46:06 -04:00
if err != nil {
return err
}
} else {
2017-10-14 19:41:13 -04:00
oldacct , err := GetAccount ( tx , a . AccountId , a . UserId )
2015-07-11 08:58:36 -04:00
if err != nil {
return err
}
2015-08-05 21:25:25 -04:00
a . AccountVersion = oldacct . AccountVersion + 1
2015-07-11 08:58:36 -04:00
2017-10-14 14:20:50 -04:00
count , err := tx . Update ( a )
2015-06-27 17:46:06 -04:00
if err != nil {
return err
}
if count != 1 {
return errors . New ( "Updated more than one account" )
}
}
return nil
}
2017-10-14 14:20:50 -04:00
func InsertAccount ( tx * Tx , a * Account ) error {
return insertUpdateAccount ( tx , a , true )
2015-06-27 17:46:06 -04:00
}
2017-10-14 14:20:50 -04:00
func UpdateAccount ( tx * Tx , a * Account ) error {
return insertUpdateAccount ( tx , a , false )
2015-06-27 17:46:06 -04:00
}
2017-10-14 14:20:50 -04:00
func DeleteAccount ( tx * Tx , a * Account ) error {
2015-07-04 21:11:00 -04:00
if a . ParentAccountId != - 1 {
// Re-parent splits to this account's parent account if this account isn't a root account
2017-10-14 14:20:50 -04:00
_ , err := tx . Exec ( "UPDATE splits SET AccountId=? WHERE AccountId=?" , a . ParentAccountId , a . AccountId )
2015-07-04 21:11:00 -04:00
if err != nil {
return err
}
} else {
// Delete splits if this account is a root account
2017-10-14 14:20:50 -04:00
_ , err := tx . Exec ( "DELETE FROM splits WHERE AccountId=?" , a . AccountId )
2015-07-04 21:11:00 -04:00
if err != nil {
return err
}
2015-06-29 07:25:48 -04:00
}
// Re-parent child accounts to this account's parent account
2017-10-14 14:20:50 -04:00
_ , err := tx . Exec ( "UPDATE accounts SET ParentAccountId=? WHERE ParentAccountId=?" , a . ParentAccountId , a . AccountId )
2015-06-29 07:25:48 -04:00
if err != nil {
return err
}
2017-10-14 14:20:50 -04:00
count , err := tx . Delete ( a )
2015-06-29 07:25:48 -04:00
if err != nil {
return err
}
if count != 1 {
return errors . New ( "Was going to delete more than one account" )
}
return nil
}
2017-10-14 14:20:50 -04:00
func AccountHandler ( r * http . Request , tx * Tx ) ResponseWriterWriter {
user , err := GetUserFromSession ( tx , r )
2015-06-27 17:46:06 -04:00
if err != nil {
2017-10-14 14:20:50 -04:00
return NewError ( 1 /*Not Signed In*/ )
2015-06-27 17:46:06 -04:00
}
if r . Method == "POST" {
2017-11-11 08:05:09 -05:00
// if URL looks like /v1/accounts/[0-9]+/imports, use the account
2016-02-02 21:46:27 -05:00
// import handler
if accountImportRE . MatchString ( r . URL . Path ) {
var accountid int64
2016-02-15 11:28:44 -05:00
var importtype string
2017-11-11 08:05:09 -05:00
n , err := GetURLPieces ( r . URL . Path , "/v1/accounts/%d/imports/%s" , & accountid , & importtype )
2016-02-02 21:46:27 -05:00
2016-02-15 11:28:44 -05:00
if err != nil || n != 2 {
2016-02-02 21:46:27 -05:00
log . Print ( err )
2017-10-14 14:20:50 -04:00
return NewError ( 999 /*Internal Error*/ )
2016-02-02 21:46:27 -05:00
}
2017-10-14 14:20:50 -04:00
return AccountImportHandler ( tx , r , user , accountid , importtype )
2016-02-02 21:46:27 -05:00
}
2015-06-27 17:46:06 -04:00
account_json := r . PostFormValue ( "account" )
if account_json == "" {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
var account Account
err := account . Read ( account_json )
if err != nil {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
account . AccountId = - 1
account . UserId = user . UserId
2015-08-05 21:25:25 -04:00
account . AccountVersion = 0
2015-06-27 17:46:06 -04:00
2017-10-14 14:20:50 -04:00
security , err := GetSecurity ( tx , account . SecurityId , user . UserId )
2016-10-16 08:19:11 -04:00
if err != nil {
log . Print ( err )
2017-10-14 14:20:50 -04:00
return NewError ( 999 /*Internal Error*/ )
2016-10-16 08:19:11 -04:00
}
if security == nil {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
2017-10-14 14:20:50 -04:00
err = InsertAccount ( tx , & account )
2015-06-27 17:46:06 -04:00
if err != nil {
if _ , ok := err . ( ParentAccountMissingError ) ; ok {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
} else {
log . Print ( err )
2017-10-14 14:20:50 -04:00
return NewError ( 999 /*Internal Error*/ )
2015-06-27 17:46:06 -04:00
}
}
2017-10-14 14:20:50 -04:00
return ResponseWrapper { 201 , & account }
2015-06-27 17:46:06 -04:00
} else if r . Method == "GET" {
2015-07-11 08:58:36 -04:00
var accountid int64
2017-11-11 08:05:09 -05:00
n , err := GetURLPieces ( r . URL . Path , "/v1/accounts/%d" , & accountid )
2015-07-11 08:58:36 -04:00
if err != nil || n != 1 {
2015-06-27 17:46:06 -04:00
//Return all Accounts
var al AccountList
2017-10-14 14:20:50 -04:00
accounts , err := GetAccounts ( tx , user . UserId )
2015-06-27 17:46:06 -04:00
if err != nil {
log . Print ( err )
2017-10-14 14:20:50 -04:00
return NewError ( 999 /*Internal Error*/ )
2015-06-27 17:46:06 -04:00
}
al . Accounts = accounts
2017-10-14 14:20:50 -04:00
return & al
2015-06-27 17:46:06 -04:00
} else {
2015-07-11 08:58:36 -04:00
// if URL looks like /account/[0-9]+/transactions, use the account
// transaction handler
if accountTransactionsRE . MatchString ( r . URL . Path ) {
2017-10-14 14:20:50 -04:00
return AccountTransactionsHandler ( tx , r , user , accountid )
2015-07-11 08:58:36 -04:00
}
2015-06-29 07:25:48 -04:00
// Return Account with this Id
2017-10-14 14:20:50 -04:00
account , err := GetAccount ( tx , accountid , user . UserId )
2015-06-27 17:46:06 -04:00
if err != nil {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
2015-07-11 08:58:36 -04:00
2017-10-14 14:20:50 -04:00
return account
2015-06-27 17:46:06 -04:00
}
} else {
accountid , err := GetURLID ( r . URL . Path )
if err != nil {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
if r . Method == "PUT" {
account_json := r . PostFormValue ( "account" )
if account_json == "" {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
var account Account
err := account . Read ( account_json )
if err != nil || account . AccountId != accountid {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
account . UserId = user . UserId
2017-10-14 14:20:50 -04:00
security , err := GetSecurity ( tx , account . SecurityId , user . UserId )
2016-10-16 08:19:11 -04:00
if err != nil {
log . Print ( err )
2017-10-14 14:20:50 -04:00
return NewError ( 999 /*Internal Error*/ )
2016-10-16 08:19:11 -04:00
}
if security == nil {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
2017-10-11 21:19:14 -04:00
if account . ParentAccountId == account . AccountId {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2017-10-11 21:19:14 -04:00
}
2017-10-14 14:20:50 -04:00
err = UpdateAccount ( tx , & account )
2015-06-27 17:46:06 -04:00
if err != nil {
2017-10-11 21:19:14 -04:00
if _ , ok := err . ( ParentAccountMissingError ) ; ok {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2017-10-12 21:20:13 -04:00
} else if _ , ok := err . ( CircularAccountsError ) ; ok {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2017-10-11 21:19:14 -04:00
} else {
log . Print ( err )
2017-10-14 14:20:50 -04:00
return NewError ( 999 /*Internal Error*/ )
2017-10-11 21:19:14 -04:00
}
2015-06-27 17:46:06 -04:00
}
2017-10-14 14:20:50 -04:00
return & account
2015-06-27 17:46:06 -04:00
} else if r . Method == "DELETE" {
2017-10-14 14:20:50 -04:00
account , err := GetAccount ( tx , accountid , user . UserId )
2015-06-27 17:46:06 -04:00
if err != nil {
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-27 17:46:06 -04:00
}
2017-10-14 14:20:50 -04:00
err = DeleteAccount ( tx , account )
2015-06-29 07:25:48 -04:00
if err != nil {
2015-06-27 17:46:06 -04:00
log . Print ( err )
2017-10-14 14:20:50 -04:00
return NewError ( 999 /*Internal Error*/ )
2015-06-27 17:46:06 -04:00
}
2017-10-14 14:20:50 -04:00
return SuccessWriter { }
2015-06-27 17:46:06 -04:00
}
}
2017-10-14 14:20:50 -04:00
return NewError ( 3 /*Invalid Request*/ )
2015-06-25 22:36:58 -04:00
}