mirror of
https://github.com/injoyai/tdx.git
synced 2025-11-26 21:25:35 +08:00
Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
80ecdec737 | ||
|
|
ecf0365879 | ||
|
|
f1da1182ce |
20
example/CodesHTTP/main.go
Normal file
20
example/CodesHTTP/main.go
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/injoyai/logs"
|
||||||
|
"github.com/injoyai/tdx/extend"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
go extend.ListenCodesHTTP(10033)
|
||||||
|
|
||||||
|
<-time.After(time.Second * 3)
|
||||||
|
c := extend.DialCodesHTTP("http://localhost:10033")
|
||||||
|
stocks, err := c.GetStocks()
|
||||||
|
logs.PanicErr(err)
|
||||||
|
|
||||||
|
for _, v := range stocks {
|
||||||
|
println(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
22
example/TradesToKlines/main.go
Normal file
22
example/TradesToKlines/main.go
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/injoyai/logs"
|
||||||
|
"github.com/injoyai/tdx"
|
||||||
|
"github.com/injoyai/tdx/example/common"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
common.Test(func(c *tdx.Client) {
|
||||||
|
|
||||||
|
resp, err := c.GetHistoryTradeAll("20251010", "sz000001")
|
||||||
|
logs.PanicErr(err)
|
||||||
|
|
||||||
|
ks := resp.List.Klines()
|
||||||
|
|
||||||
|
for _, v := range ks {
|
||||||
|
logs.Debug(v)
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
66
extend/codes-server.go
Normal file
66
extend/codes-server.go
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
package extend
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"github.com/injoyai/conv"
|
||||||
|
"github.com/injoyai/tdx"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"path/filepath"
|
||||||
|
)
|
||||||
|
|
||||||
|
func ListenCodesHTTP(port int, filename ...string) error {
|
||||||
|
code, err := tdx.DialCodes(conv.Default(filepath.Join(tdx.DefaultDatabaseDir, "codes.db"), filename...))
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return http.ListenAndServe(fmt.Sprintf(":%d", port), http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch r.RequestURI {
|
||||||
|
case "/stocks":
|
||||||
|
ls := code.GetStocks()
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write(conv.Bytes(ls))
|
||||||
|
case "/etfs":
|
||||||
|
ls := code.GetETFs()
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write(conv.Bytes(ls))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
func DialCodesHTTP(address string) *CodesHTTP {
|
||||||
|
return &CodesHTTP{address: address}
|
||||||
|
}
|
||||||
|
|
||||||
|
type CodesHTTP struct {
|
||||||
|
address string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *CodesHTTP) getList(path string) ([]string, error) {
|
||||||
|
resp, err := http.DefaultClient.Get(this.address + path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
return nil, fmt.Errorf("http code:%d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
bs, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
ls := []string(nil)
|
||||||
|
err = json.Unmarshal(bs, &ls)
|
||||||
|
return ls, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *CodesHTTP) GetStocks() ([]string, error) {
|
||||||
|
return this.getList("/stocks")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *CodesHTTP) GetETFs() ([]string, error) {
|
||||||
|
return this.getList("/etfs")
|
||||||
|
}
|
||||||
@@ -67,11 +67,7 @@ func (this *PullTrade) PullYear(ctx context.Context, m *tdx.Manage, year int, co
|
|||||||
tss = append(tss, resp.List...)
|
tss = append(tss, resp.List...)
|
||||||
|
|
||||||
//转成分时K线
|
//转成分时K线
|
||||||
ks, err := resp.List.Klines1()
|
ks := resp.List.Klines()
|
||||||
if err != nil {
|
|
||||||
logs.Err(err)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
kss1 = append(kss1, ks...)
|
kss1 = append(kss1, ks...)
|
||||||
kss5 = append(kss5, ks.Merge(5)...)
|
kss5 = append(kss5, ks.Merge(5)...)
|
||||||
|
|||||||
@@ -233,6 +233,14 @@ func FixKlineTime(ks []*Kline) []*Kline {
|
|||||||
|
|
||||||
type Klines []*Kline
|
type Klines []*Kline
|
||||||
|
|
||||||
|
// LastPrice 获取最后一个K线的收盘价
|
||||||
|
func (this Klines) LastPrice() Price {
|
||||||
|
if len(this) == 0 {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
return this[len(this)-1].Close
|
||||||
|
}
|
||||||
|
|
||||||
func (this Klines) Len() int {
|
func (this Klines) Len() int {
|
||||||
return len(this)
|
return len(this)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ package protocol
|
|||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/injoyai/base/types"
|
||||||
"github.com/injoyai/conv"
|
"github.com/injoyai/conv"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@@ -121,52 +122,108 @@ func (trade) Decode(bs []byte, c TradeCache) (*TradeResp, error) {
|
|||||||
|
|
||||||
type Trades []*Trade
|
type Trades []*Trade
|
||||||
|
|
||||||
func (this Trades) Kline() (k *Kline, err error) {
|
// Klines 合并分时成交成k线
|
||||||
k = &Kline{}
|
func (this Trades) Klines() Klines {
|
||||||
for i, v := range this {
|
//按天分割
|
||||||
switch i {
|
m := make(types.SortMap[int64, Trades])
|
||||||
|
for _, v := range this {
|
||||||
|
//获取当天零点的时间戳
|
||||||
|
unix := time.Date(v.Time.Year(), v.Time.Month(), v.Time.Day(), 0, 0, 0, 0, v.Time.Location()).Unix()
|
||||||
|
m[unix] = append(m[unix], v)
|
||||||
|
}
|
||||||
|
|
||||||
|
//按天排序
|
||||||
|
mKline := types.SortMap[int64, Klines]{}
|
||||||
|
for date, v := range m {
|
||||||
|
//生成一分钟k线
|
||||||
|
t := time.Unix(date, 0)
|
||||||
|
mKline[date] = v.klinesForDay(t)
|
||||||
|
}
|
||||||
|
//按时间排序
|
||||||
|
lss := mKline.Sort()
|
||||||
|
ls := Klines{}
|
||||||
|
for _, v := range lss {
|
||||||
|
ls = append(ls, v...)
|
||||||
|
}
|
||||||
|
return ls
|
||||||
|
}
|
||||||
|
|
||||||
|
// Kline 合并分时成交成1个k线,注意分时成交时间保持一致
|
||||||
|
func (this Trades) Kline(t time.Time, last Price) *Kline {
|
||||||
|
k := &Kline{
|
||||||
|
Time: t,
|
||||||
|
Last: last,
|
||||||
|
Open: last,
|
||||||
|
High: last,
|
||||||
|
Low: last,
|
||||||
|
Close: last,
|
||||||
|
}
|
||||||
|
first := 0
|
||||||
|
for _, v := range this {
|
||||||
|
if v.Price <= 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
switch first {
|
||||||
case 0:
|
case 0:
|
||||||
k.Time = v.Time
|
|
||||||
k.Open = v.Price
|
k.Open = v.Price
|
||||||
k.High = v.Price
|
k.High = v.Price
|
||||||
k.Low = v.Price
|
k.Low = v.Price
|
||||||
k.Close = v.Price
|
k.Close = v.Price
|
||||||
case len(this) - 1:
|
default:
|
||||||
|
k.High = conv.Select(k.High < v.Price, v.Price, k.High)
|
||||||
|
k.Low = conv.Select(k.Low > v.Price, v.Price, k.Low)
|
||||||
|
}
|
||||||
k.Close = v.Price
|
k.Close = v.Price
|
||||||
}
|
|
||||||
k.High = conv.Select(v.Price > k.High, v.Price, k.High)
|
|
||||||
k.Low = conv.Select(v.Price < k.Low, v.Price, k.Low)
|
|
||||||
k.Volume += int64(v.Volume)
|
k.Volume += int64(v.Volume)
|
||||||
k.Amount += v.Amount()
|
k.Amount += v.Price * Price(v.Volume) * 100
|
||||||
|
first++
|
||||||
}
|
}
|
||||||
return
|
return k
|
||||||
}
|
}
|
||||||
|
|
||||||
// Klines1 1分K线
|
// kline1 生成一分钟k线,一天
|
||||||
func (this Trades) Klines1() (Klines, error) {
|
func (this Trades) klinesForDay(date time.Time) Klines {
|
||||||
m := make(map[int64]Trades)
|
_930 := 570 //9:30 的分钟
|
||||||
|
_1130 := 690 //11:30 的分钟
|
||||||
|
_1300 := 780 //13:00 的分钟
|
||||||
|
_1500 := 900 //15:00 的分钟
|
||||||
|
keys := []int(nil)
|
||||||
|
//早上
|
||||||
|
m := map[int]Trades{}
|
||||||
|
for i := 1; i <= 120; i++ {
|
||||||
|
keys = append(keys, _930+i)
|
||||||
|
m[_930+i] = []*Trade{}
|
||||||
|
}
|
||||||
|
//下午
|
||||||
|
for i := 1; i <= 120; i++ {
|
||||||
|
keys = append(keys, _1300+i)
|
||||||
|
m[_1300+i] = []*Trade{}
|
||||||
|
}
|
||||||
|
//获取开盘价,有可能前几分钟没有数据,先遍历一遍
|
||||||
|
var open Price
|
||||||
for _, v := range this {
|
for _, v := range this {
|
||||||
//小于9点30的数据归类到9点30
|
if v.Price > 0 {
|
||||||
if v.Time.Hour() == 9 && v.Time.Minute() < 30 {
|
open = v.Price
|
||||||
v.Time = time.Date(v.Time.Year(), v.Time.Month(), v.Time.Day(), 9, 30, 0, 0, v.Time.Location())
|
break
|
||||||
}
|
}
|
||||||
//15:00之前和11:30之前+1
|
|
||||||
if (v.Time.Hour() >= 13 && v.Time.Hour() < 15) || (v.Time.Hour() == 11 && v.Time.Minute() < 30) || v.Time.Hour() < 11 {
|
|
||||||
v.Time = v.Time.Add(time.Minute)
|
|
||||||
}
|
}
|
||||||
m[v.Time.Unix()] = append(m[v.Time.Unix()], v)
|
//分组,按
|
||||||
}
|
for _, v := range this {
|
||||||
|
ms := minutes(v.Time)
|
||||||
ls := Klines(nil)
|
t := conv.Select(ms <= _930, _930, ms)
|
||||||
for _, v := range m {
|
t++
|
||||||
k, err := v.Kline()
|
t = conv.Select(t > _1130 && t <= _1300, _1130, t)
|
||||||
if err != nil {
|
t = conv.Select(t > _1500, _1500, t)
|
||||||
return nil, err
|
m[t] = append(m[t], v)
|
||||||
}
|
}
|
||||||
|
//合并
|
||||||
|
ls := []*Kline(nil)
|
||||||
|
for _, v := range keys {
|
||||||
|
k := m[v].Kline(time.Date(date.Year(), date.Month(), date.Day(), v/60, v%60, 0, 0, date.Location()), open)
|
||||||
|
open = k.Close
|
||||||
ls = append(ls, k)
|
ls = append(ls, k)
|
||||||
}
|
}
|
||||||
ls.Sort()
|
return ls
|
||||||
return ls, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type TradeCache struct {
|
type TradeCache struct {
|
||||||
|
|||||||
@@ -317,3 +317,7 @@ func AddPrefix(code string) string {
|
|||||||
}
|
}
|
||||||
return code
|
return code
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func minutes(t time.Time) int {
|
||||||
|
return t.Hour()*60 + t.Minute()
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user