# Copyright (C) The Arvados Authors. All rights reserved.
#
# SPDX-License-Identifier: AGPL-3.0

require 'test_helper'

class CollectionsApiTest < ActionDispatch::IntegrationTest
  fixtures :all

  test "should get index" do
    get "/arvados/v1/collections", {:format => :json}, auth(:active)
    assert_response :success
    assert_equal "arvados#collectionList", json_response['kind']
  end

  test "get index with filters= (empty string)" do
    get "/arvados/v1/collections", {:format => :json, :filters => ''}, auth(:active)
    assert_response :success
    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 => ''}, auth(:active)
    assert_response :success
    assert_equal "arvados#collectionList", json_response['kind']
  end

  test "get index with select= (valid attribute)" do
    get "/arvados/v1/collections", {
          :format => :json,
          :select => ['portable_data_hash'].to_json
        }, auth(:active)
    assert_response :success
    assert json_response['items'][0].keys.include?('portable_data_hash')
    assert not(json_response['items'][0].keys.include?('uuid'))
  end

  test "get index with select= (invalid attribute) responds 422" do
    get "/arvados/v1/collections", {
          :format => :json,
          :select => ['bogus'].to_json
        }, auth(:active)
    assert_response 422
    assert_match(/Invalid attribute.*bogus/, json_response['errors'].join(' '))
  end

  test "get index with select= (invalid attribute type) responds 422" do
    get "/arvados/v1/collections", {
          :format => :json,
          :select => [['bogus']].to_json
        }, auth(:active)
    assert_response 422
    assert_match(/Invalid attribute.*bogus/, json_response['errors'].join(' '))
  end

  test "controller 404 response is json" do
    get "/arvados/v1/thingsthatdonotexist", {:format => :xml}, auth(:active)
    assert_response 404
    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", {}, auth(:active)
    assert_response 404
    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')

    # 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

  test "create and get collection with properties" do
    # create collection to be searched for
    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 200
    assert_not_nil json_response['uuid']
    assert_not_nil json_response['properties']
    assert_empty json_response['properties']

    # update collection's description
    put "/arvados/v1/collections/#{json_response['uuid']}", {
      format: :json,
      collection: { properties: {'property_1' => 'value_1'} }
    }, auth(:active)
    assert_response :success
    assert_equal 'value_1', json_response['properties']['property_1']
  end
end
