Files
services/stock/handler/stock.go
2021-06-29 13:27:47 +01:00

238 lines
6.1 KiB
Go

package handler
import (
"context"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
"regexp"
"time"
"github.com/micro/micro/v3/service/errors"
"github.com/micro/micro/v3/service/logger"
pb "github.com/micro/services/stock/proto"
"github.com/patrickmn/go-cache"
)
var (
re = regexp.MustCompile(`\d{4}-\d{2}-\d{2}`)
)
type Stock struct {
Api string
Key string
Cache *cache.Cache
}
type OrderBook struct {
Symbol string
Results []*Order
}
type Order struct {
Symbol string
Ask float64
Bid float64
Asize int32
Bsize int32
T int64
}
type Quote struct {
Symbol string
Ask float64
Bid float64
Asize int32
Bsize int32
Timestamp int64
}
type History struct {
Symbol string
Open float64
High float64
Low float64
Close float64
Volume int32
From string
}
func (s *Stock) OrderBook(ctx context.Context, req *pb.OrderBookRequest, rsp *pb.OrderBookResponse) error {
if len(req.Stock) <= 0 || len(req.Stock) > 5 {
return errors.BadRequest("stock.orderbook", "invalid symbol")
}
if len(req.Date) == 0 {
return errors.BadRequest("stock.orderbook", "missing date")
}
if req.Limit <= 0 {
req.Limit = 25
}
uri := fmt.Sprintf("%shistory/stock/all?apikey=%s&stock=%s&date=%s&limit=%d", s.Api, s.Key, req.Stock, req.Date, req.Limit)
if len(req.Start) > 0 {
t, err := time.Parse(time.RFC3339Nano, req.Start)
if err != nil {
return errors.BadRequest("stock.orderbook", "invalid start datetime")
}
uri = fmt.Sprintf("%s&ts=%d", uri, t.UTC().UnixNano())
}
if len(req.End) > 0 {
t, err := time.Parse(time.RFC3339Nano, req.End)
if err != nil {
return errors.BadRequest("stock.orderbook", "invalid end datetime")
}
uri = fmt.Sprintf("%s&te=%d", uri, t.UTC().UnixNano())
}
resp, err := http.Get(uri)
if err != nil {
logger.Errorf("Failed to get orderbook: %v\n", err)
return errors.InternalServerError("stock.orderbook", "failed to get orderbook")
}
defer resp.Body.Close()
b, _ := ioutil.ReadAll(resp.Body)
if resp.StatusCode != 200 {
logger.Errorf("Failed to get orderbook (non 200): %d %v\n", resp.StatusCode, string(b))
return errors.InternalServerError("stock.orderbook", "failed to get orderbook")
}
var respBody OrderBook
if err := json.Unmarshal(b, &respBody); err != nil {
logger.Errorf("Failed to unmarshal orderbook: %v\n", err)
return errors.InternalServerError("stock.orderbook", "failed to get orderbook")
}
rsp.Symbol = respBody.Symbol
rsp.Date = req.Date
for _, result := range respBody.Results {
rsp.Orders = append(rsp.Orders, &pb.Order{
AskPrice: result.Ask,
BidPrice: result.Bid,
AskSize: result.Asize,
BidSize: result.Bsize,
Timestamp: time.Unix(0, result.T).UTC().Format(time.RFC3339Nano),
})
}
return nil
}
func (s *Stock) History(ctx context.Context, req *pb.HistoryRequest, rsp *pb.HistoryResponse) error {
if len(req.Stock) <= 0 || len(req.Stock) > 5 {
return errors.BadRequest("stock.history", "invalid symbol")
}
if !re.MatchString(req.Date) {
return errors.BadRequest("stock.history", "invalid date")
}
uri := fmt.Sprintf("%shistory/stock/open-close?stock=%s&date=%s&apikey=%s", s.Api, req.Stock, req.Date, s.Key)
resp, err := http.Get(uri)
if err != nil {
logger.Errorf("Failed to get history: %v\n", err)
return errors.InternalServerError("stock.history", "failed to get history")
}
defer resp.Body.Close()
b, _ := ioutil.ReadAll(resp.Body)
if resp.StatusCode != 200 {
logger.Errorf("Failed to get history (non 200): %d %v\n", resp.StatusCode, string(b))
return errors.InternalServerError("stock.history", "failed to get history")
}
var respBody History
if err := json.Unmarshal(b, &respBody); err != nil {
logger.Errorf("Failed to unmarshal history: %v\n", err)
return errors.InternalServerError("stock.history", "failed to get history")
}
rsp.Symbol = respBody.Symbol
rsp.Open = respBody.Open
rsp.Close = respBody.Close
rsp.High = respBody.High
rsp.Low = respBody.Low
rsp.Date = respBody.From
rsp.Volume = respBody.Volume
return nil
}
func (s *Stock) Quote(ctx context.Context, req *pb.QuoteRequest, rsp *pb.QuoteResponse) error {
if len(req.Symbol) <= 0 || len(req.Symbol) > 5 {
return errors.BadRequest("stock.quote", "invalid symbol")
}
uri := fmt.Sprintf("%slast/stock/%s?apikey=%s", s.Api, req.Symbol, s.Key)
resp, err := http.Get(uri)
if err != nil {
logger.Errorf("Failed to get quote: %v\n", err)
return errors.InternalServerError("stock.quote", "failed to get quote")
}
defer resp.Body.Close()
b, _ := ioutil.ReadAll(resp.Body)
if resp.StatusCode != 200 {
logger.Errorf("Failed to get quote (non 200): %d %v\n", resp.StatusCode, string(b))
return errors.InternalServerError("stock.quote", "failed to get quote")
}
var respBody Quote
if err := json.Unmarshal(b, &respBody); err != nil {
logger.Errorf("Failed to unmarshal quote: %v\n", err)
return errors.InternalServerError("stock.quote", "failed to get quote")
}
rsp.Symbol = respBody.Symbol
rsp.AskPrice = respBody.Ask
rsp.BidPrice = respBody.Bid
rsp.AskSize = respBody.Asize
rsp.BidSize = respBody.Bsize
rsp.Timestamp = time.Unix(0, respBody.Timestamp*int64(time.Millisecond)).UTC().Format(time.RFC3339Nano)
return nil
}
func (s *Stock) Price(ctx context.Context, req *pb.PriceRequest, rsp *pb.PriceResponse) error {
if len(req.Symbol) <= 0 || len(req.Symbol) > 5 {
return errors.BadRequest("stock.price", "invalid symbol")
}
uri := fmt.Sprintf("%slast/trade/stock/%s?apikey=%s", s.Api, req.Symbol, s.Key)
resp, err := http.Get(uri)
if err != nil {
logger.Errorf("Failed to get price: %v\n", err)
return errors.InternalServerError("stock.trade", "failed to get price")
}
defer resp.Body.Close()
b, _ := ioutil.ReadAll(resp.Body)
if resp.StatusCode != 200 {
logger.Errorf("Failed to get price (non 200): %d %v\n", resp.StatusCode, string(b))
return errors.InternalServerError("stock.price", "failed to get price")
}
var respBody map[string]interface{}
if err := json.Unmarshal(b, &respBody); err != nil {
logger.Errorf("Failed to unmarshal price: %v\n", err)
return errors.InternalServerError("stock.price", "failed to get price")
}
rsp.Symbol = req.Symbol
rsp.Price = respBody["price"].(float64)
return nil
}