market_data 0.1.2 → 0.2.1

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: 02bae11e4d0ffd6cbc6c1e9211188bf980c54f59c61cb36d9d32628975ace153
4
- data.tar.gz: 661a761e828644036057ec05d6711ddeb1462e9885de3baf8319b4e483ea8333
3
+ metadata.gz: b63212c9e8ada21f198021bc3537392d9a6a34933682ebe789923883c2f38131
4
+ data.tar.gz: 9afa24ef746a81f115628f65aff4bccfdbfbf3086887444d62409545be76b7fb
5
5
  SHA512:
6
- metadata.gz: 48056be17c0be8068325457f25e64d4a38e38d9b3573cf707960348d5ebc53597ea6beb96edf5a7f96da593d8480539cae01d4852a2b1ee658cb6f29665562ab
7
- data.tar.gz: e7277b27c830b40a5e3de5fbbd83d53b69ab02e463b42e00ba2a72eb82265b1e972a42656a264e448e18387442cd81989f0cb047cdf093f11322281722bacd27
6
+ metadata.gz: e21e172593fcee9340de7ca5ac37373db2d07479e2c912c01d7bf05399e012edcf405caf12c2cbf3ad0545b4ae8a166d6c98f40f67367e60040f818356866666
7
+ data.tar.gz: a3fcd6d6368c073621e9f400be5412b82e88ada5cce2b00a26b4e6697b97b6beb568564f64032c6c8eb777db93fe7afa3f324c616465e2913b838fd02ffa3eed
data/CHANGELOG.md CHANGED
@@ -12,4 +12,16 @@
12
12
  ## [0.1.2] - 2024-09-24
13
13
 
14
14
  - Change validations for bulk_quotes method
15
- - Add `token` method to `Client` class
15
+ - Add `token` method to `Client` class
16
+
17
+ ## [0.2.0] - 2024-09-30
18
+
19
+ - Internal rework of code. New modules for models, mappers, constants and errors
20
+ - Introduced unit tests for almost all modules.
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
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
data/Gemfile CHANGED
@@ -7,4 +7,8 @@ gemspec
7
7
 
8
8
  gem "rake", "~> 13.0"
9
9
 
10
- gem "minitest", "~> 5.0"
10
+ group :test do
11
+ gem "minitest", "~> 5.0"
12
+ gem 'simplecov', require: false
13
+ gem 'mocha'
14
+ end
data/Gemfile.lock CHANGED
@@ -1,13 +1,23 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- market_data (0.1.0)
4
+ market_data (0.1.2)
5
5
 
6
6
  GEM
7
7
  remote: https://rubygems.org/
8
8
  specs:
9
+ docile (1.4.1)
9
10
  minitest (5.16.3)
11
+ mocha (2.4.5)
12
+ ruby2_keywords (>= 0.0.5)
10
13
  rake (13.0.6)
14
+ ruby2_keywords (0.0.5)
15
+ simplecov (0.22.0)
16
+ docile (~> 1.1)
17
+ simplecov-html (~> 0.11)
18
+ simplecov_json_formatter (~> 0.1)
19
+ simplecov-html (0.13.1)
20
+ simplecov_json_formatter (0.1.4)
11
21
 
12
22
  PLATFORMS
13
23
  arm64-darwin-23
@@ -15,7 +25,9 @@ PLATFORMS
15
25
  DEPENDENCIES
16
26
  market_data!
17
27
  minitest (~> 5.0)
28
+ mocha
18
29
  rake (~> 13.0)
30
+ simplecov
19
31
 
20
32
  BUNDLED WITH
21
33
  2.4.1
data/README.md CHANGED
@@ -1,34 +1,144 @@
1
1
  # MarketData
2
2
 
3
- TODO: Delete this and the text below, and describe your gem
3
+ A Ruby wrapper for the [MarketData API](https://www.marketdata.app/docs/api).
4
4
 
5
- Welcome to your new gem! In this directory, you'll find the files you need to be able to package up your Ruby library into a gem. Put your Ruby code in the file `lib/market_data`. To experiment with that code, run `bin/console` for an interactive prompt.
5
+ ![coverage](https://img.shields.io/badge/coverage%3A-84.71%25-yellow.svg)
6
6
 
7
7
  ## Installation
8
8
 
9
- TODO: Replace `UPDATE_WITH_YOUR_GEM_NAME_PRIOR_TO_RELEASE_TO_RUBYGEMS_ORG` with your gem name right after releasing it to RubyGems.org. Please do not do it earlier due to security reasons. Alternatively, replace this section with instructions to install your gem from git if you don't plan to release to RubyGems.org.
9
+ $ gem install market_data
10
+ $ bundle install
10
11
 
11
- Install the gem and add to the application's Gemfile by executing:
12
+ ## Usage
12
13
 
13
- $ bundle add UPDATE_WITH_YOUR_GEM_NAME_PRIOR_TO_RELEASE_TO_RUBYGEMS_ORG
14
+ You must instantiate a `MarketData::Client` before running any method. You'll need your MarketData API token first.
14
15
 
15
- If bundler is not being used to manage dependencies, install the gem by executing:
16
+ $ client = MarketData::Client.new "YOUR_API_TOKEN"
17
+
18
+ $ client.quote("AAPL")
19
+ $ => <struct MarketData::Models::Quote symbol="AAPL", ask=231.42, askSize=2, bid=231.4, .....
16
20
 
17
- $ gem install UPDATE_WITH_YOUR_GEM_NAME_PRIOR_TO_RELEASE_TO_RUBYGEMS_ORG
21
+ ### Quotes
22
+ For getting a single quote, run the `quote` method just as the example above.
23
+
24
+ #### Optional parameters
25
+
26
+ * `w52`
27
+
28
+ For getting fields with 52-week low and high information, pass true as the `w52` parameter.
29
+
30
+ $ q = client.quote("AAPL", w52=true)
31
+ $ => q.high52 = 252.11
32
+ $ => q.low52 = 222.35
33
+ **It is `false` by default.**
34
+
35
+ * `extended`
36
+
37
+ For getting a quote when market is on extended hours, you have to supply the `extended` parameter as true.
38
+
39
+ **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.
40
+
41
+ ### Bulk quotes
42
+
43
+ For getting multiple quotes in a single request, use the `bulk_candle` method.
44
+
45
+ $ quotes = client.bulk_quotes(["AAPL", "AMD", "NOTAQUOTE])
46
+
47
+ $ quotes["AMD"] => <struct MarketData::Models::Quote symbol="AMD", ask=150.42, askSize=2, bid=146.4, .....
48
+
49
+ $ quotes["NOTAQUOTE"] = nil
50
+
51
+ If a quote is not found, the hashmap will return a nil value for that ticker's key.
52
+
53
+ #### Optional parameters
54
+
55
+ * `snapshot`
56
+
57
+ If snapshot is true, any supplied array of symbols will be ignored and a complete snapshot of the market ticker's will be returned.
58
+
59
+ $ quotes = client.bulk_quotes([], snapshot = true)
60
+
61
+ $ quotes["A"] => <struct MarketData::Models::Quote symbol="A", ask=56.32, askSize=45, bid=67, .....
62
+ ....
63
+ $ quotes["Z"] => <struct MarketData::Models::Quote symbol="Z", ask=25, askSize=3, bid=14.5, .....
64
+
65
+ **This could use all you API credits. Use with caution.**
66
+
67
+ * `extended`
68
+
69
+ For getting a quote when market is on extended hours, you have to supply the `extended` parameter as true.
70
+
71
+ **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.
72
+
73
+ ### Candles
74
+ For getting ticker candles, you'll need to specify:
75
+ * a ticker symbol
76
+ * a resolution (`M`, `D`, `W`, etc. See [docs](https://www.marketdata.app/docs/api/stocks/candles#request-parameters) for a complete list)
77
+ * a strategy to specfy a date range. You can use `from` and `to` OR `to` and `countback`.
78
+
79
+
80
+ As an example, for getting candles for last week, for the first strategy:
81
+
82
+ $ quotes = client.candles("AAPL", "D", (Time.current - 1.week).iso8601, Time.current.iso8601, nil)
83
+
84
+ and for the second
85
+
86
+ $ quotes = client.candles("AAPL", "D", nil, Time.current.iso8601, 7)
87
+
88
+ `to` and `from` can receive an ISO 8601 compliant utc format or a unix timestamp.
89
+
90
+ ### Bulk candles
91
+
92
+ 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.)
93
+
94
+ It returns a hashmap with the ticker symbol as a key.
95
+
96
+ $ candles = cl.bulk_candles(["AAPL", "AMD", "NOTAQUOTE"])
97
+
98
+ $ candles["AMD"]
99
+ $ => #<struct MarketData::Models::Candle symbol="AMD", open=174.05, high=174.05, low=169.55, close=171.02, volume=33391035, time=1728446400>
100
+ $ candles["AAPL"]
101
+ $ => #<struct MarketData::Models::Candle symbol="AAPL", open=225.23, high=229.75, low=224.83, close=229.54, volume=31398884, time=1728446400>
102
+ $ candles["NOTAQUOTE"] => nil
103
+
104
+ If a quote is not found, the hashmap will return a nil value for that ticker's key.
105
+
106
+ ## ROADMAP
107
+
108
+ The following is an ordered list of next expected developments, based on the endpoints present in the [docs](https://www.marketdata.app/docs/api)
109
+
110
+ From Stocks endpoints:
111
+ - [X] Stocks
112
+ - [X] Bulk Stocks
113
+ - [X] Candles
114
+ - [X] Bulk Candles
115
+ - [X] Support for new optional parameters for Quotes and Bulk Quotes
116
+ - [ ] Earnings
117
+
118
+ From Markets endpoints:
119
+ - [ ] Status
120
+
121
+ From Indices endpoints:
122
+ - [ ] Quotes
123
+ - [ ] Candles
124
+
125
+ From Stocks endpoints:
126
+ - [ ] Support for optional parameters for Candles
127
+ - [ ] Support for optional parameters for Bulk Candles
18
128
 
19
- ## Usage
20
129
 
21
- TODO: Write usage instructions here
130
+ ## Tests
22
131
 
23
- ## Development
132
+ Run tests with
24
133
 
25
- After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake test` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
134
+ $ rake
26
135
 
27
- To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
136
+ To run tests and check coverage
28
137
 
138
+ $ rake && open coverage/index.html
29
139
  ## Contributing
30
140
 
31
- Bug reports and pull requests are welcome on GitHub at https://github.com/[USERNAME]/market_data. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [code of conduct](https://github.com/[USERNAME]/market_data/blob/main/CODE_OF_CONDUCT.md).
141
+ Leave an issue or contact me at sebagonz91@gmail.com
32
142
 
33
143
  ## License
34
144
 
@@ -0,0 +1,13 @@
1
+ module MarketData
2
+ module Constants
3
+ SECOND = 1
4
+ MINUTE = 60
5
+ HOUR = MINUTE * 60
6
+ DAY = HOUR * 24
7
+ WEEK = DAY * 7
8
+ YEAR = DAY * 365
9
+
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
+ end
13
+ end
@@ -1,16 +1,30 @@
1
+ require 'json'
2
+
1
3
  module MarketData
2
4
 
3
5
  class ClientError < StandardError;end
4
6
  class UnauthorizedError < ClientError; end
5
7
  class RateLimitedError < ClientError; end
6
8
  class BadParameterError < ClientError; end
9
+ class NotFoundError < ClientError; end
7
10
 
8
11
  module Errors
9
12
  def handle_error e
10
13
  er = e.io
14
+ parsed_info = JSON.parse(er.string)
11
15
  case er.status[0]
16
+ when "400"
17
+ if parsed_info["s"] == "error"
18
+ raise BadParameterError.new(parsed_info["errmsg"])
19
+ end
20
+ raise BadParameterError
21
+ when "404"
22
+ if parsed_info["s"] == "no_data"
23
+ raise NotFoundError.new("no candle information was found for the request")
24
+ end
25
+ raise NotFoundError
12
26
  when "401"
13
- raise UnauthorizedError
27
+ raise UnauthorizedError.new(parsed_info["errmsg"])
14
28
  when "429"
15
29
  raise RateLimitedError
16
30
  else
@@ -0,0 +1,67 @@
1
+ require 'market_data/models'
2
+
3
+ module MarketData
4
+ module Mappers
5
+ include MarketData::Models
6
+
7
+ SYMBOL_RESPONSE_KEY = "symbol"
8
+ STATUS_RESPONSE_KEY = "s"
9
+
10
+ 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
+ )
26
+ end
27
+
28
+ def map_bulk_quotes response
29
+ h = Hash.new
30
+ size = response[SYMBOL_RESPONSE_KEY].size
31
+ for i in 0..(size - 1) do
32
+ qquote = map_quote(response, i)
33
+ h[response[SYMBOL_RESPONSE_KEY][i]] = !qquote.blank? ? qquote : nil
34
+ end
35
+ h
36
+ end
37
+
38
+ def map_candles response, symbol
39
+ 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
+ )
52
+ end
53
+ ar
54
+ end
55
+
56
+ def map_bulk_candles response
57
+ 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])
62
+ h[response[SYMBOL_RESPONSE_KEY][i]] = !candle.blank? ? candle : nil
63
+ end
64
+ h
65
+ end
66
+ end
67
+ end
@@ -0,0 +1,19 @@
1
+ require 'market_data/constants'
2
+
3
+ module MarketData
4
+ module Models
5
+ include MarketData::Constants
6
+
7
+ Quote = Struct.new(*Constants::QUOTE_FIELDS) do
8
+ def blank?
9
+ (QUOTE_FIELDS - [:symbol]).all? { |mmethod| self[mmethod].nil?}
10
+ end
11
+ end
12
+
13
+ Candle = Struct.new(*Constants::CANDLE_FIELDS) do
14
+ def blank?
15
+ (CANDLE_FIELDS - [:symbol]).all? { |mmethod| self[mmethod].nil?}
16
+ end
17
+ end
18
+ end
19
+ end
@@ -1,61 +1,98 @@
1
1
  require 'market_data/conn'
2
2
  require 'market_data/errors'
3
+ require 'market_data/mappers'
3
4
 
4
5
  module MarketData
5
6
  module Quotes
7
+ include MarketData::Mappers
6
8
  include MarketData::Errors
7
9
  include MarketData::Conn
8
10
 
9
- SYMBOL_RESPONSE_KEY = "symbol"
10
- STATUS_RESPONSE_KEY = "s"
11
-
12
- QUOTE_FIELDS = [:symbol, :ask, :askSize, :bid, :bidSize, :mid, :last, :change, :changepct, :volume, :updated, :high52, :low52]
13
11
  @@single = "/v1/stocks/quotes/"
14
12
  @@bulk = "/v1/stocks/bulkquotes/"
13
+ @@candles = "/v1/stocks/candles/"
14
+ @@bulk_candles = "/v1/stocks/bulkcandles/"
15
15
 
16
- # Quote is the struct to hold ticker request information
17
- Quote = Struct.new(*QUOTE_FIELDS) do
18
- def blank?
19
- (QUOTE_FIELDS - [:symbol]).all? { |mmethod| self[mmethod].nil?}
20
- end
21
- end
22
-
23
- def quote(symbol, w52 = false)
16
+ def quote(symbol, w52 = false, extended = false)
24
17
  path_hash = { host: MarketData.base_host, path: @@single + symbol }
25
- if w52
26
- path_hash[:query] = URI.encode_www_form({"52week" => true })
27
- end
28
- res = do_connect(get_uri path_hash)
29
- Quote.new(*res.except(STATUS_RESPONSE_KEY).values.map { |ar| ar[0] })
18
+ query_hash = {}
19
+ if w52
20
+ query_hash["52week"] = true
21
+ end
22
+ # MarketData API considers extended as true by default
23
+ if !extended
24
+ query_hash[:extended] = false
25
+ end
26
+ if !query_hash.empty?
27
+ path_hash[:query] = URI.encode_www_form(query_hash)
28
+ end
29
+ res = do_connect(get_uri path_hash)
30
+ map_quote(res)
30
31
  end
31
32
 
32
- def bulk_quotes(symbols, snapshot = false)
33
+ def bulk_quotes(symbols, snapshot = false, extended = false)
33
34
  path_hash = { host: MarketData.base_host, path: @@bulk }
34
35
  query_hash = {}
35
36
 
37
+ # MarketData API considers extended as true by default
38
+ if !extended
39
+ query_hash[:extended] = false
40
+ end
41
+
36
42
  if snapshot
37
43
  query_hash[:snapshot] = true
38
44
  else
39
- if not symbols.is_a?(Array) || symbols.size < 1
45
+ if !symbols.is_a?(Array) || symbols.size < 1
40
46
  raise BadParameterError.new("symbols must be a non-empty list")
41
47
  end
42
- query_hash = { symbols: symbols.join(",") }
48
+ query_hash[:symbols] = symbols.join(",")
43
49
  end
44
-
45
50
  path_hash[:query] = URI.encode_www_form(query_hash)
46
51
 
47
52
  res = do_connect(get_uri path_hash)
48
- Quotes.map_quotes(res)
53
+ map_bulk_quotes res
54
+ end
55
+
56
+ def candles(symbol, opts = {})
57
+ defaults = {resolution: "D", from: nil, to: Time.now.utc.to_i, countback: nil}
58
+ opts = defaults.merge(opts)
59
+
60
+ query_hash = {to: opts[:to]}
61
+
62
+ # TODO Move method validations into own class
63
+ # TODO check to is either iso8601 or unix
64
+ if opts[:from].nil? && opts[:countback].nil?
65
+ raise BadParameterError.new("either :from or :countback must be supplied")
66
+ end
67
+
68
+ if opts[:from].nil?
69
+ query_hash[:countback] = opts[:countback]
70
+ else
71
+ query_hash[:from] = opts[:from]
72
+ end
73
+
74
+ path_hash = { host: MarketData.base_host, path: @@candles + opts[:resolution] + "/" + symbol }
75
+ path_hash[:query] = URI.encode_www_form(query_hash)
76
+
77
+ res = do_connect(get_uri path_hash)
78
+ map_candles res, symbol
49
79
  end
50
80
 
51
- def self.map_quotes(quotes)
52
- h = Hash.new
53
- size = quotes[SYMBOL_RESPONSE_KEY].size
54
- for i in 0..(size - 1) do
55
- qquote = Quote.new(*quotes.except(STATUS_RESPONSE_KEY).values.map { |ar| ar[i] })
56
- h[quotes[SYMBOL_RESPONSE_KEY][i]] = !qquote.blank? ? qquote : nil
81
+ def bulk_candles(symbols, resolution = "D")
82
+ unless resolution == "daily" || resolution == "1D" || resolution == "D"
83
+ raise BadParameterError.new("only daily resolution is allowed for this endpoint")
84
+ end
85
+ path_hash = { host: MarketData.base_host, path: @@bulk_candles + resolution + "/" }
86
+
87
+ if !symbols.is_a?(Array) || symbols.size < 1
88
+ raise BadParameterError.new("symbols must be a non-empty list")
57
89
  end
58
- h
90
+ query_hash = { symbols: symbols.join(",") }
91
+
92
+ path_hash[:query] = URI.encode_www_form(query_hash)
93
+
94
+ res = do_connect(get_uri path_hash)
95
+ map_bulk_candles res
59
96
  end
60
97
  end
61
98
  end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module MarketData
4
- VERSION = "0.1.2"
4
+ VERSION = "0.2.1"
5
5
  end
data/lib/market_data.rb CHANGED
@@ -13,10 +13,6 @@ module MarketData
13
13
  def initialize token
14
14
  @access_token = token
15
15
  end
16
-
17
- def token
18
- @access_token
19
- end
20
16
  end
21
17
 
22
18
  def self.base_host
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.1.2
4
+ version: 0.2.1
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-24 00:00:00.000000000 Z
11
+ date: 2024-10-09 00:00:00.000000000 Z
12
12
  dependencies: []
13
13
  description: A Ruby client for the MarketData API.
14
14
  email:
@@ -26,7 +26,10 @@ files:
26
26
  - Rakefile
27
27
  - lib/market_data.rb
28
28
  - lib/market_data/conn.rb
29
+ - lib/market_data/constants.rb
29
30
  - lib/market_data/errors.rb
31
+ - lib/market_data/mappers.rb
32
+ - lib/market_data/models.rb
30
33
  - lib/market_data/quotes.rb
31
34
  - lib/market_data/version.rb
32
35
  - market_data.gemspec