X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/ebddfbe7fbca80ecf852bbb20af9c6b584394691..b805211887332f43b42c74f52f2b43686546003d:/services/api/test/test_helper.rb diff --git a/services/api/test/test_helper.rb b/services/api/test/test_helper.rb index 1bd1b51263..843d4f1b23 100644 --- a/services/api/test/test_helper.rb +++ b/services/api/test/test_helper.rb @@ -1,45 +1,216 @@ +# Copyright (C) The Arvados Authors. All rights reserved. +# +# SPDX-License-Identifier: AGPL-3.0 + +require 'update_permissions' + ENV["RAILS_ENV"] = "test" +unless ENV["NO_COVERAGE_TEST"] + begin + verbose_orig = $VERBOSE + begin + $VERBOSE = nil + require 'simplecov' + require 'simplecov-rcov' + ensure + $VERBOSE = verbose_orig + end + class SimpleCov::Formatter::MergedFormatter + def format(result) + SimpleCov::Formatter::HTMLFormatter.new.format(result) + SimpleCov::Formatter::RcovFormatter.new.format(result) + end + end + SimpleCov.formatter = SimpleCov::Formatter::MergedFormatter + SimpleCov.start do + add_filter '/test/' + add_filter 'initializers/secret_token' + end + rescue Exception => e + $stderr.puts "SimpleCov unavailable (#{e}). Proceeding without." + end +end + require File.expand_path('../../config/environment', __FILE__) require 'rails/test_help' +require 'mocha' +require 'mocha/minitest' module ArvadosTestSupport def json_response - ActiveSupport::JSON.decode @response.body + Oj.strict_load response.body end def api_token(api_client_auth_name) - api_client_authorizations(api_client_auth_name).api_token + api_client_authorizations(api_client_auth_name).token end def auth(api_client_auth_name) - {'HTTP_AUTHORIZATION' => "OAuth2 #{api_token(api_client_auth_name)}"} + {'HTTP_AUTHORIZATION' => "Bearer #{api_token(api_client_auth_name)}"} + end + + def show_errors model + return lambda { model.errors.full_messages.inspect } end end class ActiveSupport::TestCase - # Setup all fixtures in test/fixtures/*.(yml|csv) for all tests in alphabetical order. - # - # Note: You'll currently still have to declare fixtures explicitly in integration tests - # -- they do not yet inherit this setting + include FactoryBot::Syntax::Methods fixtures :all include ArvadosTestSupport + include CurrentApiClient - teardown do + setup do Thread.current[:api_client_ip_address] = nil Thread.current[:api_client_authorization] = nil Thread.current[:api_client_uuid] = nil Thread.current[:api_client] = nil + Thread.current[:token] = nil Thread.current[:user] = nil + restore_configuration + end + + teardown do + # Confirm that any changed configuration doesn't include non-symbol keys + $arvados_config.keys.each do |conf_name| + conf = Rails.configuration.send(conf_name) + confirm_keys_as_symbols(conf, conf_name) if conf.respond_to?('keys') + end + end + + def assert_equal(expect, *args) + if expect.nil? + assert_nil(*args) + else + super + end + end + + def assert_not_allowed + # Provide a block that calls a Rails boolean "true or false" success value, + # like model.save or model.destroy. This method will test that it either + # returns false, or raises a Permission Denied exception. + begin + refute(yield) + rescue ArvadosModel::PermissionDeniedError + end + end + + def add_permission_link from_who, to_what, perm_type + act_as_system_user do + Link.create!(tail_uuid: from_who.uuid, + head_uuid: to_what.uuid, + link_class: 'permission', + name: perm_type) + end + end + + def confirm_keys_as_symbols(conf, conf_name) + assert(conf.is_a?(ActiveSupport::OrderedOptions), "#{conf_name} should be an OrderedOptions object") + conf.keys.each do |k| + assert(k.is_a?(Symbol), "Key '#{k}' on section '#{conf_name}' should be a Symbol") + confirm_keys_as_symbols(conf[k], "#{conf_name}.#{k}") if conf[k].respond_to?('keys') + end + end + + def restore_configuration + # Restore configuration settings changed during tests + ConfigLoader.copy_into_config $arvados_config, Rails.configuration + ConfigLoader.copy_into_config $remaining_config, Rails.configuration + end + + def set_user_from_auth(auth_name) + client_auth = api_client_authorizations(auth_name) + client_auth.user.forget_cached_group_perms + Thread.current[:api_client_authorization] = client_auth + Thread.current[:api_client] = client_auth.api_client + Thread.current[:user] = client_auth.user + Thread.current[:token] = client_auth.token end def expect_json self.request.headers["Accept"] = "text/json" end - def authorize_with(api_client_auth_name) - ArvadosApiToken.new.call ({"rack.input" => "", "HTTP_AUTHORIZATION" => "OAuth2 #{api_client_authorizations(api_client_auth_name).api_token}"}) + def authorize_with api_client_auth_name + authorize_with_token api_client_authorizations(api_client_auth_name).token + end + + def authorize_with_token token + t = token + t = t.token if t.respond_to? :token + ArvadosApiToken.new.call("rack.input" => "", + "HTTP_AUTHORIZATION" => "Bearer #{t}") + end + + def salt_token(fixture:, remote:) + auth = api_client_authorizations(fixture) + uuid = auth.uuid + token = auth.api_token + hmac = OpenSSL::HMAC.hexdigest('sha1', token, remote) + return "v2/#{uuid}/#{hmac}" + end + + def self.skip_slow_tests? + !(ENV['RAILS_TEST_SHORT'] || '').empty? + end + + def self.skip(*args, &block) + end + + def self.slow_test(name, &block) + test(name, &block) unless skip_slow_tests? + end +end + +class ActionController::TestCase + setup do + @test_counter = 0 + self.request.headers['Accept'] = 'application/json' + self.request.headers['Content-Type'] = 'application/json' + end + + def check_counter action + @test_counter += 1 + if @test_counter == 2 + assert_equal 1, 2, "Multiple actions in functional test" + end + end + + [:get, :post, :put, :patch, :delete].each do |method| + define_method method do |action, *args| + check_counter action + # After Rails 5.0 upgrade, some params don't get properly serialized. + # One case are filters: [['attr', 'op', 'val']] become [['attr'], ['op'], ['val']] + # if not passed upstream as a JSON string. + if args[0].is_a?(Hash) && args[0][:params].is_a?(Hash) + args[0][:params].each do |key, _| + next if key == :exclude_script_versions # Job Reuse tests + # Keys could be: :filters, :where, etc + if [Array, Hash].include?(args[0][:params][key].class) + args[0][:params][key] = SafeJSON.dump(args[0][:params][key]) + end + end + end + super action, *args + end + end + + def self.suite + s = super + def s.run(*args) + @test_case.startup() + begin + super + ensure + @test_case.shutdown() + end + end + s end + def self.startup; end + def self.shutdown; end end class ActionDispatch::IntegrationTest @@ -48,9 +219,11 @@ class ActionDispatch::IntegrationTest Thread.current[:api_client_authorization] = nil Thread.current[:api_client_uuid] = nil Thread.current[:api_client] = nil + Thread.current[:token] = nil Thread.current[:user] = nil end end # Ensure permissions are computed from the test fixtures. -User.invalidate_permissions_cache +refresh_permissions +refresh_trashed