market_data 0.3.0 → 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +9 -1
- data/Gemfile.lock +1 -1
- data/README.md +4 -4
- data/lib/market_data/conn.rb +12 -0
- data/lib/market_data/constants.rb +21 -0
- data/lib/market_data/indexes.rb +27 -0
- data/lib/market_data/mappers.rb +27 -1
- data/lib/market_data/markets.rb +17 -0
- data/lib/market_data/models.rb +6 -0
- data/lib/market_data/quotes.rb +17 -27
- data/lib/market_data/validations.rb +41 -7
- data/lib/market_data/version.rb +1 -1
- data/lib/market_data.rb +4 -1
- metadata +4 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 76061b4644a7a3092a819581faa9b7141dfb88e064b3a78a3615aa4c76ac1b0a
|
4
|
+
data.tar.gz: '09d87650d7235f8419524a0bb4cba5a43dd1d8d1a4e0eae4cb1b2cc24fc27dbf'
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d369b7b3d1e391decd1679262e2c0f9ef8fe20463aac5ef3ebc05ce00a46c86ea94d13441e8d8d2f76ed78220a3f45b99f3cb4caf80da52d115b8e97d54fdb7f
|
7
|
+
data.tar.gz: 50a5fc1e57d70f12722d84ecac720a4649a1e71ea6f4fa12f7d521f15b2ca8940c19f95e33f93d78b2b998e299f015091812a033b61011f6a1d80ea07a98c84e
|
data/CHANGELOG.md
CHANGED
@@ -29,4 +29,12 @@
|
|
29
29
|
## [0.3.0] - 2024-10-09
|
30
30
|
|
31
31
|
- Add support for Earnings endpoint under the `earnings` method
|
32
|
-
- Introduced `Validations` module for parameter validation logic
|
32
|
+
- Introduced `Validations` module for parameter validation logic
|
33
|
+
|
34
|
+
## [0.4.0] - 2024-10-14
|
35
|
+
|
36
|
+
- Add support for Markets status endpoint under the `market_status` method
|
37
|
+
|
38
|
+
## [0.5.0] - 2024-10-15
|
39
|
+
|
40
|
+
- Add support for Indexes endpoints under `index_quote` and `ìndex_candles`
|
data/Gemfile.lock
CHANGED
data/README.md
CHANGED
@@ -2,7 +2,7 @@
|
|
2
2
|
|
3
3
|
A Ruby wrapper for the [MarketData API](https://www.marketdata.app/docs/api).
|
4
4
|
|
5
|
-

|
6
6
|
|
7
7
|
## Installation
|
8
8
|
|
@@ -139,11 +139,11 @@ From Stocks endpoints:
|
|
139
139
|
- [X] Earnings
|
140
140
|
|
141
141
|
From Markets endpoints:
|
142
|
-
- [
|
142
|
+
- [X] Status
|
143
143
|
|
144
144
|
From Indices endpoints:
|
145
|
-
- [
|
146
|
-
- [
|
145
|
+
- [X] Quotes
|
146
|
+
- [X] Candles
|
147
147
|
|
148
148
|
From Stocks endpoints:
|
149
149
|
- [ ] Support for optional parameters for Candles
|
data/lib/market_data/conn.rb
CHANGED
@@ -6,6 +6,18 @@ module MarketData
|
|
6
6
|
module Conn
|
7
7
|
include MarketData::Errors
|
8
8
|
|
9
|
+
def do_request path, query
|
10
|
+
path_hash = {
|
11
|
+
host: MarketData.base_host,
|
12
|
+
path: path,
|
13
|
+
}
|
14
|
+
path_hash[:query] = URI.encode_www_form(query) unless query.empty?
|
15
|
+
|
16
|
+
do_connect(
|
17
|
+
get_uri path_hash
|
18
|
+
)
|
19
|
+
end
|
20
|
+
|
9
21
|
def do_connect(path)
|
10
22
|
begin
|
11
23
|
conn = URI.open(path, get_auth_headers)
|
@@ -47,5 +47,26 @@ module MarketData
|
|
47
47
|
high52: "high52",
|
48
48
|
low52: "low52",
|
49
49
|
}
|
50
|
+
MARKET_STATUS_FIELD_MAPPING = {
|
51
|
+
date: "date",
|
52
|
+
status: "status"
|
53
|
+
}
|
54
|
+
INDEX_QUOTE_FIELD_MAPPING = {
|
55
|
+
symbol: "symbol",
|
56
|
+
last: "last",
|
57
|
+
change: "change",
|
58
|
+
change_pct: "changepct",
|
59
|
+
high52: "52weekHigh",
|
60
|
+
low52: "52weekLow",
|
61
|
+
updated: "updated",
|
62
|
+
}
|
63
|
+
INDEX_CANDLE_FIELD_MAPPING = {
|
64
|
+
symbol: "symbol",
|
65
|
+
open: "o",
|
66
|
+
close: "c",
|
67
|
+
low: "l",
|
68
|
+
high: "h",
|
69
|
+
time: "t",
|
70
|
+
}
|
50
71
|
end
|
51
72
|
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
module MarketData
|
2
|
+
module Indexes
|
3
|
+
include MarketData::Conn
|
4
|
+
include MarketData::Mappers
|
5
|
+
include MarketData::Validations
|
6
|
+
|
7
|
+
@@quotes = "/v1/indices/quotes/"
|
8
|
+
@@candles = "/v1/indices/candles/"
|
9
|
+
|
10
|
+
def index_quote(symbol, opts = {w52: nil})
|
11
|
+
query = validate_index_quote_input!(**opts)
|
12
|
+
|
13
|
+
map_index_quote(
|
14
|
+
do_request(@@quotes + symbol, query)
|
15
|
+
)
|
16
|
+
end
|
17
|
+
|
18
|
+
def index_candles(symbol, opts = {resolution: nil, from: nil, to: nil, countback: nil})
|
19
|
+
query = validate_index_candles_input!(**opts)
|
20
|
+
|
21
|
+
map_index_candles(
|
22
|
+
do_request("#{@@candles}#{opts[:resolution]}/#{symbol}", query),
|
23
|
+
symbol
|
24
|
+
)
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
data/lib/market_data/mappers.rb
CHANGED
@@ -5,6 +5,7 @@ module MarketData
|
|
5
5
|
include MarketData::Models
|
6
6
|
SYMBOL_RESPONSE_KEY = "symbol"
|
7
7
|
STATUS_RESPONSE_KEY = "s"
|
8
|
+
OPEN_RESPONSE_KEY = "o"
|
8
9
|
|
9
10
|
def map_quote response, i=0
|
10
11
|
Quote.new(**map_fields_for(response, :quote, i))
|
@@ -21,7 +22,7 @@ module MarketData
|
|
21
22
|
|
22
23
|
def map_candles response, symbol
|
23
24
|
ar = []
|
24
|
-
(0..(response[
|
25
|
+
(0..(response[OPEN_RESPONSE_KEY].size - 1)).each do |i|
|
25
26
|
args = map_fields_for(response, :candle, i)
|
26
27
|
args[:symbol] = symbol
|
27
28
|
ar << Candle.new(**args)
|
@@ -47,6 +48,27 @@ module MarketData
|
|
47
48
|
ar
|
48
49
|
end
|
49
50
|
|
51
|
+
def map_market_status response
|
52
|
+
MarketData::Models::MarketStatus.new(
|
53
|
+
date: response["date"][0],
|
54
|
+
status: response["status"][0],
|
55
|
+
)
|
56
|
+
end
|
57
|
+
|
58
|
+
def map_index_quote response
|
59
|
+
IndexQuote.new(**map_fields_for(response, :index_quote))
|
60
|
+
end
|
61
|
+
|
62
|
+
def map_index_candles response, symbol
|
63
|
+
ar = []
|
64
|
+
(0..(response[OPEN_RESPONSE_KEY].size - 1)).each do |i|
|
65
|
+
args = map_fields_for(response, :index_candle, i)
|
66
|
+
args[:symbol] = symbol
|
67
|
+
ar << Models::IndexCandle.new(**args)
|
68
|
+
end
|
69
|
+
ar
|
70
|
+
end
|
71
|
+
|
50
72
|
def map_fields_for(response, kind, i=0)
|
51
73
|
mapping = {}
|
52
74
|
case kind
|
@@ -54,6 +76,10 @@ module MarketData
|
|
54
76
|
mapping = Constants::CANDLE_FIELD_MAPPING
|
55
77
|
when :earning
|
56
78
|
mapping = Constants::EARNING_FIELD_MAPPING
|
79
|
+
when :index_candle
|
80
|
+
mapping = Constants::INDEX_CANDLE_FIELD_MAPPING
|
81
|
+
when :index_quote
|
82
|
+
mapping = Constants::INDEX_QUOTE_FIELD_MAPPING
|
57
83
|
when :quote
|
58
84
|
mapping = Constants::QUOTE_FIELD_MAPPING
|
59
85
|
else
|
@@ -0,0 +1,17 @@
|
|
1
|
+
module MarketData
|
2
|
+
module Markets
|
3
|
+
include MarketData::Validations
|
4
|
+
|
5
|
+
@@status = "/v1/markets/status/"
|
6
|
+
|
7
|
+
def market_status(country: nil, date: nil, from: nil, to: nil, countback: nil)
|
8
|
+
query = validate_market_status_input!(country: country, date: date, from: from, to: to, countback: countback)
|
9
|
+
|
10
|
+
path_hash = { host: MarketData.base_host, path: @@status }
|
11
|
+
path_hash[:query] = URI.encode_www_form(query)
|
12
|
+
|
13
|
+
res = do_connect(get_uri path_hash)
|
14
|
+
map_market_status res
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
data/lib/market_data/models.rb
CHANGED
@@ -15,5 +15,11 @@ module MarketData
|
|
15
15
|
end
|
16
16
|
|
17
17
|
Earning = Struct.new(*Constants::EARNING_FIELD_MAPPING.keys)
|
18
|
+
|
19
|
+
MarketStatus = Struct.new(*Constants::MARKET_STATUS_FIELD_MAPPING.keys)
|
20
|
+
|
21
|
+
IndexQuote = Struct.new(*Constants::INDEX_QUOTE_FIELD_MAPPING.keys)
|
22
|
+
|
23
|
+
IndexCandle = Struct.new(*Constants::INDEX_CANDLE_FIELD_MAPPING.keys)
|
18
24
|
end
|
19
25
|
end
|
data/lib/market_data/quotes.rb
CHANGED
@@ -18,24 +18,18 @@ module MarketData
|
|
18
18
|
|
19
19
|
def quote(symbol, w52 = false, extended = false)
|
20
20
|
query = validate_quotes_input!(symbol: symbol, w52: w52, extended: extended)
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
end
|
26
|
-
|
27
|
-
res = do_connect(get_uri path_hash)
|
28
|
-
map_quote(res)
|
21
|
+
|
22
|
+
map_quote(
|
23
|
+
do_request @@single + symbol, query
|
24
|
+
)
|
29
25
|
end
|
30
26
|
|
31
27
|
def bulk_quotes(symbols, snapshot = false, extended = false)
|
32
28
|
query = validate_bulk_quotes_input!(symbols: symbols, snapshot: snapshot, extended: extended)
|
33
29
|
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
res = do_connect(get_uri path_hash)
|
38
|
-
map_bulk_quotes res
|
30
|
+
map_bulk_quotes(
|
31
|
+
do_request @@bulk, query
|
32
|
+
)
|
39
33
|
end
|
40
34
|
|
41
35
|
def candles(symbol, opts = {})
|
@@ -43,31 +37,27 @@ module MarketData
|
|
43
37
|
opts = defaults.merge(opts)
|
44
38
|
query = validate_candles_input!(**opts)
|
45
39
|
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
map_candles res, symbol
|
40
|
+
map_candles(
|
41
|
+
do_request(@@candles + opts[:resolution] + "/" + symbol, query),
|
42
|
+
symbol
|
43
|
+
)
|
51
44
|
end
|
52
45
|
|
53
46
|
def bulk_candles(symbols, resolution = "D")
|
54
47
|
query = validate_bulk_candles_input!(symbols: symbols, resolution: resolution)
|
55
48
|
query = query.except(:resolution)
|
56
49
|
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
res = do_connect(get_uri path_hash)
|
61
|
-
map_bulk_candles res
|
50
|
+
map_bulk_candles(
|
51
|
+
do_request @@bulk_candles + resolution + "/", query
|
52
|
+
)
|
62
53
|
end
|
63
54
|
|
64
55
|
def earnings(symbol, opts = {from: nil, to: nil, countback: nil, date: nil, report: nil})
|
65
|
-
path_hash = { host: MarketData.base_host(), path: @@earnings + symbol}
|
66
56
|
query = validate_earnings_input!(**opts)
|
67
|
-
path_hash[:query] = URI.encode_www_form(query)
|
68
57
|
|
69
|
-
|
70
|
-
|
58
|
+
map_earning(
|
59
|
+
do_request @@earnings + symbol, query
|
60
|
+
)
|
71
61
|
end
|
72
62
|
end
|
73
63
|
end
|
@@ -56,10 +56,7 @@ module MarketData
|
|
56
56
|
r.merge({symbols: symbols.join(",")})
|
57
57
|
end
|
58
58
|
|
59
|
-
def validate_candles_input!(
|
60
|
-
resolution: nil, from: nil, to: nil, countback: nil
|
61
|
-
)
|
62
|
-
|
59
|
+
def validate_candles_input!(resolution: nil, from: nil, to: nil, countback: nil)
|
63
60
|
state, response = validate_from_to_countback_strategy(from: from, to: to, countback: countback)
|
64
61
|
if state == :invalid
|
65
62
|
raise BadParameterError.new(response)
|
@@ -73,9 +70,7 @@ module MarketData
|
|
73
70
|
response.merge(res)
|
74
71
|
end
|
75
72
|
|
76
|
-
def validate_earnings_input!(
|
77
|
-
from: nil, to: nil, countback: nil, date: nil, report: nil
|
78
|
-
)
|
73
|
+
def validate_earnings_input!(from: nil, to: nil, countback: nil, date: nil, report: nil)
|
79
74
|
if !date.nil?
|
80
75
|
return {date: date}
|
81
76
|
end
|
@@ -91,6 +86,45 @@ module MarketData
|
|
91
86
|
return response
|
92
87
|
end
|
93
88
|
|
89
|
+
def validate_market_status_input!(country: nil, date: nil, from: nil, to: nil, countback: nil)
|
90
|
+
result = {}
|
91
|
+
|
92
|
+
if [country, date, from, to, countback].all? { |x| x.nil? }
|
93
|
+
return result
|
94
|
+
end
|
95
|
+
|
96
|
+
if !country.nil?
|
97
|
+
result.merge!({country: country})
|
98
|
+
end
|
99
|
+
|
100
|
+
if !date.nil?
|
101
|
+
# date has higher priority than from-to-countback
|
102
|
+
return result.merge({date: date})
|
103
|
+
end
|
104
|
+
|
105
|
+
if [from, to, countback].all? { |x| x.nil? }
|
106
|
+
return result
|
107
|
+
else
|
108
|
+
state, response = validate_from_to_countback_strategy(from: from, to: to, countback: countback)
|
109
|
+
if state == :invalid
|
110
|
+
raise BadParameterError.new(response)
|
111
|
+
end
|
112
|
+
|
113
|
+
return result.merge(response)
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
117
|
+
def validate_index_quote_input!(w52: nil)
|
118
|
+
if w52.nil? || !w52
|
119
|
+
return {}
|
120
|
+
end
|
121
|
+
{"52week" => w52}
|
122
|
+
end
|
123
|
+
|
124
|
+
def validate_index_candles_input!(resolution: nil, from: nil, to: nil, countback: nil)
|
125
|
+
validate_candles_input!(resolution: resolution, from: from, to: to, countback: countback)
|
126
|
+
end
|
127
|
+
|
94
128
|
def validate_from_to_countback_strategy(
|
95
129
|
from: nil, to: nil, countback: nil
|
96
130
|
)
|
data/lib/market_data/version.rb
CHANGED
data/lib/market_data.rb
CHANGED
@@ -2,13 +2,16 @@
|
|
2
2
|
|
3
3
|
require_relative "market_data/version"
|
4
4
|
require "market_data/quotes"
|
5
|
+
require "market_data/markets"
|
6
|
+
require "market_data/indexes"
|
5
7
|
|
6
8
|
module MarketData
|
7
9
|
@@base_host = "api.marketdata.app"
|
8
10
|
|
9
11
|
class Client
|
10
12
|
include MarketData::Quotes
|
11
|
-
|
13
|
+
include MarketData::Markets
|
14
|
+
include MarketData::Indexes
|
12
15
|
|
13
16
|
def initialize token
|
14
17
|
@access_token = token
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: market_data
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.5.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Sebastián González
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date: 2024-10-
|
11
|
+
date: 2024-10-16 00:00:00.000000000 Z
|
12
12
|
dependencies: []
|
13
13
|
description: A Ruby client for the MarketData API.
|
14
14
|
email:
|
@@ -28,7 +28,9 @@ files:
|
|
28
28
|
- lib/market_data/conn.rb
|
29
29
|
- lib/market_data/constants.rb
|
30
30
|
- lib/market_data/errors.rb
|
31
|
+
- lib/market_data/indexes.rb
|
31
32
|
- lib/market_data/mappers.rb
|
33
|
+
- lib/market_data/markets.rb
|
32
34
|
- lib/market_data/models.rb
|
33
35
|
- lib/market_data/quotes.rb
|
34
36
|
- lib/market_data/validations.rb
|