market_data 0.2.0 → 0.3.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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: '082a3b709ed32b745807906d0324a3b839a90c95e55bc199b564db75518d1fbc'
4
- data.tar.gz: dab55b3c50200dbf92e071dc0bd3eab5b8d712928f82d045c92a80a5f09b0979
3
+ metadata.gz: c83d3644178f120e069d283ebfc132d543ac8dbe2aca9734388d2cd56274bf7c
4
+ data.tar.gz: eeef52d93f45e64b4869749bd787b0af576924a0b617e8897c4eada6dcf813b2
5
5
  SHA512:
6
- metadata.gz: 64fca38c4e459cb4244f845ad4137939249c353b2e91922cc0bb416a2b1129871916c80bf0d44e3a71e8c48b797de2a27f378192a790286aed9d2f985345c556
7
- data.tar.gz: 97b5508677cbff6418254771f78e68eb48c0ee66c067274663a40ad2c6cd00af3355dac9b93bd812e90290f2fb1c10af2d0b6ba3aa447bcc760f43024ccc8dc9
6
+ metadata.gz: b01a71a756244d6461e19cad34abb3b25df2705b43e03e47a936a66a471879f8ae0038e49ab01f062ec0db4c046e40e2d80d3162a5c065f535ffa7877efed548
7
+ data.tar.gz: fe1a289aa65eb0e6a0682b9e263a232f0eb2b913a808b9c6ac5271e46b52123a4c805a586bc5b2a1b9aeda92c17fb640815e6230907a9dcf6a0c5379bcd34b59
data/CHANGELOG.md CHANGED
@@ -19,4 +19,14 @@
19
19
  - Internal rework of code. New modules for models, mappers, constants and errors
20
20
  - Introduced unit tests for almost all modules.
21
21
  - Introduced coverage. Currently at 85%
22
- - Add functionality for [Candles](https://www.marketdata.app/docs/api/stocks/candles) and [Bulk Candles](https://www.marketdata.app/docs/api/stocks/bulkcandles) endpoints
22
+ - Add functionality for [Candles](https://www.marketdata.app/docs/api/stocks/candles) and [Bulk Candles](https://www.marketdata.app/docs/api/stocks/bulkcandles) endpoints
23
+
24
+ ## [0.2.1] - 2024-10-09
25
+
26
+ - Fix broken tests
27
+ - Add support for new optional parameters for `quotes` and `bulk_quotes` endpoint
28
+
29
+ ## [0.3.0] - 2024-10-09
30
+
31
+ - Add support for Earnings endpoint under the `earnings` method
32
+ - Introduced `Validations` module for parameter validation logic
data/README.md CHANGED
@@ -2,17 +2,130 @@
2
2
 
3
3
  A Ruby wrapper for the [MarketData API](https://www.marketdata.app/docs/api).
4
4
 
5
+ ![coverage](https://img.shields.io/badge/coverage%3A-87.77%25-yellow.svg)
6
+
5
7
  ## Installation
6
8
 
7
- $ gem install market-data
8
- $ bundle install
9
+ $ gem install market_data
9
10
 
10
11
  ## Usage
11
12
 
12
- $ client = MarketData.Client.new "YOUR_API_TOKEN"
13
+ You must instantiate a `MarketData::Client` before running any method. You'll need your MarketData API token first.
14
+
15
+ $ client = MarketData::Client.new "YOUR_API_TOKEN"
16
+
13
17
  $ client.quote("AAPL")
14
18
  $ => <struct MarketData::Models::Quote symbol="AAPL", ask=231.42, askSize=2, bid=231.4, .....
15
19
 
20
+ ### Quotes
21
+ For getting a single quote, run the `quote` method just as the example above.
22
+
23
+ #### Optional parameters
24
+
25
+ * `w52`
26
+
27
+ For getting fields with 52-week low and high information, pass true as the `w52` parameter.
28
+
29
+ $ q = client.quote("AAPL", w52=true)
30
+ $ => q.high52 = 252.11
31
+ $ => q.low52 = 222.35
32
+ **It is `false` by default.**
33
+
34
+ * `extended`
35
+
36
+ For getting a quote when market is on extended hours, you have to supply the `extended` parameter as true.
37
+
38
+ **It is false by default**, so if you fetch a quote during extended hours without the parameter, you'll always get the quote at closing time.
39
+
40
+ ### Bulk quotes
41
+
42
+ For getting multiple quotes in a single request, use the `bulk_candle` method.
43
+
44
+ $ quotes = client.bulk_quotes(["AAPL", "AMD", "NOTAQUOTE])
45
+
46
+ $ quotes["AMD"] => <struct MarketData::Models::Quote symbol="AMD", ask=150.42, askSize=2, bid=146.4, .....
47
+
48
+ $ quotes["NOTAQUOTE"] = nil
49
+
50
+ If a quote is not found, the hashmap will return a nil value for that ticker's key.
51
+
52
+ #### Optional parameters
53
+
54
+ * `snapshot`
55
+
56
+ If snapshot is true, any supplied array of symbols will be ignored and a complete snapshot of the market ticker's will be returned.
57
+
58
+ $ quotes = client.bulk_quotes([], snapshot = true)
59
+
60
+ $ quotes["A"] => <struct MarketData::Models::Quote symbol="A", ask=56.32, askSize=45, bid=67, .....
61
+ ....
62
+ $ quotes["Z"] => <struct MarketData::Models::Quote symbol="Z", ask=25, askSize=3, bid=14.5, .....
63
+
64
+ **This could use all you API credits. Use with caution.**
65
+
66
+ * `extended`
67
+
68
+ For getting a quote when market is on extended hours, you have to supply the `extended` parameter as true.
69
+
70
+ **It is false by default**, so if you fetch a quote during extended hours without the parameter, you'll get the quote at closing time.
71
+
72
+ ### Candles
73
+ For getting ticker candles, you'll need to specify:
74
+ * a ticker symbol
75
+ * a resolution (`M`, `D`, `W`, etc. See [docs](https://www.marketdata.app/docs/api/stocks/candles#request-parameters) for a complete list)
76
+ * a strategy to specfy a date range. You can use `from` and `to` OR `to` and `countback`.
77
+
78
+
79
+ As an example, for getting candles for last week, for the first strategy:
80
+
81
+ $ quotes = client.candles("AAPL", "D", (Time.current - 1.week).iso8601, Time.current.iso8601, nil)
82
+
83
+ and for the second
84
+
85
+ $ quotes = client.candles("AAPL", "D", nil, Time.current.iso8601, 7)
86
+
87
+ `to` and `from` can receive an ISO 8601 compliant utc format or a unix timestamp.
88
+
89
+ ### Bulk candles
90
+
91
+ For the `bulk_candles` method you pass a array of ticker symbols. Resolution is daily by default, although any daily variation will work as well (like `2D`, `3D`, etc.)
92
+
93
+ It returns a hashmap with the ticker symbol as a key.
94
+
95
+ $ candles = client.bulk_candles(["AAPL", "AMD", "NOTAQUOTE"])
96
+
97
+ $ candles["AMD"]
98
+ $ => #<struct MarketData::Models::Candle symbol="AMD", open=174.05, high=174.05, low=169.55, close=171.02, volume=33391035, time=1728446400>
99
+ $ candles["AAPL"]
100
+ $ => #<struct MarketData::Models::Candle symbol="AAPL", open=225.23, high=229.75, low=224.83, close=229.54, volume=31398884, time=1728446400>
101
+ $ candles["NOTAQUOTE"] => nil
102
+
103
+ If a quote is not found, the hashmap will return a nil value for that ticker's key.
104
+
105
+ ### Earnings
106
+
107
+ See the API [docs](https://www.marketdata.app/docs/api/stocks/earnings) for parameter specification.
108
+
109
+ $ client.earnings("AAPL", from: (Time.now - MarketData::Constants::YEAR).iso8601, to: Time.now.iso8601, countback: nil, report: nil, date: nil)
110
+ $ => [#<struct MarketData::Models::Earning
111
+ symbol="AAPL",
112
+ fiscal_year=2023,
113
+ fiscal_quarter=4,
114
+ date=1696046400,
115
+ report_date=1698897600,
116
+ report_time="after close",
117
+ currency="USD",
118
+ reported_eps=1.46,
119
+ estimated_eps=1.39,
120
+ surprise_eps=0.07,
121
+ surprise_eps_pct=0.0504,
122
+ updated=1728273600>,
123
+ #<struct MarketData::Models::Earning
124
+ symbol="AAPL",
125
+ fiscal_year=2024,
126
+ fiscal_quarter=1,
127
+ ...
128
+
16
129
  ## ROADMAP
17
130
 
18
131
  The following is an ordered list of next expected developments, based on the endpoints present in the [docs](https://www.marketdata.app/docs/api)
@@ -22,19 +135,19 @@ From Stocks endpoints:
22
135
  - [X] Bulk Stocks
23
136
  - [X] Candles
24
137
  - [X] Bulk Candles
25
- - [X] Support for optional parameters for Bulk Candles
26
- - [] Earnings
138
+ - [X] Support for new optional parameters for Quotes and Bulk Quotes
139
+ - [X] Earnings
27
140
 
28
141
  From Markets endpoints:
29
- - [] Status
142
+ - [ ] Status
30
143
 
31
144
  From Indices endpoints:
32
- - [] Quotes
33
- - [] Candles
145
+ - [ ] Quotes
146
+ - [ ] Candles
34
147
 
35
148
  From Stocks endpoints:
36
- - [] Support for optional parameters for Candles
37
- - [] Support for optional parameters for Bulk Candles
149
+ - [ ] Support for optional parameters for Candles
150
+ - [ ] Support for optional parameters for Bulk Candles
38
151
 
39
152
 
40
153
  ## Tests
@@ -5,9 +5,47 @@ module MarketData
5
5
  HOUR = MINUTE * 60
6
6
  DAY = HOUR * 24
7
7
  WEEK = DAY * 7
8
+ MONTH_30 = DAY * 30
9
+ MONTH_31 = DAY * 31
8
10
  YEAR = DAY * 365
9
11
 
10
- QUOTE_FIELDS = [:symbol, :ask, :askSize, :bid, :bidSize, :mid, :last, :change, :changepct, :volume, :updated, :high52, :low52]
11
- CANDLE_FIELDS = [:symbol, :open, :high, :low, :close, :volume, :time]
12
+ EARNING_FIELD_MAPPING = {
13
+ symbol: "symbol",
14
+ fiscal_year: "fiscalYear",
15
+ fiscal_quarter: "fiscalQuarter",
16
+ date: "date",
17
+ report_date: "reportDate",
18
+ report_time: "reportTime",
19
+ currency: "currency",
20
+ reported_eps: "reportedEPS",
21
+ estimated_eps: "estimatedEPS",
22
+ surprise_eps: "surpriseEPS",
23
+ surprise_eps_pct: "surpriseEPSpct",
24
+ updated: "updated"
25
+ }
26
+ CANDLE_FIELD_MAPPING = {
27
+ symbol: "symbol",
28
+ open: "o",
29
+ close: "c",
30
+ low: "l",
31
+ high: "h",
32
+ volume: "v",
33
+ time: "t",
34
+ }
35
+ QUOTE_FIELD_MAPPING = {
36
+ symbol: "symbol",
37
+ ask: "ask",
38
+ ask_size: "askSize",
39
+ bid: "bid",
40
+ bid_size: "bidSize",
41
+ mid: "mid",
42
+ last: "last",
43
+ change: "change",
44
+ change_pct: "changepct",
45
+ volume: "volume",
46
+ updated: "updated",
47
+ high52: "high52",
48
+ low52: "low52",
49
+ }
12
50
  end
13
51
  end
@@ -3,32 +3,16 @@ require 'market_data/models'
3
3
  module MarketData
4
4
  module Mappers
5
5
  include MarketData::Models
6
-
7
6
  SYMBOL_RESPONSE_KEY = "symbol"
8
7
  STATUS_RESPONSE_KEY = "s"
9
8
 
10
9
  def map_quote response, i=0
11
- Quote.new(
12
- symbol: response["symbol"][i],
13
- ask: response["ask"][i],
14
- askSize: response["askSize"][i],
15
- bid: response["bid"][i],
16
- bidSize: response["bidSize"][i],
17
- mid: response["mid"][i],
18
- last: response["last"][i],
19
- change: response["change"][i],
20
- changepct: response["changepct"][i],
21
- volume: response["volume"][i],
22
- updated: response["updated"][i],
23
- high52: response.fetch("high52", nil),
24
- low52: response.fetch("low52", nil),
25
- )
10
+ Quote.new(**map_fields_for(response, :quote, i))
26
11
  end
27
12
 
28
13
  def map_bulk_quotes response
29
14
  h = Hash.new
30
- size = response[SYMBOL_RESPONSE_KEY].size
31
- for i in 0..(size - 1) do
15
+ (0..(response[SYMBOL_RESPONSE_KEY].size - 1)).each do |i|
32
16
  qquote = map_quote(response, i)
33
17
  h[response[SYMBOL_RESPONSE_KEY][i]] = !qquote.blank? ? qquote : nil
34
18
  end
@@ -37,31 +21,50 @@ module MarketData
37
21
 
38
22
  def map_candles response, symbol
39
23
  ar = []
40
- size = response["o"].size
41
-
42
- for i in 0..(size - 1) do
43
- ar << Candle.new(
44
- open: response["o"][i],
45
- high: response["h"][i],
46
- low: response["l"][i],
47
- close: response["c"][i],
48
- volume: response["v"][i],
49
- time: response["t"][i],
50
- symbol: symbol
51
- )
24
+ (0..(response["o"].size - 1)).each do |i|
25
+ args = map_fields_for(response, :candle, i)
26
+ args[:symbol] = symbol
27
+ ar << Candle.new(**args)
52
28
  end
29
+
53
30
  ar
54
31
  end
55
32
 
56
33
  def map_bulk_candles response
57
34
  h = Hash.new
58
- size = response[SYMBOL_RESPONSE_KEY].size
59
-
60
- for i in 0..(size - 1) do
61
- candle = Candle.new(symbol: response["symbol"][i], open: response["o"][i], high: response["h"][i], low: response["l"][i], close: response["c"][i], volume: response["v"][i], time: response["t"][i])
35
+ (0..(response[SYMBOL_RESPONSE_KEY].size - 1)).each do |i|
36
+ candle = Candle.new(**map_fields_for(response, :candle, i))
62
37
  h[response[SYMBOL_RESPONSE_KEY][i]] = !candle.blank? ? candle : nil
63
38
  end
64
39
  h
65
40
  end
41
+
42
+ def map_earning response
43
+ ar = []
44
+ (0..(response[SYMBOL_RESPONSE_KEY].size - 1)).each do |i|
45
+ ar << Earning.new(**map_fields_for(response, :earning, i))
46
+ end
47
+ ar
48
+ end
49
+
50
+ def map_fields_for(response, kind, i=0)
51
+ mapping = {}
52
+ case kind
53
+ when :candle
54
+ mapping = Constants::CANDLE_FIELD_MAPPING
55
+ when :earning
56
+ mapping = Constants::EARNING_FIELD_MAPPING
57
+ when :quote
58
+ mapping = Constants::QUOTE_FIELD_MAPPING
59
+ else
60
+ raise BadParameterError.new("unrecognized model for mapping: #{kind}")
61
+ end
62
+
63
+ r = {}
64
+ mapping.each do |field, mapped|
65
+ r.store(field, response.fetch(mapped, nil).nil? ? nil : response.fetch(mapped)[i])
66
+ end
67
+ r
68
+ end
66
69
  end
67
70
  end
@@ -2,18 +2,18 @@ require 'market_data/constants'
2
2
 
3
3
  module MarketData
4
4
  module Models
5
- include MarketData::Constants
6
-
7
- Quote = Struct.new(*Constants::QUOTE_FIELDS) do
5
+ Quote = Struct.new(*Constants::QUOTE_FIELD_MAPPING.keys) do
8
6
  def blank?
9
- (QUOTE_FIELDS - [:symbol]).all? { |mmethod| self[mmethod].nil?}
7
+ (Constants::QUOTE_FIELD_MAPPING.keys - [:symbol]).all? { |mmethod| self[mmethod].nil?}
10
8
  end
11
9
  end
12
10
 
13
- Candle = Struct.new(*Constants::CANDLE_FIELDS) do
11
+ Candle = Struct.new(*Constants::CANDLE_FIELD_MAPPING.keys) do
14
12
  def blank?
15
- (CANDLE_FIELDS - [:symbol]).all? { |mmethod| self[mmethod].nil?}
13
+ (Constants::CANDLE_FIELD_MAPPING.keys - [:symbol]).all? { |mmethod| self[mmethod].nil?}
16
14
  end
17
15
  end
16
+
17
+ Earning = Struct.new(*Constants::EARNING_FIELD_MAPPING.keys)
18
18
  end
19
19
  end
@@ -1,41 +1,38 @@
1
1
  require 'market_data/conn'
2
2
  require 'market_data/errors'
3
3
  require 'market_data/mappers'
4
+ require 'market_data/validations'
4
5
 
5
6
  module MarketData
6
7
  module Quotes
7
8
  include MarketData::Mappers
8
9
  include MarketData::Errors
9
10
  include MarketData::Conn
11
+ include MarketData::Validations
10
12
 
11
13
  @@single = "/v1/stocks/quotes/"
12
14
  @@bulk = "/v1/stocks/bulkquotes/"
13
15
  @@candles = "/v1/stocks/candles/"
14
16
  @@bulk_candles = "/v1/stocks/bulkcandles/"
17
+ @@earnings = "/v1/stocks/earnings/"
15
18
 
16
- def quote(symbol, w52 = false)
19
+ def quote(symbol, w52 = false, extended = false)
20
+ query = validate_quotes_input!(symbol: symbol, w52: w52, extended: extended)
21
+
17
22
  path_hash = { host: MarketData.base_host, path: @@single + symbol }
18
- if w52
19
- path_hash[:query] = URI.encode_www_form({"52week" => true })
23
+ if !query.empty?
24
+ path_hash[:query] = URI.encode_www_form(query)
20
25
  end
26
+
21
27
  res = do_connect(get_uri path_hash)
22
28
  map_quote(res)
23
29
  end
24
30
 
25
- def bulk_quotes(symbols, snapshot = false)
26
- path_hash = { host: MarketData.base_host, path: @@bulk }
27
- query_hash = {}
28
-
29
- if snapshot
30
- query_hash[:snapshot] = true
31
- else
32
- if !symbols.is_a?(Array) || symbols.size < 1
33
- raise BadParameterError.new("symbols must be a non-empty list")
34
- end
35
- query_hash = { symbols: symbols.join(",") }
36
- end
31
+ def bulk_quotes(symbols, snapshot = false, extended = false)
32
+ query = validate_bulk_quotes_input!(symbols: symbols, snapshot: snapshot, extended: extended)
37
33
 
38
- path_hash[:query] = URI.encode_www_form(query_hash)
34
+ path_hash = { host: MarketData.base_host, path: @@bulk }
35
+ path_hash[:query] = URI.encode_www_form(query)
39
36
 
40
37
  res = do_connect(get_uri path_hash)
41
38
  map_bulk_quotes res
@@ -44,43 +41,33 @@ module MarketData
44
41
  def candles(symbol, opts = {})
45
42
  defaults = {resolution: "D", from: nil, to: Time.now.utc.to_i, countback: nil}
46
43
  opts = defaults.merge(opts)
47
-
48
- query_hash = {to: opts[:to]}
49
-
50
- # TODO Move method validations into own class
51
- # TODO check to is either iso8601 or unix
52
- if opts[:from].nil? && opts[:countback].nil?
53
- raise BadParameterError.new("either :from or :countback must be supplied")
54
- end
55
-
56
- if opts[:from].nil?
57
- query_hash[:countback] = opts[:countback]
58
- else
59
- query_hash[:from] = opts[:from]
60
- end
44
+ query = validate_candles_input!(**opts)
61
45
 
62
46
  path_hash = { host: MarketData.base_host, path: @@candles + opts[:resolution] + "/" + symbol }
63
- path_hash[:query] = URI.encode_www_form(query_hash)
47
+ path_hash[:query] = URI.encode_www_form(query)
64
48
 
65
49
  res = do_connect(get_uri path_hash)
66
50
  map_candles res, symbol
67
51
  end
68
52
 
69
53
  def bulk_candles(symbols, resolution = "D")
70
- unless resolution == "daily" || resolution == "1D" || resolution == "D"
71
- raise BadParameterError.new("only daily resolution is allowed for this endpoint")
72
- end
54
+ query = validate_bulk_candles_input!(symbols: symbols, resolution: resolution)
55
+ query = query.except(:resolution)
56
+
73
57
  path_hash = { host: MarketData.base_host, path: @@bulk_candles + resolution + "/" }
58
+ path_hash[:query] = URI.encode_www_form(query)
74
59
 
75
- if !symbols.is_a?(Array) || symbols.size < 1
76
- raise BadParameterError.new("symbols must be a non-empty list")
77
- end
78
- query_hash = { symbols: symbols.join(",") }
60
+ res = do_connect(get_uri path_hash)
61
+ map_bulk_candles res
62
+ end
79
63
 
80
- path_hash[:query] = URI.encode_www_form(query_hash)
64
+ 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
+ query = validate_earnings_input!(**opts)
67
+ path_hash[:query] = URI.encode_www_form(query)
81
68
 
82
69
  res = do_connect(get_uri path_hash)
83
- map_bulk_candles res
70
+ map_earning res
84
71
  end
85
72
  end
86
73
  end
@@ -0,0 +1,128 @@
1
+ require 'date'
2
+
3
+ module MarketData
4
+ module Validations
5
+ include MarketData::Errors
6
+
7
+ VALID_DAILY_RESOLUTION = ['daily', 'D', '1D', '2D', '3D', '4D', '5D']
8
+ VALID_RESOLUTIONS = [
9
+ *VALID_DAILY_RESOLUTION,
10
+ 'weekly', 'W', '1W', '2W', '3W', '4W', '5W',
11
+ 'monthly', 'M', '1M', '2M', '3M', '4M', '5M',
12
+ 'yearly', 'Y', '1Y', '2Y', '3Y', '4Y', '5Y',
13
+ ]
14
+
15
+ def validate_quotes_input!(symbol: nil, w52: nil, extended: nil)
16
+ result = {}
17
+
18
+ if w52
19
+ result.merge!({"52week" => true})
20
+ end
21
+ # MarketData API considers extended as true by default. Should be included
22
+ # in the query when false
23
+ if !extended
24
+ result.merge!({extended: false})
25
+ end
26
+
27
+ result
28
+ end
29
+
30
+ def validate_bulk_quotes_input!(symbols: nil, snapshot: nil, extended: nil)
31
+ result = {extended: false}
32
+ if snapshot
33
+ result.merge!({snapshot: true})
34
+ else
35
+ if !symbols.kind_of?(Array) || symbols.size < 2
36
+ raise BadParameterError.new("symbols must be list with at least 2 symbols")
37
+ end
38
+ result.merge!({symbols: symbols.join(",")})
39
+ end
40
+ if extended
41
+ result.merge!({extended: true})
42
+ end
43
+
44
+ result
45
+ end
46
+
47
+ def validate_bulk_candles_input!(resolution: nil, symbols: nil)
48
+ s, r = validate_resolution(resolution, VALID_DAILY_RESOLUTION)
49
+ if s == :invalid
50
+ raise BadParameterError.new(r)
51
+ end
52
+ if !symbols.kind_of?(Array) || symbols.size < 2
53
+ raise BadParameterError.new("symbols must be list with at least 2 symbols")
54
+ end
55
+
56
+ r.merge({symbols: symbols.join(",")})
57
+ end
58
+
59
+ def validate_candles_input!(
60
+ resolution: nil, from: nil, to: nil, countback: nil
61
+ )
62
+
63
+ state, response = validate_from_to_countback_strategy(from: from, to: to, countback: countback)
64
+ if state == :invalid
65
+ raise BadParameterError.new(response)
66
+ end
67
+
68
+ state, res = validate_resolution(resolution)
69
+ if state == :invalid
70
+ raise BadParameterError.new(res)
71
+ end
72
+
73
+ response.merge(res)
74
+ end
75
+
76
+ def validate_earnings_input!(
77
+ from: nil, to: nil, countback: nil, date: nil, report: nil
78
+ )
79
+ if !date.nil?
80
+ return {date: date}
81
+ end
82
+ if !report.nil?
83
+ return {report: report}
84
+ end
85
+
86
+ state, response = validate_from_to_countback_strategy(from: from, to: to, countback: countback)
87
+ if state == :invalid
88
+ raise BadParameterError.new(response)
89
+ end
90
+
91
+ return response
92
+ end
93
+
94
+ def validate_from_to_countback_strategy(
95
+ from: nil, to: nil, countback: nil
96
+ )
97
+ if !from.nil? && !to.nil?
98
+ return :valid, {from: from, to: to}
99
+ end
100
+ if !to.nil? && !countback.nil? && from.nil?
101
+ return :valid, {to: to, countback: countback}
102
+ end
103
+
104
+ return :invalid, "supply either :from and :to, or :to and :countback"
105
+ end
106
+
107
+ def validate_resolution resolution, allowed_values = VALID_RESOLUTIONS
108
+ if VALID_RESOLUTIONS.include? resolution
109
+ return :valid, {resolution: resolution}
110
+ end
111
+ return :invalid, "invalid resolution: #{resolution}"
112
+ end
113
+
114
+ def time_valid?(t)
115
+ if t.kind_of?(String)
116
+ begin
117
+ DateTime.iso8601(t)
118
+ return true
119
+ rescue
120
+ return false
121
+ end
122
+ end
123
+ if t.kind_of?(Integer)
124
+ return t > 0
125
+ end
126
+ end
127
+ end
128
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module MarketData
4
- VERSION = "0.2.0"
4
+ VERSION = "0.3.0"
5
5
  end
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.2.0
4
+ version: 0.3.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-09-30 00:00:00.000000000 Z
11
+ date: 2024-10-11 00:00:00.000000000 Z
12
12
  dependencies: []
13
13
  description: A Ruby client for the MarketData API.
14
14
  email:
@@ -31,6 +31,7 @@ files:
31
31
  - lib/market_data/mappers.rb
32
32
  - lib/market_data/models.rb
33
33
  - lib/market_data/quotes.rb
34
+ - lib/market_data/validations.rb
34
35
  - lib/market_data/version.rb
35
36
  - market_data.gemspec
36
37
  - sig/market_data.rbs