X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/e102a3b3100577137def302ecd0f810bc46b808a..f6089c82da72f331ba5a44874ce267b18bcaf557:/services/api/test/integration/collections_api_test.rb diff --git a/services/api/test/integration/collections_api_test.rb b/services/api/test/integration/collections_api_test.rb index 5a3250ea0d..93ed1563f7 100644 --- a/services/api/test/integration/collections_api_test.rb +++ b/services/api/test/integration/collections_api_test.rb @@ -4,35 +4,273 @@ class CollectionsApiTest < ActionDispatch::IntegrationTest fixtures :all test "should get index" do - get "/arvados/v1/collections", {:format => :json}, {'HTTP_AUTHORIZATION' => "OAuth2 #{api_client_authorizations(:active).api_token}"} + get "/arvados/v1/collections", {:format => :json}, auth(:active) assert_response :success - assert_equal "arvados#collectionList", jresponse['kind'] + assert_equal "arvados#collectionList", json_response['kind'] end test "get index with filters= (empty string)" do - get "/arvados/v1/collections", {:format => :json, :filters => ''}, {'HTTP_AUTHORIZATION' => "OAuth2 #{api_client_authorizations(:active).api_token}"} + get "/arvados/v1/collections", {:format => :json, :filters => ''}, auth(:active) assert_response :success - assert_equal "arvados#collectionList", jresponse['kind'] + assert_equal "arvados#collectionList", json_response['kind'] + end + + test "get index with invalid filters (array of strings) responds 422" do + get "/arvados/v1/collections", { + :format => :json, + :filters => ['uuid', '=', 'ad02e37b6a7f45bbe2ead3c29a109b8a+54'].to_json + }, auth(:active) + assert_response 422 + assert_match /nvalid element.*not an array/, json_response['errors'].join(' ') + end + + test "get index with invalid filters (unsearchable column) responds 422" do + get "/arvados/v1/collections", { + :format => :json, + :filters => [['this_column_does_not_exist', '=', 'bogus']].to_json + }, auth(:active) + assert_response 422 + assert_match /nvalid attribute/, json_response['errors'].join(' ') + end + + test "get index with invalid filters (invalid operator) responds 422" do + get "/arvados/v1/collections", { + :format => :json, + :filters => [['uuid', ':-(', 'displeased']].to_json + }, auth(:active) + assert_response 422 + assert_match /nvalid operator/, json_response['errors'].join(' ') + end + + test "get index with invalid filters (invalid operand type) responds 422" do + get "/arvados/v1/collections", { + :format => :json, + :filters => [['uuid', '=', {foo: 'bar'}]].to_json + }, auth(:active) + assert_response 422 + assert_match /nvalid operand type/, json_response['errors'].join(' ') end test "get index with where= (empty string)" do - get "/arvados/v1/collections", {:format => :json, :where => ''}, {'HTTP_AUTHORIZATION' => "OAuth2 #{api_client_authorizations(:active).api_token}"} + get "/arvados/v1/collections", {:format => :json, :where => ''}, auth(:active) assert_response :success - assert_equal "arvados#collectionList", jresponse['kind'] + assert_equal "arvados#collectionList", json_response['kind'] end test "controller 404 response is json" do - get "/arvados/v1/thingsthatdonotexist", {:format => :xml}, {'HTTP_AUTHORIZATION' => "OAuth2 #{api_client_authorizations(:active).api_token}"} + get "/arvados/v1/thingsthatdonotexist", {:format => :xml}, auth(:active) assert_response 404 - assert_equal 1, jresponse['errors'].length - assert_equal true, jresponse['errors'][0].is_a?(String) + assert_equal 1, json_response['errors'].length + assert_equal true, json_response['errors'][0].is_a?(String) end test "object 404 response is json" do - get "/arvados/v1/groups/zzzzz-j7d0g-o5ba971173cup4f", {}, {'HTTP_AUTHORIZATION' => "OAuth2 #{api_client_authorizations(:active).api_token}"} + get "/arvados/v1/groups/zzzzz-j7d0g-o5ba971173cup4f", {}, auth(:active) assert_response 404 - assert_equal 1, jresponse['errors'].length - assert_equal true, jresponse['errors'][0].is_a?(String) + assert_equal 1, json_response['errors'].length + assert_equal true, json_response['errors'][0].is_a?(String) + end + + test "store collection as json" do + signing_opts = { + key: Rails.configuration.blob_signing_key, + api_token: api_token(:active), + } + signed_locator = Blob.sign_locator('bad42fa702ae3ea7d888fef11b46f450+44', + signing_opts) + post "/arvados/v1/collections", { + format: :json, + collection: "{\"manifest_text\":\". #{signed_locator} 0:44:md5sum.txt\\n\",\"portable_data_hash\":\"ad02e37b6a7f45bbe2ead3c29a109b8a+54\"}" + }, auth(:active) + assert_response 200 + assert_equal 'ad02e37b6a7f45bbe2ead3c29a109b8a+54', json_response['portable_data_hash'] + end + + test "store collection with manifest_text only" do + signing_opts = { + key: Rails.configuration.blob_signing_key, + api_token: api_token(:active), + } + signed_locator = Blob.sign_locator('bad42fa702ae3ea7d888fef11b46f450+44', + signing_opts) + post "/arvados/v1/collections", { + format: :json, + collection: "{\"manifest_text\":\". #{signed_locator} 0:44:md5sum.txt\\n\"}" + }, auth(:active) + assert_response 200 + assert_equal 'ad02e37b6a7f45bbe2ead3c29a109b8a+54', json_response['portable_data_hash'] + end + + test "store collection then update name" do + signing_opts = { + key: Rails.configuration.blob_signing_key, + api_token: api_token(:active), + } + signed_locator = Blob.sign_locator('bad42fa702ae3ea7d888fef11b46f450+44', + signing_opts) + post "/arvados/v1/collections", { + format: :json, + collection: "{\"manifest_text\":\". #{signed_locator} 0:44:md5sum.txt\\n\",\"portable_data_hash\":\"ad02e37b6a7f45bbe2ead3c29a109b8a+54\"}" + }, auth(:active) + assert_response 200 + assert_equal 'ad02e37b6a7f45bbe2ead3c29a109b8a+54', json_response['portable_data_hash'] + + put "/arvados/v1/collections/#{json_response['uuid']}", { + format: :json, + collection: { name: "a name" } + }, auth(:active) + + assert_response 200 + assert_equal 'ad02e37b6a7f45bbe2ead3c29a109b8a+54', json_response['portable_data_hash'] + assert_equal 'a name', json_response['name'] + + get "/arvados/v1/collections/#{json_response['uuid']}", { + format: :json, + }, auth(:active) + + assert_response 200 + assert_equal 'ad02e37b6a7f45bbe2ead3c29a109b8a+54', json_response['portable_data_hash'] + assert_equal 'a name', json_response['name'] + end + + test "update description for a collection, and search for that description" do + collection = collections(:multilevel_collection_1) + + # update collection's description + put "/arvados/v1/collections/#{collection['uuid']}", { + format: :json, + collection: { description: "something specific" } + }, auth(:active) + assert_response :success + assert_equal 'something specific', json_response['description'] + + # get the collection and verify newly added description + get "/arvados/v1/collections/#{collection['uuid']}", { + format: :json, + }, auth(:active) + assert_response 200 + assert_equal 'something specific', json_response['description'] + + # search + search_using_filter 'specific', 1 + search_using_filter 'not specific enough', 0 + end + + test "create collection, update manifest, and search with filename" do + # create collection + signed_manifest = Collection.sign_manifest(". bad42fa702ae3ea7d888fef11b46f450+44 0:44:my_test_file.txt\n", api_token(:active)) + post "/arvados/v1/collections", { + format: :json, + collection: {manifest_text: signed_manifest}.to_json, + }, auth(:active) + assert_response :success + assert_equal true, json_response['manifest_text'].include?('my_test_file.txt') + assert_includes json_response['manifest_text'], 'my_test_file.txt' + + created = json_response + + # search using the filename + search_using_filter 'my_test_file.txt', 1 + + # update the collection's manifest text + signed_manifest = Collection.sign_manifest(". bad42fa702ae3ea7d888fef11b46f450+44 0:44:my_updated_test_file.txt\n", api_token(:active)) + put "/arvados/v1/collections/#{created['uuid']}", { + format: :json, + collection: {manifest_text: signed_manifest}.to_json, + }, auth(:active) + assert_response :success + assert_equal created['uuid'], json_response['uuid'] + assert_includes json_response['manifest_text'], 'my_updated_test_file.txt' + assert_not_includes json_response['manifest_text'], 'my_test_file.txt' + + # search using the new filename + search_using_filter 'my_updated_test_file.txt', 1 + search_using_filter 'my_test_file.txt', 0 + search_using_filter 'there_is_no_such_file.txt', 0 + end + + def search_using_filter search_filter, expected_items + get '/arvados/v1/collections', { + :filters => [['any', 'ilike', "%#{search_filter}%"]].to_json + }, auth(:active) + assert_response :success + response_items = json_response['items'] + assert_not_nil response_items + if expected_items == 0 + assert_empty response_items + else + refute_empty response_items + first_item = response_items.first + assert_not_nil first_item + end end + test "search collection using full text search" do + # create collection to be searched for + signed_manifest = Collection.sign_manifest(". 85877ca2d7e05498dd3d109baf2df106+95+A3a4e26a366ee7e4ed3e476ccf05354761be2e4ae@545a9920 0:95:file_in_subdir1\n./subdir2/subdir3 2bbc341c702df4d8f42ec31f16c10120+64+A315d7e7bad2ce937e711fc454fae2d1194d14d64@545a9920 0:32:file1_in_subdir3.txt 32:32:file2_in_subdir3.txt\n./subdir2/subdir3/subdir4 2bbc341c702df4d8f42ec31f16c10120+64+A315d7e7bad2ce937e711fc454fae2d1194d14d64@545a9920 0:32:file3_in_subdir4.txt 32:32:file4_in_subdir4.txt\n", api_token(:active)) + post "/arvados/v1/collections", { + format: :json, + collection: {description: 'specific collection description', manifest_text: signed_manifest}.to_json, + }, auth(:active) + assert_response :success + assert_equal true, json_response['manifest_text'].include?('file4_in_subdir4.txt') + + created = json_response + + # search using the filename + search_using_full_text_search 'subdir2', 0 + search_using_full_text_search 'subdir2:*', 1 + search_using_full_text_search 'subdir2/subdir3/subdir4', 1 + search_using_full_text_search 'file4:*', 1 + search_using_full_text_search 'file4_in_subdir4.txt', 1 + search_using_full_text_search 'subdir2 file4:*', 0 # first word is incomplete + search_using_full_text_search 'subdir2/subdir3/subdir4 file4:*', 1 + search_using_full_text_search 'subdir2/subdir3/subdir4 file4_in_subdir4.txt', 1 + search_using_full_text_search 'ile4', 0 # not a prefix match + end + + def search_using_full_text_search search_filter, expected_items + get '/arvados/v1/collections', { + :filters => [['any', '@@', search_filter]].to_json + }, auth(:active) + assert_response :success + response_items = json_response['items'] + assert_not_nil response_items + if expected_items == 0 + assert_empty response_items + else + refute_empty response_items + first_item = response_items.first + assert_not_nil first_item + end + end + + # search for the filename in the file_names column and expect error + test "full text search not supported for individual columns" do + get '/arvados/v1/collections', { + :filters => [['name', '@@', 'General']].to_json + }, auth(:active) + assert_response 422 + end + + [ + 'quick fox', + 'quick_brown fox', + 'brown_ fox', + 'fox dogs', + ].each do |search_filter| + test "full text search ignores special characters and finds with filter #{search_filter}" do + # description: The quick_brown_fox jumps over the lazy_dog + # full text search treats '_' as space apparently + get '/arvados/v1/collections', { + :filters => [['any', '@@', search_filter]].to_json + }, auth(:active) + assert_response 200 + response_items = json_response['items'] + assert_not_nil response_items + first_item = response_items.first + refute_empty first_item + assert_equal first_item['description'], 'The quick_brown_fox jumps over the lazy_dog' + end + end end