require_relative 'market.rb' require 'net/http' require 'openssl' require 'json' require 'bigdecimal' module Kaesen # Lakebtc Wrapper Class # https://www.lakebtc.com/s/api class Lakebtc < Market @@nonce = 0 def initialize(options = {}) super() @name = "Lakebtc" @api_key = ENV["LAKEBTC_KEY"] @api_secret = ENV["LAKEBTC_SECRET"] @url_public = "https://www.LakeBTC.com/api_v1" @url_private = @url_public options.each do |key, value| instance_variable_set("@#{key}", value) end yield(self) if block_given? end ############################################################# # API for public information ############################################################# # Get ticker information. # @return [hash] ticker # ask: [BigDecimal] 最良売気配値 # bid: [BigDecimal] 最良買気配値 # last: [BigDecimal] 最近値(?用語要チェック), last price # high: [BigDecimal] 高値 # low: [BigDecimal] 安値 # volume: [BigDecimal] 取引量 # ltimestamp: [int] ローカルタイムスタンプ # vwap: [BigDecimal] 過去24時間の加重平均 def ticker h = get_ssl(@url_public + "/ticker") # the id of BTCJPY is 5. h = h["JPY"] { "ask" => BigDecimal.new(h["ask"].to_s), "bid" => BigDecimal.new(h["bid"].to_s), "last" => BigDecimal.new(h["last"].to_s), # "high" # "low" # "volume" "ltimestamp" => Time.now.to_i, # "vwap" } end # Get order book. # @abstract # @return [hash] array of market depth # asks: [Array] 売りオーダー # price : [BigDecimal] # size : [BigDecimal] # bids: [Array] 買いオーダー # price : [BigDecimal] # size : [BigDecimal] # ltimestamp: [int] ローカルタイムスタンプ def depth h = get_ssl(@url_public + "/bcorderbook?symbol=btcjpy") # the id of BTCJPY is 5. { "asks" => h["asks"].map{|a,b| [BigDecimal.new(a.to_s), BigDecimal.new(b.to_s)]}, # to_s でないと誤差が生じる "bids" => h["bids"].map{|a,b| [BigDecimal.new(a.to_s), BigDecimal.new(b.to_s)]}, # to_s でないと誤差が生じる "ltimestamp" => Time.now.to_i, } end private def initialize_https(uri) https = Net::HTTP.new(uri.host, uri.port) https.use_ssl = true https.open_timeout = 5 https.read_timeout = 15 https.verify_mode = OpenSSL::SSL::VERIFY_PEER https.verify_depth = 5 https end # Connect to address via https, and return json response. def get_ssl(address) uri = URI.parse(address) begin https = initialize_https(uri) https.start {|w| response = w.get(uri.request_uri) case response when Net::HTTPSuccess json = JSON.parse(response.body) raise JSONException, response.body if json == nil return json else raise ConnectionFailedException, "Failed to connect to #{@name}." end } rescue raise end end def get_nonce pre_nonce = @@nonce next_nonce = (Time.now.to_i) * 100 if next_nonce <= pre_nonce @@nonce = pre_nonce + 1 else @@nonce = next_nonce end return @@nonce end def get_sign(req) secret = @api_secret text = req.body OpenSSL::HMAC::hexdigest(OpenSSL::Digest.new('sha512'), secret, text) end # Connect to address via https, and return json response. def post_ssl(address, data={}) uri = URI.parse(address) data["nonce"] = get_nonce begin req = Net::HTTP::Post.new(uri) req.set_form_data(data) req["Key"] = @api_key req["Sign"] = get_sign(req) https = initialize_https(uri) https.start {|w| response = w.request(req) case response when Net::HTTPSuccess json = JSON.parse(response.body) raise JSONException, response.body if json == nil return json else raise ConnectionFailedException, "Failed to connect to #{@name}: " + response.value end } rescue raise end end end end