mirror of
https://github.com/aclindsa/ofxgo.git
synced 2024-11-15 01:20:05 -05:00
171 lines
6.5 KiB
Go
171 lines
6.5 KiB
Go
package ofxgo
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"github.com/golang/go/src/encoding/xml"
|
|
)
|
|
|
|
type AcctInfoRequest struct {
|
|
XMLName xml.Name `xml:"ACCTINFOTRNRQ"`
|
|
TrnUID UID `xml:"TRNUID"`
|
|
CltCookie Int `xml:"CLTCOOKIE"`
|
|
DtAcctUp Date `xml:"ACCTINFORQ>DTACCTUP"`
|
|
}
|
|
|
|
func (r *AcctInfoRequest) Name() string {
|
|
return "ACCTINFOTRNRQ"
|
|
}
|
|
|
|
func (r *AcctInfoRequest) Valid() (bool, error) {
|
|
if ok, err := r.TrnUID.Valid(); !ok {
|
|
return false, err
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
type HolderInfo struct {
|
|
XMLName xml.Name
|
|
FirstName String `xml:"FIRSTNAME"`
|
|
MiddleName String `xml:"MIDDLENAME,omitempty"`
|
|
LastName String `xml:"LASTNAME"`
|
|
Addr1 String `xml:"ADDR1"`
|
|
Addr2 String `xml:"ADDR2,omitempty"`
|
|
Addr3 String `xml:"ADDR3,omitempty"`
|
|
City String `xml:"CITY"`
|
|
State String `xml:"STATE"`
|
|
PostalCode String `xml:"POSTALCODE"`
|
|
Country String `xml:"COUNTRY,omitempty"`
|
|
DayPhone String `xml:"DAYPHONE,omitempty"`
|
|
EvePhone String `xml:"EVEPHONE,omitempty"`
|
|
Email String `xml:"EMAIL,omitempty"`
|
|
HolderType String `xml:"HOLDERTYPE,omitempty"` // One of INDIVIDUAL, JOINT, CUSTODIAL, TRUST, OTHER
|
|
}
|
|
|
|
type BankAcct struct {
|
|
XMLName xml.Name // BANKACCTTO or BANKACCTFROM
|
|
BankId String `xml:"BANKID"`
|
|
BranchId String `xml:"BRANCHID,omitempty"` // Unused in USA
|
|
AcctId String `xml:"ACCTID"`
|
|
AcctType String `xml:"ACCTTYPE"` // One of CHECKING, SAVINGS, MONEYMRKT, CREDITLINE, CD
|
|
AcctKey String `xml:"ACCTKEY,omitempty"` // Unused in USA
|
|
}
|
|
|
|
type BankAcctInfo struct {
|
|
XMLName xml.Name `xml:"BANKACCTINFO"`
|
|
BankAcctFrom BankAcct `xml:"BANKACCTFROM"`
|
|
SupTxDl Boolean `xml:"SUPTXDL"` // Supports downloading transactions (as opposed to balance only)
|
|
XferSrc Boolean `xml:"XFERSRC"` // Enabled as source for intra/interbank transfer
|
|
XferDest Boolean `xml:"XFERDEST"` // Enabled as destination for intra/interbank transfer
|
|
MaturityDate Date `xml:"MATURITYDATE,omitempty"` // Maturity date for CD, if CD
|
|
MaturityAmt Amount `xml:"MATURITYAMOUNT,omitempty"` // Maturity amount for CD, if CD
|
|
MinBalReq Amount `xml:"MINBALREQ,omitempty"` // Minimum balance required to avoid service fees
|
|
AcctClassification String `xml:"ACCTCLASSIFICATION,omitempty"` // One of PERSONAL, BUSINESS, CORPORATE, OTHER
|
|
OverdraftLimit Amount `xml:"OVERDRAFTLIMIT,omitempty"`
|
|
SvcStatus String `xml:"SVCSTATUS"` // One of AVAIL (available, but not yet requested), PEND (requested, but not yet available), ACTIVE
|
|
}
|
|
|
|
// Make pointers to these structs print nicely
|
|
func (bai *BankAcctInfo) String() string {
|
|
return fmt.Sprintf("%+v", *bai)
|
|
}
|
|
|
|
type CCAcct struct {
|
|
XMLName xml.Name // CCACCTTO or CCACCTFROM
|
|
AcctId String `xml:"ACCTID"`
|
|
AcctKey String `xml:"ACCTKEY,omitempty"` // Unused in USA
|
|
}
|
|
|
|
type CCAcctInfo struct {
|
|
XMLName xml.Name `xml:"CCACCTINFO"`
|
|
CCAcctFrom CCAcct `xml:"CCACCTFROM"`
|
|
SupTxDl Boolean `xml:"SUPTXDL"` // Supports downloading transactions (as opposed to balance only)
|
|
XferSrc Boolean `xml:"XFERSRC"` // Enabled as source for intra/interbank transfer
|
|
XferDest Boolean `xml:"XFERDEST"` // Enabled as destination for intra/interbank transfer
|
|
AcctClassification String `xml:"ACCTCLASSIFICATION,omitempty"` // One of PERSONAL, BUSINESS, CORPORATE, OTHER
|
|
SvcStatus String `xml:"SVCSTATUS"` // One of AVAIL (available, but not yet requested), PEND (requested, but not yet available), ACTIVE
|
|
}
|
|
|
|
// Make pointers to these structs print nicely
|
|
func (ci *CCAcctInfo) String() string {
|
|
return fmt.Sprintf("%+v", *ci)
|
|
}
|
|
|
|
type InvAcct struct {
|
|
XMLName xml.Name // INVACCTTO or INVACCTFROM
|
|
BrokerId String `xml:"BROKERID"`
|
|
AcctId String `xml:"ACCTID"`
|
|
}
|
|
|
|
type InvAcctInfo struct {
|
|
XMLName xml.Name `xml:"INVACCTINFO"`
|
|
INVAcctFrom InvAcct `xml:"INVACCTFROM"`
|
|
UsProductType String `xml:"USPRODUCTTYPE"` // One of 401K, 403B, IRA, KEOGH, OTHER, SARSEP, SIMPLE, NORMAL, TDA, TRUST, UGMA
|
|
Checking Boolean `xml:"CHECKING"` // Has check-writing privileges
|
|
SvcStatus String `xml:"SVCSTATUS"` // One of AVAIL (available, but not yet requested), PEND (requested, but not yet available), ACTIVE
|
|
InvAcctType String `xml:"INVACCTTYPE,omitempty"` // One of INDIVIDUAL, JOINT, TRUST, CORPORATE
|
|
OptionLevel String `xml:"OPTIONLEVEL,omitempty"` // Text desribing option trading privileges
|
|
}
|
|
|
|
// Make pointers to these structs print nicely
|
|
func (iai *InvAcctInfo) String() string {
|
|
return fmt.Sprintf("%+v", *iai)
|
|
}
|
|
|
|
type AcctInfo struct {
|
|
XMLName xml.Name `xml:"ACCTINFO"`
|
|
Name String `xml:"NAME,omitempty"`
|
|
Desc String `xml:"DESC,omitempty"`
|
|
Phone String `xml:"PHONE,omitempty"`
|
|
PrimaryHolder HolderInfo `xml:"HOLDERINFO>PRIMARYHOLDER,omitempty"`
|
|
SecondaryHolder HolderInfo `xml:"HOLDERINFO>SECONDARYHOLDER,omitempty"`
|
|
// Only one of the rest of the fields will be valid for any given AcctInfo
|
|
BankAcctInfo *BankAcctInfo `xml:"BANKACCTINFO,omitempty"`
|
|
CCAcctInfo *CCAcctInfo `xml:"CCACCTINFO,omitempty"`
|
|
InvAcctInfo *InvAcctInfo `xml:"INVACCTINFO,omitempty"`
|
|
// TODO LOANACCTINFO
|
|
// TODO BPACCTINFO?
|
|
}
|
|
|
|
type AcctInfoResponse struct {
|
|
XMLName xml.Name `xml:"ACCTINFOTRNRS"`
|
|
TrnUID UID `xml:"TRNUID"`
|
|
DtAcctUp Date `xml:"ACCTINFORS>DTACCTUP"`
|
|
AcctInfo []AcctInfo `xml:"ACCTINFORS>ACCTINFO,omitempty"`
|
|
}
|
|
|
|
func (air AcctInfoResponse) Name() string {
|
|
return "ACCTINFORS"
|
|
}
|
|
|
|
func (air AcctInfoResponse) Valid() (bool, error) {
|
|
//TODO implement
|
|
return true, nil
|
|
}
|
|
|
|
func DecodeSignupMessageSet(d *xml.Decoder, start xml.StartElement) ([]Message, error) {
|
|
var msgs []Message
|
|
for {
|
|
tok, err := d.Token()
|
|
if err != nil {
|
|
return nil, err
|
|
} else if end, ok := tok.(xml.EndElement); ok && end.Name.Local == start.Name.Local {
|
|
// If we found the end of our starting element, we're done parsing
|
|
return msgs, nil
|
|
} else if startElement, ok := tok.(xml.StartElement); ok {
|
|
switch startElement.Name.Local {
|
|
case "ACCTINFOTRNRS":
|
|
var info AcctInfoResponse
|
|
if err := d.DecodeElement(&info, &startElement); err != nil {
|
|
return nil, err
|
|
}
|
|
msgs = append(msgs, Message(info))
|
|
default:
|
|
return nil, errors.New("Unsupported signup response tag: " + startElement.Name.Local)
|
|
}
|
|
} else {
|
|
return nil, errors.New("Didn't find an opening element")
|
|
}
|
|
}
|
|
}
|