# encoding: UTF-8 # # Copyright (c) 2010-2015 GoodData Corporation. All rights reserved. # This source code is licensed under the BSD-style license found in the # LICENSE file in the root directory of this source tree. require 'terminal-table' require 'securerandom' require 'monitor' require 'thread_safe' require 'rest-client' require_relative '../version' require_relative '../exceptions/exceptions' module RestClient module AbstractResponse alias_method :old_follow_redirection, :follow_redirection def follow_redirection(request = nil, result = nil, &block) fail 'Using monkey patched version of RestClient::AbstractResponse#follow_redirection which is guaranteed to be compatible only with RestClient 1.8.0' if RestClient::VERSION != '1.8.0' # @args[:cookies] = request.cookies # old_follow_redirection(request, result, &block) new_args = @args.dup url = headers[:location] url = URI.parse(request.url).merge(url).to_s if url !~ /^http/ new_args[:url] = url if request fail MaxRedirectsReached if request.max_redirects == 0 new_args[:password] = request.password new_args[:user] = request.user new_args[:headers] = request.headers new_args[:max_redirects] = request.max_redirects - 1 # TODO: figure out what to do with original :cookie, :cookies values new_args[:cookies] = get_redirection_cookies(request, result, new_args) end Request.execute(new_args, &block) end # Returns cookies which should be passed when following redirect # # @param request [RestClient::Request] Original request # @param result [Net::HTTPResponse] Response # @param args [Hash] Original arguments # @return [Hash] Cookies to be passsed when following redirect def get_redirection_cookies(request, _result, _args) request.cookies end end end module GoodData module Rest # Wrapper of low-level HTTP/REST client/library class Connection include MonitorMixin DEFAULT_URL = 'https://secure.gooddata.com' LOGIN_PATH = '/gdc/account/login' TOKEN_PATH = '/gdc/account/token' KEYS_TO_SCRUB = [:password, :verifyPassword, :authorizationToken] ID_LENGTH = 16 DEFAULT_HEADERS = { :content_type => :json, :accept => [:json, :zip], :user_agent => GoodData.gem_version_string } DEFAULT_WEBDAV_HEADERS = { :user_agent => GoodData.gem_version_string } DEFAULT_LOGIN_PAYLOAD = { :headers => DEFAULT_HEADERS, :verify_ssl => true } RETRYABLE_ERRORS = [ RestClient::InternalServerError, RestClient::RequestTimeout, RestClient::MethodNotAllowed, SystemCallError, Timeout::Error ] RETRIES_ON_TOO_MANY_REQUESTS_ERROR = 12 RETRY_TIME_INITIAL_VALUE = 1 RETRY_TIME_COEFFICIENT = 1.5 RETRYABLE_ERRORS << Net::ReadTimeout if Net.const_defined?(:ReadTimeout) class << self def construct_login_payload(username, password) res = { 'postUserLogin' => { 'login' => username, 'password' => password, 'remember' => 1 } } res end # Generate random string with URL safe base64 encoding # # @param [String] length Length of random string to be generated # # @return [String] Generated random string def generate_string(length = ID_LENGTH) SecureRandom.urlsafe_base64(length) end # Retry block if exception thrown def retryable(options = {}, &_block) opts = { :tries => 1, :on => RETRYABLE_ERRORS }.merge(options) retry_exception = opts[:on] retries = opts[:tries] too_many_requests_tries = RETRIES_ON_TOO_MANY_REQUESTS_ERROR unless retry_exception.is_a?(Array) retry_exception = [retry_exception] end retry_time = RETRY_TIME_INITIAL_VALUE begin return yield rescue RestClient::Unauthorized, RestClient::Forbidden => e # , RestClient::Unauthorized => e raise e unless options[:refresh_token] raise e if options[:dont_reauth] options[:refresh_token].call # (dont_reauth: true) retry if (retries -= 1) > 0 rescue RestClient::TooManyRequests, RestClient::ServiceUnavailable GoodData.logger.warn "Too many requests, retrying in #{retry_time} seconds" sleep retry_time retry_time *= RETRY_TIME_COEFFICIENT # 10 requests with 1.5 coefficent should take ~ 3 mins to finish retry if (too_many_requests_tries -= 1) > 1 rescue *retry_exception => e GoodData.logger.warn e.inspect retry if (retries -= 1) > 1 end yield end end attr_reader :request_params # backward compatibility alias_method :cookies, :request_params attr_reader :server attr_reader :stats attr_reader :user def initialize(opts) super() @stats = ThreadSafe::Hash.new headers = opts[:headers] || {} @webdav_headers = DEFAULT_WEBDAV_HEADERS.merge(headers) @user = nil @server = nil @opts = opts # Initialize cookies reset_cookies! @at_exit_handler_installed = nil end # Connect using username and password def connect(username, password, options = {}) server = options[:server] || Helpers::AuthHelper.read_server options = DEFAULT_LOGIN_PAYLOAD.merge(options) headers = options[:headers] || {} options = options.merge(headers) @server = RestClient::Resource.new server, options # Install at_exit handler first unless @at_exit_handler_installed begin at_exit { disconnect if @user } rescue RestClient::Unauthorized GoodData.logger.info 'Already logged out' ensure @at_exit_handler_installed = true end end # Reset old cookies first if options[:sst_token] merge_cookies!('GDCAuthSST' => options[:sst_token]) get('/gdc/account/token', @request_params) @user = get(get('/gdc/app/account/bootstrap')['bootstrapResource']['accountSetting']['links']['self']) GoodData.logger.info("Connected using SST to server #{@server.url} to profile \"#{@user['accountSetting']['login']}\"") @auth = {} refresh_token :dont_reauth => true else GoodData.logger.info("Connected using username \"#{username}\" to server #{@server.url}") credentials = Connection.construct_login_payload(username, password) generate_session_id @auth = post(LOGIN_PATH, credentials, :dont_reauth => true)['userLogin'] refresh_token :dont_reauth => true @user = get(@auth['profile']) end GoodData.logger.info('Connection successful') rescue RestClient::Unauthorized => e GoodData.logger.info('Bad Login or Password') GoodData.logger.info('Connection failed') raise e rescue RestClient::Forbidden => e GoodData.logger.info('Connection failed') raise e end # Disconnect def disconnect # TODO: Wrap somehow url = @auth['state'] begin clear_session_id delete url if url rescue RestClient::Unauthorized GoodData.logger.info 'Already disconnected' end @auth = nil @server = nil @user = nil reset_cookies! end def download(what, where, options = {}) # handle the path (directory) given in what ilast_slash = what.rindex('/') if ilast_slash.nil? what_dir = '' else # take the directory from the path what_dir = what[0..ilast_slash - 1] # take the filename from the path what = what[ilast_slash + 1..-1] end option_dir = options[:directory] || '' option_dir = option_dir[0..-2] if option_dir[-1] == '/' # join the otion dir with the what_dir # [option dir empty, what dir empty] => the joined dir dir_hash = { [true, true] => '', [true, false] => what_dir, [false, true] => option_dir, [false, false] => "#{what_dir}/#{option_dir}" } dir = dir_hash[[option_dir.empty?, what_dir.empty?]] staging_uri = options[:staging_url].to_s base_url = dir.empty? ? staging_uri : URI.join(staging_uri, "#{dir}/").to_s url = URI.join(base_url, CGI.escape(what)).to_s b = proc do raw = { :headers => { :user_agent => GoodData.gem_version_string }, :method => :get, :url => url, :verify_ssl => (@opts[:verify_ssl] == false || @opts[:verify_ssl] == OpenSSL::SSL::VERIFY_NONE) ? OpenSSL::SSL::VERIFY_NONE : OpenSSL::SSL::VERIFY_PEER }.merge(cookies) if where.is_a?(IO) || where.is_a?(StringIO) RestClient::Request.execute(raw) do |chunk, _x, response| if response.code.to_s != '200' fail ArgumentError, "Error downloading #{url}. Got response: #{response.code} #{response} #{response.body}" end where.write chunk end else # Assume it is a string or file File.open(where, 'w') do |f| RestClient::Request.execute(raw) do |chunk, _x, response| if response.code.to_s != '200' fail ArgumentError, "Error downloading #{url}. Got response: #{response.code} #{response} #{response.body}" end f.write chunk end end end end res = nil GoodData::Rest::Connection.retryable(:tries => 2, :refresh_token => proc { refresh_token }) do res = b.call end res end def refresh_token(_options = {}) begin # rubocop:disable RedundantBegin get TOKEN_PATH, :dont_reauth => true # avoid infinite loop GET fails with 401 rescue Exception => e # rubocop:disable RescueException puts e.message raise e end end # Returns server URI # # @return [String] server uri def server_url @server && @server.url end # HTTP DELETE # # @param uri [String] Target URI def delete(uri, options = {}) options = log_info(options) GoodData.logger.debug "DELETE: #{@server.url}#{uri}" profile "DELETE #{uri}" do b = proc do params = fresh_request_params(options[:request_id]) begin @server[uri].delete(params) rescue RestClient::Exception => e # log the error if it happens log_error(e, uri, params) raise e end end process_response(options, &b) end end # Helper for logging error # # @param e [RuntimeException] Exception to log # @param uri [String] Uri on which the request failed # @param uri [Hash] Additional params def log_error(e, uri, params) return if e.response && e.response.code == 401 && !uri.include?('token') && !uri.include?('login') GoodData.logger.error(format_error(e, params)) end # HTTP GET # # @param uri [String] Target URI def get(uri, options = {}, &user_block) options = log_info(options) GoodData.logger.debug "GET: #{@server.url}#{uri}, #{options}" profile "GET #{uri}" do b = proc do params = fresh_request_params(options[:request_id]).merge(options) begin @server[uri].get(params, &user_block) rescue RestClient::Exception => e # log the error if it happens log_error(e, uri, params) raise e end end process_response(options, &b) end end # HTTP PUT # # @param uri [String] Target URI def put(uri, data, options = {}) options = log_info(options) payload = data.is_a?(Hash) ? data.to_json : data GoodData.logger.debug "PUT: #{@server.url}#{uri}, #{scrub_params(data, KEYS_TO_SCRUB)}" profile "PUT #{uri}" do b = proc do params = fresh_request_params(options[:request_id]) begin @server[uri].put(payload, params) rescue RestClient::Exception => e # log the error if it happens log_error(e, uri, params) raise e end end process_response(options, &b) end end # HTTP POST # # @param uri [String] Target URI def post(uri, data = nil, options = {}) options = log_info(options) GoodData.logger.debug "POST: #{@server.url}#{uri}, #{scrub_params(data, KEYS_TO_SCRUB)}" profile "POST #{uri}" do payload = data.is_a?(Hash) ? data.to_json : data b = proc do params = fresh_request_params(options[:request_id]) begin @server[uri].post(payload, params) rescue RestClient::Exception => e # log the error if it happens log_error(e, uri, params) raise e end end process_response(options, &b) end end # Reader method for SST token # # @return uri [String] SST token def sst_token request_params[:cookies]['GDCAuthSST'] end def stats_table(values = stats) sorted = values.sort_by { |_k, v| v[:avg] } Terminal::Table.new :headings => %w(title avg min max total calls) do |t| overall = { :avg => 0, :calls => 0, :total => 0 } sorted.each do |l| avg = l[1][:avg] min = l[1][:min] max = l[1][:max] total = l[1][:total] calls = l[1][:calls] row = [ l[0], sprintf('%.3f', avg), sprintf('%.3f', min), sprintf('%.3f', max), sprintf('%.3f', total), calls ] overall[:min] = min if overall[:min].nil? || min < overall[:min] overall[:max] = max if overall[:max].nil? || max > overall[:max] overall[:total] += total overall[:calls] += calls overall[:avg] += avg t.add_row row end overall[:avg] = overall[:avg] / sorted.length row = [ 'TOTAL', sprintf('%.3f', overall[:avg]), sprintf('%.3f', overall[:min]), sprintf('%.3f', overall[:max]), sprintf('%.3f', overall[:total]), overall[:calls] ] t.add_row :separator t.add_row row end end # Reader method for TT token # # @return uri [String] TT token def tt_token request_params[:cookies]['GDCAuthTT'] end # Uploads a file to GoodData server def upload(file, options = {}) dir = options[:directory] || '' staging_uri = options[:staging_url].to_s url = dir.empty? ? staging_uri : URI.join(staging_uri, "#{dir}/").to_s # Make a directory, if needed create_webdav_dir_if_needed url unless dir.empty? webdav_filename = options[:filename] || File.basename(file) do_stream_file URI.join(url, CGI.escape(webdav_filename)), file end def generate_request_id "#{session_id}:#{call_id}" end private def create_webdav_dir_if_needed(url) return if webdav_dir_exists?(url) method = :mkcol GoodData.logger.debug "#{method}: #{url}" b = proc do raw = { :method => method, :url => url, :headers => @webdav_headers, :verify_ssl => (@opts[:verify_ssl] == false || @opts[:verify_ssl] == OpenSSL::SSL::VERIFY_NONE) ? OpenSSL::SSL::VERIFY_NONE : OpenSSL::SSL::VERIFY_PEER }.merge(cookies) RestClient::Request.execute(raw) end GoodData::Rest::Connection.retryable(:tries => 2, :refresh_token => proc { refresh_token }) do b.call end end def do_stream_file(uri, filename, _options = {}) GoodData.logger.info "Uploading file user storage #{uri}" to_upload = File.new(filename) cookies_str = request_params[:cookies].map { |cookie| "#{cookie[0]}=#{cookie[1]}" }.join(';') req = Net::HTTP::Put.new(uri.path, 'User-Agent' => GoodData.gem_version_string, 'Cookie' => cookies_str) req.content_length = to_upload.size req.body_stream = to_upload http = Net::HTTP.new(uri.host, uri.port) http.use_ssl = true http.verify_mode = (@opts[:verify_ssl] == false || @opts[:verify_ssl] == OpenSSL::SSL::VERIFY_NONE) ? OpenSSL::SSL::VERIFY_NONE : OpenSSL::SSL::VERIFY_PEER response = nil GoodData::Rest::Connection.retryable(:tries => 2, :refresh_token => proc { refresh_token }) do response = http.start { |client| client.request(req) } end response end def format_error(e, params) "#{params[:x_gdc_request]} #{e.inspect}" end # generate session id to be passed as the first part to # x_gdc_request header def session_id @session_id ||= Connection.generate_string end def call_id Connection.generate_string end def generate_session_id @session_id = Connection.generate_string end def clear_session_id @session_id = nil end # log info_message given in options and make sure request_id is there def log_info(options) # if info_message given, log it with request_id (given or generated) if options[:info_message] request_id = options[:request_id] || generate_request_id GoodData.logger.info "#{options[:info_message]} Request id: #{request_id}" end options end # request heders with freshly generated request id def fresh_request_params(request_id = nil) @request_params.merge(:x_gdc_request => request_id || generate_request_id) end def merge_cookies!(cookies) @request_params[:cookies].merge! cookies end def process_response(options = {}, &block) retries = options[:tries] || 3 opts = { tries: retries, refresh_token: proc { refresh_token unless options[:dont_reauth] } }.merge(options) response = GoodData::Rest::Connection.retryable(opts) do block.call end merge_cookies! response.cookies content_type = response.headers[:content_type] return response if options[:process] == false if content_type == 'application/json' || content_type == 'application/json;charset=UTF-8' result = response.to_str == '""' ? {} : MultiJson.load(response.to_str) GoodData.logger.debug "Request ID: #{response.headers[:x_gdc_request]} - Response: #{result.inspect}" elsif ['text/plain;charset=UTF-8', 'text/plain; charset=UTF-8', 'text/plain'].include?(content_type) result = response GoodData.logger.debug 'Response: plain text' elsif content_type == 'application/zip' result = response GoodData.logger.debug 'Response: a zipped stream' elsif response.headers[:content_length].to_s == '0' result = nil GoodData.logger.debug 'Response: Empty response possibly 204' elsif response.code == 204 result = nil GoodData.logger.debug 'Response: 204 no content' else fail "Unsupported response content type '%s':\n%s" % [content_type, response.to_str[0..127]] end result rescue RestClient::Exception => e GoodData.logger.debug "Response: #{e.response}" raise $ERROR_INFO end def profile(title, &block) t1 = Time.now res = block.call t2 = Time.now delta = t2 - t1 update_stats title, delta res end def reset_cookies! @request_params = { :cookies => {} } end def scrub_params(params, keys) keys = keys.reduce([]) { |a, e| a.concat([e.to_s, e.to_sym]) } new_params = GoodData::Helpers.deep_dup(params) GoodData::Helpers.hash_dfs(new_params) do |k, _key| keys.each do |key_to_scrub| k[key_to_scrub] = ('*' * k[key_to_scrub].length) if k && k.key?(key_to_scrub) && k[key_to_scrub] end end new_params end # TODO: Store PH_MAP for wildcarding of URLs in reports in separate file PH_MAP = [ ['/gdc/account/profile/{id}', %r{/gdc/account/profile/[\w]+}], ['/gdc/account/login/{id}', %r{/gdc/account/login/[\w]+}], ['/gdc/account/domains/{id}/users?login={login}', %r{/gdc/account/domains/[\w\d-]+/users\?login=[^&$]+}], ['/gdc/account/domains/{id}', %r{/gdc/account/domains/[\w\d-]+}], ['/gdc/app/projects/{id}/execute', %r{/gdc/app/projects/[\w]+/execute}], ['/gdc/datawarehouse/instances/{id}', %r{/gdc/datawarehouse/instances/[\w]+}], ['/gdc/datawarehouse/executions/{id}', %r{/gdc/datawarehouse/executions/[\w]+}], ['/gdc/domains/{id}/segments/{segment}/synchronizeClients/results/{result}/details?offset={offset}&limit={limit}', %r{/gdc/domains/[\w]+/segments/[\w-]+/synchronizeClients/results/[\w]+/details/\?offset=[\d]+&limit=[\d]+}], ['/gdc/domains/{id}/segments/{segment}/synchronizeClients/results/{result}', %r{/gdc/domains/[\w]+/segments/[\w-]+/synchronizeClients/results/[\w]+}], ['/gdc/domains/{id}/segments/{segment}/', %r{/gdc/domains/[\w]+/segments/[\w-]+/}], ['/gdc/domains/{id}/segments/{segment}', %r{/gdc/domains/[\w]+/segments/[\w-]+}], ['/gdc/domains/{id}/clients?segment={segment}', %r{/gdc/domains/[\w]+/clients\?segment=[\w-]+}], ['/gdc/domains/{id}/', %r{/gdc/domains/[\w]+/}], ['/gdc/exporter/result/{id}/{id}', %r{/gdc/exporter/result/[\w]+/[\w]+}], ['/gdc/internal/projects/{id}/objects/setPermissions', %r{/gdc/internal/projects/[\w]+/objects/setPermissions}], ['/gdc/md/{id}/variables/item/{id}', %r{/gdc/md/[\w]+/variables/item/[\d]+}], ['/gdc/md/{id}/validate/task/{id}', %r{/gdc/md/[\w]+/validate/task/[\w]+}], ['/gdc/md/{id}/using2/{id}/{id}', %r{/gdc/md/[\w]+/using2/[\d]+/[\d]+}], ['/gdc/md/{id}/using2/{id}', %r{/gdc/md/[\w]+/using2/[\d]+}], ['/gdc/md/{id}/userfilters?users={users}', %r{/gdc/md/[\w]+/userfilters\?users=[/\w]+}], ['/gdc/md/{id}/userfilters?count={count}&offset={offset}', %r{/gdc/md/[\w]+/userfilters\?count=[\d]+&offset=[\d]+}], ['/gdc/md/{id}/usedby2/{id}/{id}', %r{/gdc/md/[\w]+/usedby2/[\d]+/[\d]+}], ['/gdc/md/{id}/usedby2/{id}', %r{/gdc/md/[\w]+/usedby2/[\d]+}], ['/gdc/md/{id}/tasks/{id}/status', %r{/gdc/md/[\w]+/tasks/[\w]+/status}], ['/gdc/md/{id}/obj/{id}/validElements', %r{/gdc/md/[\w]+/obj/[\d]+/validElements(/)?(\?.*)?}], ['/gdc/md/{id}/obj/{id}/elements', %r{/gdc/md/[\w]+/obj/[\d]+/elements(/)?(\?.*)?}], ['/gdc/md/{id}/obj/{id}', %r{/gdc/md/[\w]+/obj/[\d]+}], ['/gdc/md/{id}/etltask/{id}', %r{/gdc/md/[\w]+/etltask/[\w]+}], ['/gdc/md/{id}/dataResult/{id}', %r{/gdc/md/[\w]+/dataResult/[\d]+}], ['/gdc/md/{id}', %r{/gdc/md/[\w]+}], ['/gdc/projects/{id}/users/{id}/roles', %r{/gdc/projects/[\w]+/users/[\w]+/roles}], ['/gdc/projects/{id}/users/{id}/permissions', %r{/gdc/projects/[\w]+/users/[\w]+/permissions}], ['/gdc/projects/{id}/users', %r{/gdc/projects/[\w]+/users}], ['/gdc/projects/{id}/schedules/{id}/executions/{id}', %r{/gdc/projects/[\w]+/schedules/[\w]+/executions/[\w]+}], ['/gdc/projects/{id}/schedules/{id}', %r{/gdc/projects/[\w]+/schedules/[\w]+}], ['/gdc/projects/{id}/roles/{id}', %r{/gdc/projects/[\w]+/roles/[\d]+}], ['/gdc/projects/{id}/model/view/{id}', %r{/gdc/projects/[\w]+/model/view/[\w]+}], ['/gdc/projects/{id}/model/view', %r{/gdc/projects/[\w]+/model/view}], ['/gdc/projects/{id}/model/diff/{id}', %r{/gdc/projects/[\w]+/model/diff/[\w]+}], ['/gdc/projects/{id}/model/diff', %r{/gdc/projects/[\w]+/model/diff}], ['/gdc/projects/{id}/dataload/processes/{id}/executions/{id}', %r{/gdc/projects/[\w]+/dataload/processes/[\w-]+/executions/[\w-]+}], ['/gdc/projects/{id}/dataload/processes/{id}', %r{/gdc/projects/[\w]+/dataload/processes/[\w-]+}], ['/gdc/projects/{id}/', %r{/gdc/projects/[\w]+/}], ['/gdc/projects/{id}', %r{/gdc/projects/[\w]+}] ] def update_stats(title, delta) synchronize do orig_title = title placeholders = true if placeholders PH_MAP.each do |pm| break if title.gsub!(pm[1], pm[0]) end end stat = stats[title] if stat.nil? stat = { :min => delta, :max => delta, :total => 0, :avg => 0, :calls => 0, :entries => [] } end stat[:min] = delta if delta < stat[:min] stat[:max] = delta if delta > stat[:max] stat[:total] += delta stat[:calls] += 1 stat[:avg] = stat[:total] / stat[:calls] stat[:entries] << orig_title if placeholders stats[title] = stat end end def webdav_dir_exists?(url) method = :get GoodData.logger.debug "#{method}: #{url}" b = proc do raw = { :method => method, :url => url, :headers => @webdav_headers, :verify_ssl => (@opts[:verify_ssl] == false || @opts[:verify_ssl] == OpenSSL::SSL::VERIFY_NONE) ? OpenSSL::SSL::VERIFY_NONE : OpenSSL::SSL::VERIFY_PEER }.merge(cookies) begin RestClient::Request.execute(raw) rescue RestClient::Exception => e false if e.http_code == 404 end end res = nil GoodData::Rest::Connection.retryable(:tries => 2, :refresh_token => proc { refresh_token }) do res = b.call end res end end end end