X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/b50a3bcb38cf9e79416d8cc9b0d8b66249b3d473..f0a85e273056d0ad440084c11a37c73ce25fb4f6:/services/api/test/unit/collection_test.rb diff --git a/services/api/test/unit/collection_test.rb b/services/api/test/unit/collection_test.rb index 882e26059c..a1008eec4d 100644 --- a/services/api/test/unit/collection_test.rb +++ b/services/api/test/unit/collection_test.rb @@ -1,5 +1,9 @@ +# Copyright (C) The Arvados Authors. All rights reserved. +# +# SPDX-License-Identifier: AGPL-3.0 + require 'test_helper' -require 'sweep_trashed_collections' +require 'sweep_trashed_objects' class CollectionTest < ActiveSupport::TestCase include DbCurrentTime @@ -7,7 +11,7 @@ class CollectionTest < ActiveSupport::TestCase def create_collection name, enc=nil txt = ". d41d8cd98f00b204e9800998ecf8427e+0 0:0:#{name}.txt\n" txt.force_encoding(enc) if enc - return Collection.create(manifest_text: txt) + return Collection.create(manifest_text: txt, name: name) end test 'accept ASCII manifest_text' do @@ -102,6 +106,274 @@ class CollectionTest < ActiveSupport::TestCase end end + test "auto-create version after idle setting" do + Rails.configuration.collection_versioning = true + Rails.configuration.preserve_version_if_idle = 600 # 10 minutes + act_as_user users(:active) do + # Set up initial collection + c = create_collection 'foo', Encoding::US_ASCII + assert c.valid? + assert_equal 1, c.version + assert_equal false, c.preserve_version + # Make a versionable update, it shouldn't create a new version yet + c.update_attributes!({'name' => 'bar'}) + c.reload + assert_equal 'bar', c.name + assert_equal 1, c.version + # Update modified_at to trigger a version auto-creation + fifteen_min_ago = Time.now - 15.minutes + c.update_column('modified_at', fifteen_min_ago) # Update without validations/callbacks + c.reload + assert_equal fifteen_min_ago.to_i, c.modified_at.to_i + c.update_attributes!({'name' => 'baz'}) + c.reload + assert_equal 'baz', c.name + assert_equal 2, c.version + # Make another update, no new version should be created + c.update_attributes!({'name' => 'foobar'}) + c.reload + assert_equal 'foobar', c.name + assert_equal 2, c.version + end + end + + test "preserve_version=false assignment is ignored while being true and not producing a new version" do + Rails.configuration.collection_versioning = true + Rails.configuration.preserve_version_if_idle = 3600 + act_as_user users(:active) do + # Set up initial collection + c = create_collection 'foo', Encoding::US_ASCII + assert c.valid? + assert_equal 1, c.version + assert_equal false, c.preserve_version + # This update shouldn't produce a new version, as the idle time is not up + c.update_attributes!({ + 'name' => 'bar', + 'preserve_version' => true + }) + c.reload + assert_equal 1, c.version + assert_equal 'bar', c.name + assert_equal true, c.preserve_version + # Make sure preserve_version is not disabled after being enabled, unless + # a new version is created. + c.update_attributes!({ + 'preserve_version' => false, + 'replication_desired' => 2 + }) + c.reload + assert_equal 1, c.version + assert_equal 2, c.replication_desired + assert_equal true, c.preserve_version + c.update_attributes!({'name' => 'foobar'}) + c.reload + assert_equal 2, c.version + assert_equal false, c.preserve_version + assert_equal 'foobar', c.name + end + end + + test "uuid updates on current version make older versions update their pointers" do + Rails.configuration.collection_versioning = true + Rails.configuration.preserve_version_if_idle = 0 + act_as_system_user do + # Set up initial collection + c = create_collection 'foo', Encoding::US_ASCII + assert c.valid? + assert_equal 1, c.version + # Make changes so that a new version is created + c.update_attributes!({'name' => 'bar'}) + c.reload + assert_equal 2, c.version + assert_equal 2, Collection.where(current_version_uuid: c.uuid).count + new_uuid = 'zzzzz-4zz18-somefakeuuidnow' + assert_empty Collection.where(uuid: new_uuid) + # Update UUID on current version, check that both collections point to it + c.update_attributes!({'uuid' => new_uuid}) + c.reload + assert_equal new_uuid, c.uuid + assert_equal 2, Collection.where(current_version_uuid: new_uuid).count + end + end + + test "older versions' modified_at indicate when they're created" do + Rails.configuration.collection_versioning = true + Rails.configuration.preserve_version_if_idle = 0 + act_as_user users(:active) do + # Set up initial collection + c = create_collection 'foo', Encoding::US_ASCII + assert c.valid? + # Make changes so that a new version is created + c.update_attributes!({'name' => 'bar'}) + c.reload + assert_equal 2, c.version + # Get the old version + c_old = Collection.where(current_version_uuid: c.uuid, version: 1).first + assert_not_nil c_old + + version_creation_datetime = c_old.modified_at.to_f + assert_equal c.created_at.to_f, c_old.created_at.to_f + # Current version is updated just a few milliseconds before the version is + # saved on the database. + assert_operator c.modified_at.to_f, :<, version_creation_datetime + + # Make update on current version so old version get the attribute synced; + # its modified_at should not change. + new_replication = 3 + c.update_attributes!({'replication_desired' => new_replication}) + c.reload + assert_equal new_replication, c.replication_desired + c_old.reload + assert_equal new_replication, c_old.replication_desired + assert_equal version_creation_datetime, c_old.modified_at.to_f + assert_operator c.modified_at.to_f, :>, c_old.modified_at.to_f + end + end + + test "older versions should no be directly updatable" do + Rails.configuration.collection_versioning = true + Rails.configuration.preserve_version_if_idle = 0 + act_as_user users(:active) do + # Set up initial collection + c = create_collection 'foo', Encoding::US_ASCII + assert c.valid? + # Make changes so that a new version is created + c.update_attributes!({'name' => 'bar'}) + c.reload + assert_equal 2, c.version + # Get the old version + c_old = Collection.where(current_version_uuid: c.uuid, version: 1).first + assert_not_nil c_old + # With collection versioning still being enabled, try to update + assert_raises ArvadosModel::PermissionDeniedError do + c_old.update_attributes(name: 'this was foo') + end + c_old.reload + assert_equal 'foo', c_old.name + # Try to fool the validator attempting to make c_old to look like a + # current version, it should also fail. + assert_raises ArvadosModel::PermissionDeniedError do + c_old.update_attributes(current_version_uuid: c_old.uuid) + end + c_old.reload + assert_equal c.uuid, c_old.current_version_uuid + # Now disable collection versioning, it should behave the same way + Rails.configuration.collection_versioning = false + assert_raises ArvadosModel::PermissionDeniedError do + c_old.update_attributes(name: 'this was foo') + end + c_old.reload + assert_equal 'foo', c_old.name + end + end + + [ + ['owner_uuid', 'zzzzz-tpzed-d9tiejq69daie8f', 'zzzzz-tpzed-xurymjxw79nv3jz'], + ['replication_desired', 2, 3], + ['storage_classes_desired', ['hot'], ['archive']], + ['is_trashed', true, false], + ].each do |attr, first_val, second_val| + test "sync #{attr} with older versions" do + Rails.configuration.collection_versioning = true + Rails.configuration.preserve_version_if_idle = 0 + act_as_system_user do + # Set up initial collection + c = create_collection 'foo', Encoding::US_ASCII + assert c.valid? + assert_equal 1, c.version + assert_not_equal first_val, c.attributes[attr] + # Make changes so that a new version is created and a synced field is + # updated on both + c.update_attributes!({'name' => 'bar', attr => first_val}) + c.reload + assert_equal 2, c.version + assert_equal first_val, c.attributes[attr] + assert_equal 2, Collection.where(current_version_uuid: c.uuid).count + assert_equal first_val, Collection.where(current_version_uuid: c.uuid, version: 1).first.attributes[attr] + # Only make an update on the same synced field & check that the previously + # created version also gets it. + c.update_attributes!({attr => second_val}) + c.reload + assert_equal 2, c.version + assert_equal second_val, c.attributes[attr] + assert_equal 2, Collection.where(current_version_uuid: c.uuid).count + assert_equal second_val, Collection.where(current_version_uuid: c.uuid, version: 1).first.attributes[attr] + end + end + end + + [ + [false, 'name', 'bar', false], + [false, 'description', 'The quick brown fox jumps over the lazy dog', false], + [false, 'properties', {'new_version' => true}, false], + [false, 'manifest_text', ". d41d8cd98f00b204e9800998ecf8427e 0:0:foo.txt\n", false], + [true, 'name', 'bar', true], + [true, 'description', 'The quick brown fox jumps over the lazy dog', true], + [true, 'properties', {'new_version' => true}, true], + [true, 'manifest_text', ". d41d8cd98f00b204e9800998ecf8427e 0:0:foo.txt\n", true], + # Non-versionable attribute updates shouldn't create new versions + [true, 'replication_desired', 5, false], + [false, 'replication_desired', 5, false], + ].each do |versioning, attr, val, new_version_expected| + test "update #{attr} with versioning #{versioning ? '' : 'not '}enabled should #{new_version_expected ? '' : 'not '}create a new version" do + Rails.configuration.collection_versioning = versioning + Rails.configuration.preserve_version_if_idle = 0 + act_as_user users(:active) do + # Create initial collection + c = create_collection 'foo', Encoding::US_ASCII + assert c.valid? + assert_equal 'foo', c.name + + # Check current version attributes + assert_equal 1, c.version + assert_equal c.uuid, c.current_version_uuid + + # Update attribute and check if version number should be incremented + old_value = c.attributes[attr] + c.update_attributes!({attr => val}) + assert_equal new_version_expected, c.version == 2 + assert_equal val, c.attributes[attr] + + if versioning && new_version_expected + # Search for the snapshot & previous value + assert_equal 2, Collection.where(current_version_uuid: c.uuid).count + s = Collection.where(current_version_uuid: c.uuid, version: 1).first + assert_not_nil s + assert_equal old_value, s.attributes[attr] + else + # If versioning is disabled or no versionable attribute was updated, + # only the current version should exist + assert_equal 1, Collection.where(current_version_uuid: c.uuid).count + assert_equal c, Collection.where(current_version_uuid: c.uuid).first + end + end + end + end + + test 'with versioning enabled, simultaneous updates increment version correctly' do + Rails.configuration.collection_versioning = true + Rails.configuration.preserve_version_if_idle = 0 + act_as_user users(:active) do + # Create initial collection + col = create_collection 'foo', Encoding::US_ASCII + assert col.valid? + assert_equal 1, col.version + + # Simulate simultaneous updates + c1 = Collection.where(uuid: col.uuid).first + assert_equal 1, c1.version + c1.name = 'bar' + c2 = Collection.where(uuid: col.uuid).first + c2.description = 'foo collection' + c1.save! + assert_equal 1, c2.version + # with_lock forces a reload, so this shouldn't produce an unique violation error + c2.save! + assert_equal 3, c2.version + assert_equal 'foo collection', c2.description + end + end + test 'create and update collection and verify file_names' do act_as_system_user do c = create_collection 'foo', Encoding::US_ASCII @@ -217,6 +489,81 @@ class CollectionTest < ActiveSupport::TestCase end end + test "storage_classes_desired cannot be empty" do + act_as_user users(:active) do + c = collections(:collection_owned_by_active) + c.update_attributes storage_classes_desired: ["hot"] + assert_equal ["hot"], c.storage_classes_desired + assert_raise ArvadosModel::InvalidStateTransitionError do + c.update_attributes storage_classes_desired: [] + end + end + end + + test "storage classes lists should only contain non-empty strings" do + c = collections(:storage_classes_desired_default_unconfirmed) + act_as_user users(:admin) do + assert c.update_attributes(storage_classes_desired: ["default", "a_string"], + storage_classes_confirmed: ["another_string"]) + [ + ["storage_classes_desired", ["default", 42]], + ["storage_classes_confirmed", [{the_answer: 42}]], + ["storage_classes_desired", ["default", ""]], + ["storage_classes_confirmed", [""]], + ].each do |attr, val| + assert_raise ArvadosModel::InvalidStateTransitionError do + assert c.update_attributes({attr => val}) + end + end + end + end + + test "storage_classes_confirmed* can be set by admin user" do + c = collections(:storage_classes_desired_default_unconfirmed) + act_as_user users(:admin) do + assert c.update_attributes(storage_classes_confirmed: ["default"], + storage_classes_confirmed_at: Time.now) + end + end + + test "storage_classes_confirmed* cannot be set by non-admin user" do + act_as_user users(:active) do + c = collections(:storage_classes_desired_default_unconfirmed) + # Cannot set just one at a time. + assert_raise ArvadosModel::PermissionDeniedError do + c.update_attributes storage_classes_confirmed: ["default"] + end + c.reload + assert_raise ArvadosModel::PermissionDeniedError do + c.update_attributes storage_classes_confirmed_at: Time.now + end + # Cannot set bot at once, either. + c.reload + assert_raise ArvadosModel::PermissionDeniedError do + assert c.update_attributes(storage_classes_confirmed: ["default"], + storage_classes_confirmed_at: Time.now) + end + end + end + + test "storage_classes_confirmed* can be cleared (but only together) by non-admin user" do + act_as_user users(:active) do + c = collections(:storage_classes_desired_default_confirmed_default) + # Cannot clear just one at a time. + assert_raise ArvadosModel::PermissionDeniedError do + c.update_attributes storage_classes_confirmed: [] + end + c.reload + assert_raise ArvadosModel::PermissionDeniedError do + c.update_attributes storage_classes_confirmed_at: nil + end + # Can clear both at once. + c.reload + assert c.update_attributes(storage_classes_confirmed: [], + storage_classes_confirmed_at: nil) + end + end + [0, 2, 4, nil].each do |ask| test "set replication_desired to #{ask.inspect}" do Rails.configuration.default_collection_replication = 2 @@ -348,9 +695,12 @@ class CollectionTest < ActiveSupport::TestCase assert c.valid? uuid = c.uuid + c = Collection.readable_by(current_user).where(uuid: uuid) + assert_not_empty c, 'Should be able to find live collection' + # mark collection as expired - c.update_attributes!(trash_at: Time.new.strftime("%Y-%m-%d")) - c = Collection.where(uuid: uuid) + c.first.update_attributes!(trash_at: Time.new.strftime("%Y-%m-%d")) + c = Collection.readable_by(current_user).where(uuid: uuid) assert_empty c, 'Should not be able to find expired collection' # recreate collection with the same name @@ -415,7 +765,7 @@ class CollectionTest < ActiveSupport::TestCase if fixture_name == :expired_collection # Fixture-finder shorthand doesn't find trashed collections # because they're not in the default scope. - c = Collection.unscoped.find_by_uuid('zzzzz-4zz18-mto52zx1s7sn3ih') + c = Collection.find_by_uuid('zzzzz-4zz18-mto52zx1s7sn3ih') else c = collections(fixture_name) end @@ -474,7 +824,7 @@ class CollectionTest < ActiveSupport::TestCase assert_includes(coll_uuids, collections(:docker_image).uuid) end - test "move to trash in SweepTrashedCollections" do + test "move collections to trash in SweepTrashedObjects" do c = collections(:trashed_on_next_sweep) refute_empty Collection.where('uuid=? and is_trashed=false', c.uuid) assert_raises(ActiveRecord::RecordNotUnique) do @@ -483,8 +833,8 @@ class CollectionTest < ActiveSupport::TestCase name: c.name) end end - SweepTrashedCollections.sweep_now - c = Collection.unscoped.where('uuid=? and is_trashed=true', c.uuid).first + SweepTrashedObjects.sweep_now + c = Collection.where('uuid=? and is_trashed=true', c.uuid).first assert c act_as_user users(:active) do assert Collection.create!(owner_uuid: c.owner_uuid, @@ -492,14 +842,14 @@ class CollectionTest < ActiveSupport::TestCase end end - test "delete in SweepTrashedCollections" do + test "delete collections in SweepTrashedObjects" do uuid = 'zzzzz-4zz18-3u1p5umicfpqszp' # deleted_on_next_sweep - assert_not_empty Collection.unscoped.where(uuid: uuid) - SweepTrashedCollections.sweep_now - assert_empty Collection.unscoped.where(uuid: uuid) + assert_not_empty Collection.where(uuid: uuid) + SweepTrashedObjects.sweep_now + assert_empty Collection.where(uuid: uuid) end - test "delete referring links in SweepTrashedCollections" do + test "delete referring links in SweepTrashedObjects" do uuid = collections(:trashed_on_next_sweep).uuid act_as_system_user do Link.create!(head_uuid: uuid, @@ -508,10 +858,10 @@ class CollectionTest < ActiveSupport::TestCase name: 'something') end past = db_current_time - Collection.unscoped.where(uuid: uuid). + Collection.where(uuid: uuid). update_all(is_trashed: true, trash_at: past, delete_at: past) - assert_not_empty Collection.unscoped.where(uuid: uuid) - SweepTrashedCollections.sweep_now - assert_empty Collection.unscoped.where(uuid: uuid) + assert_not_empty Collection.where(uuid: uuid) + SweepTrashedObjects.sweep_now + assert_empty Collection.where(uuid: uuid) end end