Skip to content

Commit

Permalink
Handle file replace during the migration and update pytest packages
Browse files Browse the repository at this point in the history
  • Loading branch information
davidfarkas committed Mar 8, 2018
1 parent 0118323 commit df6eeeb
Show file tree
Hide file tree
Showing 5 changed files with 58 additions and 14 deletions.
2 changes: 1 addition & 1 deletion api/jobs/handlers.py
Original file line number Diff line number Diff line change
Expand Up @@ -93,7 +93,7 @@ def download(self, **kwargs): # pragma: no cover
'hash': 'v0-' + gear['exchange']['rootfs-hash'].replace(':', '-')
})

stream = open(file_path, 'rb')
stream = file_system.open(file_path, 'rb')
set_for_download(self.response, stream=stream, filename='gear.tar')

@require_admin
Expand Down
10 changes: 8 additions & 2 deletions bin/oneoffs/migrate_storage.py
Original file line number Diff line number Diff line change
Expand Up @@ -114,11 +114,17 @@ def migrate_collections():
}
log.debug('update file in mongo: %s' % update_set)
# Update the file with the newly generated UUID
config.db[f['collection']].find_one_and_update(
{'_id': f['collection_id'], f['prefix'] + '.name': f['fileinfo']['name']},
updated_doc = config.db[f['collection']].find_one_and_update(
{'_id': f['collection_id'], f['prefix'] + '.name': f['fileinfo']['name'],
f['prefix'] + '.hash': f['fileinfo']['hash']},
{'$set': update_set}
)

if not updated_doc:
log.info('Probably the following file has been updated during the migration '
'and its hash is changed, cleaning up from the new filesystem')
config.fs.remove(f_new_path)

show_progress(i + 1, len(_files))

except Exception as e:
Expand Down
50 changes: 44 additions & 6 deletions tests/integration_tests/python/test_migrate_storage.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
import fs.move
import fs.path
import pytest
import pymongo

from api import config, util
from bson.objectid import ObjectId
Expand Down Expand Up @@ -141,7 +142,7 @@ def files_to_migrate(data_builder, api_db, as_admin, randstr, file_form):
)

move_file_to_legacy(util.path_from_uuid(file_id_1), util.path_from_hash(file_hash_1))
files.append((url_1, util.path_from_hash(file_hash_1)))
files.append((project_id, file_name_1, url_1, util.path_from_hash(file_hash_1)))
# Create an UUID file
file_name_2 = '%s.csv' % randstr()
file_content_2 = randstr()
Expand All @@ -154,7 +155,7 @@ def files_to_migrate(data_builder, api_db, as_admin, randstr, file_form):
url_2 = '/projects/' + project_id + '/files/' + file_name_2

move_file_to_legacy(util.path_from_uuid(file_id_2), util.path_from_uuid(file_id_2))
files.append((url_2, util.path_from_uuid(file_id_2)))
files.append((project_id, file_name_2, url_2, util.path_from_uuid(file_id_2)))

yield files

Expand All @@ -173,9 +174,9 @@ def test_migrate_collections(files_to_migrate, as_admin, migrate_storage):
"""Testing collection migration"""

# get file storing by hash in legacy storage
(url_1, file_path_1) = files_to_migrate[0]
(_, _, url_1, file_path_1) = files_to_migrate[0]
# get ile storing by uuid in legacy storage
(url_2, file_path_2) = files_to_migrate[1]
(_, _,url_2, file_path_2) = files_to_migrate[1]

# get the ticket
r = as_admin.get(url_1, params={'ticket': ''})
Expand Down Expand Up @@ -220,9 +221,9 @@ def test_migrate_collections_error(files_to_migrate, migrate_storage):
"""Testing that the migration script throws an exception if it couldn't migrate a file"""

# get file storing by hash in legacy storage
(url, file_path_1) = files_to_migrate[0]
(_, _, url, file_path_1) = files_to_migrate[0]
# get the other file, so we can clean up
(_, file_path_2) = files_to_migrate[1]
(_, _, _, file_path_2) = files_to_migrate[1]

# delete the file
config.legacy_fs.remove(file_path_1)
Expand Down Expand Up @@ -272,3 +273,40 @@ def test_migrate_gears_error(gears_to_migrate, migrate_storage):

# clean up
config.legacy_fs.remove(gear_file_path_2)


def test_file_replaced_handling(files_to_migrate, migrate_storage, as_admin, file_form, api_db, mocker, caplog):

origin_find_one_and_update = pymongo.collection.Collection.find_one_and_update

def mocked(*args, **kwargs):
self = args[0]
filter = args[1]
update = args[2]

as_admin.post('/projects/' + project_id + '/files', files=file_form((file_name_1, 'new_content')))

return origin_find_one_and_update(self, filter, update)


with mocker.mock_module.patch.object(pymongo.collection.Collection, 'find_one_and_update', mocked):
# get file storing by hash in legacy storage
(project_id, file_name_1, url_1, file_path_1) = files_to_migrate[0]
# get ile storing by uuid in legacy storage
(_, file_name_2, url_2, file_path_2) = files_to_migrate[1]

# run the migration
migrate_storage.migrate_collections()

file_1_id = api_db['projects'].find_one(
{'files.name': file_name_1}
)['files'][0]['_id']

file_2_id = api_db['projects'].find_one(
{'files.name': file_name_2}
)['files'][1]['_id']

assert config.fs.exists(util.path_from_uuid(file_1_id))
assert config.fs.exists(util.path_from_uuid(file_2_id))

assert any(log.message == 'Probably the following file has been updated during the migration and its hash is changed, cleaning up from the new filesystem' for log in caplog.records)
2 changes: 1 addition & 1 deletion tests/integration_tests/python/test_undelete.py
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ def test_undelete_scope(undelete, containers, as_admin, api_db):


def test_undelete_options(undelete, containers):
with pytest.raises(RuntimeError, match=r'use --include-parents'):
with pytest.raises(RuntimeError, match=r'use `--include-parents`'):
undelete('acquisitions', containers.ac_1_1_1, filename='f_1_1_1_1')

undelete('acquisitions', containers.ac_1_1_1, filename='f_1_1_1_1', include_parents=True)
Expand Down
8 changes: 4 additions & 4 deletions tests/integration_tests/requirements-integration-test.txt
Original file line number Diff line number Diff line change
Expand Up @@ -4,9 +4,9 @@ mock==2.0.0
mongomock==3.8.0
pdbpp==0.8.3
pylint==1.5.3
pytest-cov==2.2.0
pytest-mock==1.6.0
pytest-watch==3.8.0
pytest==2.8.5
pytest-cov==2.5.1
pytest-mock==1.7.0
pytest-watch==4.1.0
pytest==3.4.1
requests_mock==1.3.0
testfixtures==4.10.1

0 comments on commit df6eeeb

Please sign in to comment.