2017-10-04 19:35:59 -04:00
|
|
|
package handlers
|
|
|
|
|
|
|
|
//go:generate make
|
2015-06-25 22:36:58 -04:00
|
|
|
|
2015-06-27 08:31:38 -04:00
|
|
|
import (
|
2016-10-16 08:19:11 -04:00
|
|
|
"errors"
|
2017-12-03 06:38:22 -05:00
|
|
|
"github.com/aclindsa/moneygo/internal/models"
|
2017-12-07 20:47:55 -05:00
|
|
|
"github.com/aclindsa/moneygo/internal/store"
|
2015-06-27 08:31:38 -04:00
|
|
|
"log"
|
|
|
|
"net/http"
|
2016-10-16 08:19:11 -04:00
|
|
|
"net/url"
|
2016-10-26 06:58:14 -04:00
|
|
|
"strconv"
|
2016-10-16 08:19:11 -04:00
|
|
|
"strings"
|
2015-06-27 08:31:38 -04:00
|
|
|
)
|
|
|
|
|
2017-12-03 06:38:22 -05:00
|
|
|
func SearchSecurityTemplates(search string, _type models.SecurityType, limit int64) []*models.Security {
|
2016-10-16 08:19:11 -04:00
|
|
|
upperSearch := strings.ToUpper(search)
|
2017-12-03 06:38:22 -05:00
|
|
|
var results []*models.Security
|
2016-10-16 08:19:11 -04:00
|
|
|
for i, security := range SecurityTemplates {
|
|
|
|
if strings.Contains(strings.ToUpper(security.Name), upperSearch) ||
|
|
|
|
strings.Contains(strings.ToUpper(security.Description), upperSearch) ||
|
|
|
|
strings.Contains(strings.ToUpper(security.Symbol), upperSearch) {
|
|
|
|
if _type == 0 || _type == security.Type {
|
|
|
|
results = append(results, &SecurityTemplates[i])
|
2016-10-26 06:58:14 -04:00
|
|
|
if limit != -1 && int64(len(results)) >= limit {
|
|
|
|
break
|
|
|
|
}
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
}
|
2015-06-27 08:31:38 -04:00
|
|
|
}
|
2016-10-16 08:19:11 -04:00
|
|
|
return results
|
2015-06-27 08:31:38 -04:00
|
|
|
}
|
|
|
|
|
2017-12-03 06:38:22 -05:00
|
|
|
func FindSecurityTemplate(name string, _type models.SecurityType) *models.Security {
|
2017-02-19 07:50:36 -05:00
|
|
|
for _, security := range SecurityTemplates {
|
|
|
|
if name == security.Name && _type == security.Type {
|
|
|
|
return &security
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-12-03 06:38:22 -05:00
|
|
|
func FindCurrencyTemplate(iso4217 int64) *models.Security {
|
2017-06-21 21:25:38 -04:00
|
|
|
iso4217string := strconv.FormatInt(iso4217, 10)
|
|
|
|
for _, security := range SecurityTemplates {
|
2017-12-03 06:38:22 -05:00
|
|
|
if security.Type == models.Currency && security.AlternateId == iso4217string {
|
2017-06-21 21:25:38 -04:00
|
|
|
return &security
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-12-09 05:56:45 -05:00
|
|
|
func UpdateSecurity(tx store.Tx, s *models.Security) (err error) {
|
2017-12-07 20:08:43 -05:00
|
|
|
user, err := tx.GetUser(s.UserId)
|
2016-10-16 08:19:11 -04:00
|
|
|
if err != nil {
|
2017-10-14 14:20:50 -04:00
|
|
|
return
|
2017-12-03 06:38:22 -05:00
|
|
|
} else if user.DefaultCurrency == s.SecurityId && s.Type != models.Currency {
|
2017-06-21 21:53:01 -04:00
|
|
|
return errors.New("Cannot change security which is user's default currency to be non-currency")
|
|
|
|
}
|
|
|
|
|
2017-12-07 20:08:43 -05:00
|
|
|
err = tx.UpdateSecurity(s)
|
2016-10-16 08:19:11 -04:00
|
|
|
if err != nil {
|
2017-10-14 14:20:50 -04:00
|
|
|
return
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2015-06-27 08:31:38 -04:00
|
|
|
}
|
|
|
|
|
2017-12-09 05:56:45 -05:00
|
|
|
func ImportGetCreateSecurity(tx store.Tx, userid int64, security *models.Security) (*models.Security, error) {
|
2017-06-04 16:01:42 -04:00
|
|
|
security.UserId = userid
|
2017-02-19 07:50:36 -05:00
|
|
|
if len(security.AlternateId) == 0 {
|
|
|
|
// Always create a new local security if we can't match on the AlternateId
|
2017-12-07 20:08:43 -05:00
|
|
|
err := tx.InsertSecurity(security)
|
2017-02-19 07:50:36 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return security, nil
|
|
|
|
}
|
|
|
|
|
2017-12-07 20:47:55 -05:00
|
|
|
securities, err := tx.FindMatchingSecurities(security)
|
2017-02-19 07:50:36 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// First try to find a case insensitive match on the name or symbol
|
|
|
|
upperName := strings.ToUpper(security.Name)
|
|
|
|
upperSymbol := strings.ToUpper(security.Symbol)
|
2017-12-07 20:08:43 -05:00
|
|
|
for _, s := range *securities {
|
2017-02-19 07:50:36 -05:00
|
|
|
if (len(s.Name) > 0 && strings.ToUpper(s.Name) == upperName) ||
|
|
|
|
(len(s.Symbol) > 0 && strings.ToUpper(s.Symbol) == upperSymbol) {
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// if strings.Contains(strings.ToUpper(security.Name), upperSearch) ||
|
|
|
|
|
|
|
|
// Try to find a partial string match on the name or symbol
|
2017-12-07 20:08:43 -05:00
|
|
|
for _, s := range *securities {
|
2017-02-19 07:50:36 -05:00
|
|
|
sUpperName := strings.ToUpper(s.Name)
|
|
|
|
sUpperSymbol := strings.ToUpper(s.Symbol)
|
|
|
|
if (len(upperName) > 0 && len(s.Name) > 0 && (strings.Contains(upperName, sUpperName) || strings.Contains(sUpperName, upperName))) ||
|
|
|
|
(len(upperSymbol) > 0 && len(s.Symbol) > 0 && (strings.Contains(upperSymbol, sUpperSymbol) || strings.Contains(sUpperSymbol, upperSymbol))) {
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Give up and return the first security in the list
|
2017-12-07 20:08:43 -05:00
|
|
|
if len(*securities) > 0 {
|
|
|
|
return (*securities)[0], nil
|
2017-02-19 07:50:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// If there wasn't even one security in the list, make a new one
|
2017-12-07 20:08:43 -05:00
|
|
|
err = tx.InsertSecurity(security)
|
2017-02-19 07:50:36 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return security, nil
|
|
|
|
}
|
|
|
|
|
2017-11-12 20:17:27 -05:00
|
|
|
func SecurityHandler(r *http.Request, context *Context) ResponseWriterWriter {
|
|
|
|
user, err := GetUserFromSession(context.Tx, r)
|
2016-10-16 08:19:11 -04:00
|
|
|
if err != nil {
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(1 /*Not Signed In*/)
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if r.Method == "POST" {
|
2017-11-16 19:17:51 -05:00
|
|
|
if !context.LastLevel() {
|
|
|
|
securityid, err := context.NextID()
|
|
|
|
if err != nil {
|
|
|
|
return NewError(3 /*Invalid Request*/)
|
|
|
|
}
|
|
|
|
if context.NextLevel() != "prices" {
|
|
|
|
return NewError(3 /*Invalid Request*/)
|
|
|
|
}
|
|
|
|
return PriceHandler(r, context, user, securityid)
|
|
|
|
}
|
|
|
|
|
2017-12-03 06:38:22 -05:00
|
|
|
var security models.Security
|
2017-11-13 20:48:19 -05:00
|
|
|
if err := ReadJSON(r, &security); err != nil {
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
security.SecurityId = -1
|
|
|
|
security.UserId = user.UserId
|
|
|
|
|
2017-12-07 20:08:43 -05:00
|
|
|
err = context.Tx.InsertSecurity(&security)
|
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
|
|
|
}
|
|
|
|
|
2017-10-14 14:20:50 -04:00
|
|
|
return ResponseWrapper{201, &security}
|
2016-10-16 08:19:11 -04:00
|
|
|
} else if r.Method == "GET" {
|
2017-11-12 21:12:49 -05:00
|
|
|
if context.LastLevel() {
|
2016-10-16 08:19:11 -04:00
|
|
|
//Return all securities
|
2017-12-03 06:38:22 -05:00
|
|
|
var sl models.SecurityList
|
2016-10-16 08:19:11 -04:00
|
|
|
|
2017-12-07 20:08:43 -05:00
|
|
|
securities, err := context.Tx.GetSecurities(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
|
|
|
}
|
|
|
|
|
|
|
|
sl.Securities = securities
|
2017-10-14 14:20:50 -04:00
|
|
|
return &sl
|
2016-10-16 08:19:11 -04:00
|
|
|
} else {
|
2017-11-12 21:12:49 -05:00
|
|
|
securityid, err := context.NextID()
|
|
|
|
if err != nil {
|
|
|
|
return NewError(3 /*Invalid Request*/)
|
|
|
|
}
|
2017-11-16 19:17:51 -05:00
|
|
|
|
|
|
|
if !context.LastLevel() {
|
|
|
|
if context.NextLevel() != "prices" {
|
|
|
|
return NewError(3 /*Invalid Request*/)
|
|
|
|
}
|
|
|
|
return PriceHandler(r, context, user, securityid)
|
|
|
|
}
|
|
|
|
|
2017-12-07 20:08:43 -05:00
|
|
|
security, err := context.Tx.GetSecurity(securityid, user.UserId)
|
2016-10-16 08:19:11 -04:00
|
|
|
if err != nil {
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
|
2017-10-14 14:20:50 -04:00
|
|
|
return security
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
} else {
|
2017-11-12 21:12:49 -05:00
|
|
|
securityid, err := context.NextID()
|
2016-10-16 08:19:11 -04:00
|
|
|
if err != nil {
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
2017-11-16 19:17:51 -05:00
|
|
|
if !context.LastLevel() {
|
|
|
|
if context.NextLevel() != "prices" {
|
|
|
|
return NewError(3 /*Invalid Request*/)
|
|
|
|
}
|
|
|
|
return PriceHandler(r, context, user, securityid)
|
|
|
|
}
|
|
|
|
|
2016-10-16 08:19:11 -04:00
|
|
|
if r.Method == "PUT" {
|
2017-12-03 06:38:22 -05:00
|
|
|
var security models.Security
|
2017-11-13 20:48:19 -05:00
|
|
|
if err := ReadJSON(r, &security); err != nil || security.SecurityId != securityid {
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2015-06-27 08:31:38 -04:00
|
|
|
}
|
2016-10-16 08:19:11 -04:00
|
|
|
security.UserId = user.UserId
|
|
|
|
|
2017-11-12 20:17:27 -05:00
|
|
|
err = UpdateSecurity(context.Tx, &security)
|
2015-06-27 08:31:38 -04:00
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(999 /*Internal Error*/)
|
2015-06-27 08:31:38 -04:00
|
|
|
}
|
2016-10-16 08:19:11 -04:00
|
|
|
|
2017-10-14 14:20:50 -04:00
|
|
|
return &security
|
2016-10-16 08:19:11 -04:00
|
|
|
} else if r.Method == "DELETE" {
|
2017-12-07 20:08:43 -05:00
|
|
|
security, err := context.Tx.GetSecurity(securityid, user.UserId)
|
2016-10-16 08:19:11 -04:00
|
|
|
if err != nil {
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
|
2017-12-07 20:08:43 -05:00
|
|
|
err = context.Tx.DeleteSecurity(security)
|
2017-12-07 20:47:55 -05:00
|
|
|
if _, ok := err.(store.SecurityInUseError); ok {
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(7 /*In Use Error*/)
|
2017-10-12 20:47:44 -04:00
|
|
|
} else if err != nil {
|
2016-10-16 08:19:11 -04:00
|
|
|
log.Print(err)
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(999 /*Internal Error*/)
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
|
2017-10-14 14:20:50 -04:00
|
|
|
return SuccessWriter{}
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
}
|
2017-10-14 14:20:50 -04:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2016-10-16 08:19:11 -04:00
|
|
|
}
|
|
|
|
|
2017-11-12 20:17:27 -05:00
|
|
|
func SecurityTemplateHandler(r *http.Request, context *Context) ResponseWriterWriter {
|
2016-10-16 08:19:11 -04:00
|
|
|
if r.Method == "GET" {
|
2017-12-03 06:38:22 -05:00
|
|
|
var sl models.SecurityList
|
2016-10-16 08:19:11 -04:00
|
|
|
|
|
|
|
query, _ := url.ParseQuery(r.URL.RawQuery)
|
2016-10-26 06:58:14 -04:00
|
|
|
|
|
|
|
var limit int64 = -1
|
2016-10-16 08:19:11 -04:00
|
|
|
search := query.Get("search")
|
2017-10-05 08:06:08 -04:00
|
|
|
|
2017-12-03 06:38:22 -05:00
|
|
|
var _type models.SecurityType = 0
|
2017-10-05 08:06:08 -04:00
|
|
|
typestring := query.Get("type")
|
|
|
|
if len(typestring) > 0 {
|
2017-12-03 06:38:22 -05:00
|
|
|
_type = models.GetSecurityType(typestring)
|
2017-10-05 08:06:08 -04:00
|
|
|
if _type == 0 {
|
2017-11-12 20:17:27 -05:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2017-10-05 08:06:08 -04:00
|
|
|
}
|
|
|
|
}
|
2016-10-16 08:19:11 -04:00
|
|
|
|
2016-10-26 06:58:14 -04:00
|
|
|
limitstring := query.Get("limit")
|
|
|
|
if limitstring != "" {
|
|
|
|
limitint, err := strconv.ParseInt(limitstring, 10, 0)
|
|
|
|
if err != nil {
|
2017-11-12 20:17:27 -05:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2016-10-26 06:58:14 -04:00
|
|
|
}
|
|
|
|
limit = limitint
|
|
|
|
}
|
|
|
|
|
|
|
|
securities := SearchSecurityTemplates(search, _type, limit)
|
2016-10-16 08:19:11 -04:00
|
|
|
|
|
|
|
sl.Securities = &securities
|
2017-11-12 20:17:27 -05:00
|
|
|
return &sl
|
2015-06-27 08:31:38 -04:00
|
|
|
} else {
|
2017-11-12 20:17:27 -05:00
|
|
|
return NewError(3 /*Invalid Request*/)
|
2015-06-27 08:31:38 -04:00
|
|
|
}
|
|
|
|
}
|