From 1489cc8d327c65bc8675ebead2f663e1a6f4d6a3 Mon Sep 17 00:00:00 2001 From: willdunklin Date: Tue, 17 Oct 2023 16:05:29 -0400 Subject: [PATCH 1/7] Wrap DICOMweb assetstore importData function --- import_tracker/__init__.py | 83 +++++++++++++++++++ .../web_client/templates/importList.pug | 5 +- 2 files changed, 85 insertions(+), 3 deletions(-) diff --git a/import_tracker/__init__.py b/import_tracker/__init__.py index ff3d0e4..8b5afba 100644 --- a/import_tracker/__init__.py +++ b/import_tracker/__init__.py @@ -162,6 +162,84 @@ def shouldImportFileWrapper(self, path, params): AbstractAssetstoreAdapter.shouldImportFile = shouldImportFileWrapper +def wrapDICOMImport(assetstoreResource): + baseImportData = assetstoreResource.importData + baseImportData.description.param( + 'excludeExisting', + 'If true, then a file with an import path that is already in the ' + 'system is not imported, even if it is not in the destination ' + 'hierarchy.', dataType='boolean', required=False, default=False) + + @boundHandler(ctx=assetstoreResource) + @autoDescribeRoute(baseImportData.description) + def dwaImportDataWrapper(self, assetstore, destinationId, destinationType, filters, + progress, excludeExisting): + + user = self.getCurrentUser() + params = { + 'destinationId': destinationId, + 'destinationType': destinationType, + 'filters': filters, + 'progress': str(progress).lower(), + } + if excludeExisting: + params['excludeExisting'] = str(excludeExisting).lower() + + importRecord = AssetstoreImport().createAssetstoreImport(assetstore, params) + job = Job().createJob( + title=f'Import from {assetstore["name"]}', + type='assetstore_import', + public=False, + user=user, + kwargs=params, + ) + job = Job().updateJob(job, '%s - Starting import from %s\n' % ( + time.strftime('%Y-%m-%d %H:%M:%S'), + assetstore['name'] + ), status=JobStatus.RUNNING) + + try: + with ProgressContext(progress, user=user, title='Importing data') as ctx: + try: + jobRec = { + 'id': str(job['_id']), + 'count': 0, + 'skip': 0, + 'lastlog': time.time(), + 'logcount': 0, + } + self._importData(assetstore, params={**params, '_job': jobRec}) + + success = True + Job().updateJob(job, '%s - Finished. Checked %d, skipped %d\n' % ( + time.strftime('%Y-%m-%d %H:%M:%S'), + jobRec['count'], jobRec['skip'], + ), status=JobStatus.SUCCESS) + + except ImportTrackerCancelError: + Job().updateJob(job, '%s - Canceled' % ( + time.strftime('%Y-%m-%d %H:%M:%S'), + )) + success = 'canceled' + + except Exception as exc: + Job().updateJob(job, '%s - Failed with %s\n' % ( + time.strftime('%Y-%m-%d %H:%M:%S'), + exc, + ), status=JobStatus.ERROR) + success = False + + importRecord = AssetstoreImport().markEnded(importRecord, success) + return importRecord + + for key in {'accessLevel', 'description', 'requiredScopes'}: + setattr(dwaImportDataWrapper, key, getattr(baseImportData, key)) + + assetstoreResource.importData = dwaImportDataWrapper + assetstoreResource.removeRoute('POST', (':id', 'import')) + assetstoreResource.route('POST', (':id', 'import'), assetstoreResource.importData) + + class GirderPlugin(plugin.GirderPlugin): DISPLAY_NAME = 'import_tracker' CLIENT_SOURCE_PATH = 'web_client' @@ -178,3 +256,8 @@ def load(self, info): wrapImportData(info['apiRoot'].assetstore) info['apiRoot'].folder.route('PUT', (':id', 'move'), moveFolder) + + if hasattr(info['apiRoot'], 'dicomweb_assetstore'): + wrapDICOMImport(info['apiRoot'].dicomweb_assetstore) + else: + print("dicomweb_assetstore not found, correct large_image version may not be loaded") diff --git a/import_tracker/web_client/templates/importList.pug b/import_tracker/web_client/templates/importList.pug index 7147c4a..4f2b34b 100644 --- a/import_tracker/web_client/templates/importList.pug +++ b/import_tracker/web_client/templates/importList.pug @@ -76,9 +76,8 @@ table.g-imports-list-table span= _import._assetstoreName td(data-id=_import.params.importPath, data-toggle='tooltip', title=_import.params.importPath) span= _import.params.importPath - //- - td - span= _import.params.destinationType + td + span= _import.params.destinationType td(data-id=_import.params.destinationId, data-toggle='tooltip', title=_import._destinationPath + '\n' + _import.params.destinationId) span= _import._destinationPath if anyLeafed From 8b69dc15b0568f08ee48d96a714985f88f511462 Mon Sep 17 00:00:00 2001 From: willdunklin Date: Tue, 17 Oct 2023 16:47:09 -0400 Subject: [PATCH 2/7] Support dwas imports in import list template --- .../web_client/templates/importList.pug | 23 +++++++++++-------- 1 file changed, 13 insertions(+), 10 deletions(-) diff --git a/import_tracker/web_client/templates/importList.pug b/import_tracker/web_client/templates/importList.pug index 4f2b34b..a8b2754 100644 --- a/import_tracker/web_client/templates/importList.pug +++ b/import_tracker/web_client/templates/importList.pug @@ -21,7 +21,8 @@ table.g-imports-list-table - var anyRegex = false; var anyLeafed = false; - var anyNoProgress; + var anyImportPath = false; + var anyNoProgress = false; var otherParams = []; var showCount = false; for _import in imports @@ -31,9 +32,11 @@ table.g-imports-list-table - anyLeafed = true if !_import.params.progress - anyNoProgress = true - - Object.keys(_import.params).forEach((key) => { if (['importPath', 'destinationId', 'destinationType', 'leafFoldersAsItems', 'progress', 'fileIncludeRegex', 'fileExcludeRegex'].indexOf(key) < 0 && otherParams.indexOf(key) < 0) { otherParams.push(key); } }); + - Object.keys(_import.params).forEach((key) => { if (['importPath', 'destinationId', 'leafFoldersAsItems', 'progress', 'fileIncludeRegex', 'fileExcludeRegex'].indexOf(key) < 0 && otherParams.indexOf(key) < 0) { otherParams.push(key); } }); if _import._count - - showCount = true; + - showCount = true + if _import.params.importPath + - anyImportPath = true thead tr th Actions @@ -42,7 +45,8 @@ table.g-imports-list-table th Started th Ended th Assetstore Name - th Import Path + if anyImportPath + th Import Path //- th Destination Type th Destination Path if anyLeafed @@ -59,11 +63,11 @@ table.g-imports-list-table tr td if assetstoreExists[i] - div(style='display: flex; justify-content: flex-end;') + div(style='display: flex; padding-left: 10px; padding-right: 10px;') button.re-import-btn.btn.btn-sm.btn-success(index=i, disabled=(_import._destinationPath =='does not exist')) i.icon-cw | Re-Import - button.re-import-edit-btn.btn.btn-sm.btn-primary(index=i, style='margin-left: 5px;', data-toggle='tooltip', title='Edit Import Parameters') + button.re-import-edit-btn.btn.btn-sm.btn-primary(index=i, style='margin-left: 1em;', data-toggle='tooltip', title='Edit Import Parameters') i.icon-pencil if showCount td(data-id=_import._count, data-toggle='tooltip', title=_import._count !== 1 ? `Imported ${_import._count} times` : 'Imported once') @@ -74,10 +78,9 @@ table.g-imports-list-table span= _import.ended ? moment(_import.ended).format('YYYY-MM-DD HH:mm:ss.SSS') : '' td(data-id=_import._assetstoreName, data-toggle='tooltip', title=_import._assetstoreName) span= _import._assetstoreName - td(data-id=_import.params.importPath, data-toggle='tooltip', title=_import.params.importPath) - span= _import.params.importPath - td - span= _import.params.destinationType + if anyImportPath + td(data-id=_import.params.importPath, data-toggle='tooltip', title=_import.params.importPath) + span= _import.params.importPath td(data-id=_import.params.destinationId, data-toggle='tooltip', title=_import._destinationPath + '\n' + _import.params.destinationId) span= _import._destinationPath if anyLeafed From 4bd41be52977379b278321a48f167be922dce1bc Mon Sep 17 00:00:00 2001 From: willdunklin Date: Wed, 18 Oct 2023 12:48:04 -0400 Subject: [PATCH 3/7] Add re-import support for dwas imports --- import_tracker/web_client/views/importList.js | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/import_tracker/web_client/views/importList.js b/import_tracker/web_client/views/importList.js index edf8354..417f3e7 100644 --- a/import_tracker/web_client/views/importList.js +++ b/import_tracker/web_client/views/importList.js @@ -2,6 +2,7 @@ import $ from 'jquery'; import moment from 'moment'; import AssetstoreModel from '@girder/core/models/AssetstoreModel'; +import { AssetstoreType } from '@girder/core/constants'; import View from '@girder/core/views/View'; import router from '@girder/core/router'; import { restRequest } from '@girder/core/rest'; @@ -22,11 +23,20 @@ var importList = View.extend({ const assetstore = new AssetstoreModel({ _id: importEvent.assetstoreId }); const destType = importEvent.params.destinationType; const destId = importEvent.params.destinationId; + assetstore.off('g:imported').on('g:imported', function () { router.navigate(destType + '/' + destId, { trigger: true }); }, this).on('g:error', function (resp) { this.$('.g-validation-failed-message').text(resp.responseJSON.message); - }, this).import(importEvent.params); + }, this); + + assetstore.once('g:fetched', () => { + if (assetstore.get('type') === AssetstoreType.DICOMWEB) { + assetstore.dicomwebImport(importEvent.params); + } else { + assetstore.import(importEvent.params); + } + }).fetch(); }, 'click .re-import-edit-btn': function (e) { const index = Number($(e.currentTarget).attr('index')); From 4a01a19210ba73980bfc2dedbbd1868001958a39 Mon Sep 17 00:00:00 2001 From: willdunklin Date: Wed, 18 Oct 2023 13:15:52 -0400 Subject: [PATCH 4/7] Disable re-import editing for dwas imports for now --- import_tracker/web_client/views/importList.js | 20 ++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/import_tracker/web_client/views/importList.js b/import_tracker/web_client/views/importList.js index 417f3e7..448d252 100644 --- a/import_tracker/web_client/views/importList.js +++ b/import_tracker/web_client/views/importList.js @@ -45,10 +45,24 @@ var importList = View.extend({ return; } + // Navigate to re-import page + const navigate = (assetstoreId, importId) => { + const assetstore = new AssetstoreModel({ _id: assetstoreId }); + assetstore.once('g:fetched', () => { + if (assetstore.get('type') === AssetstoreType.DICOMWEB) { + // Avoid adding previous import data for DICOMweb imports by navigating to blank import + // TODO: Add DICOMweb-specific re-import view + router.navigate(`dicomweb_assetstore/${assetstoreId}/import`, { trigger: true }); + } else { + router.navigate(`assetstore/${assetstoreId}/re-import/${importId}`, { trigger: true }); + } + }).fetch(); + }; + const assetstoreId = importEvent.assetstoreId; - const importId = importEvent._id; + const importId = importEvent._id; // Only individual imports have an _id if (importId) { - router.navigate(`assetstore/${assetstoreId}/re-import/${importId}`, { trigger: true }); + navigate(assetstoreId, importId); return; } @@ -63,7 +77,7 @@ var importList = View.extend({ i.params.destinationType === importEvent.params.destinationType )[0]._id; - router.navigate(`assetstore/${assetstoreId}/re-import/${importId}`, { trigger: true }); + navigate(assetstoreId, importId); }); } }, From b5664aa9b843eb28f0d58b0f64a2cbf61301226d Mon Sep 17 00:00:00 2001 From: willdunklin Date: Wed, 1 Nov 2023 10:41:14 -0400 Subject: [PATCH 5/7] Pass dwas progress context from wrapped importData --- import_tracker/__init__.py | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/import_tracker/__init__.py b/import_tracker/__init__.py index 8b5afba..fd5e63e 100644 --- a/import_tracker/__init__.py +++ b/import_tracker/__init__.py @@ -1,6 +1,6 @@ import time -from girder import plugin +from girder import logger, plugin from girder.api.describe import autoDescribeRoute from girder.api.rest import boundHandler from girder.constants import AccessType @@ -208,7 +208,12 @@ def dwaImportDataWrapper(self, assetstore, destinationId, destinationType, filte 'lastlog': time.time(), 'logcount': 0, } - self._importData(assetstore, params={**params, '_job': jobRec}) + self._importData( + assetstore, + params={ + **params, + '_job': jobRec}, + progress=ctx) success = True Job().updateJob(job, '%s - Finished. Checked %d, skipped %d\n' % ( @@ -260,4 +265,4 @@ def load(self, info): if hasattr(info['apiRoot'], 'dicomweb_assetstore'): wrapDICOMImport(info['apiRoot'].dicomweb_assetstore) else: - print("dicomweb_assetstore not found, correct large_image version may not be loaded") + logger.info('dicomweb_assetstore not found') From 106b71461f96ee49b45a2da4d917c387daac1937 Mon Sep 17 00:00:00 2001 From: David Manthey Date: Tue, 7 Nov 2023 16:49:45 -0500 Subject: [PATCH 6/7] Handle optional plugins and plugin order. --- import_tracker/__init__.py | 5 +++++ import_tracker/web_client/main.js | 29 +++++++++++++++++++---------- 2 files changed, 24 insertions(+), 10 deletions(-) diff --git a/import_tracker/__init__.py b/import_tracker/__init__.py index fd5e63e..110848e 100644 --- a/import_tracker/__init__.py +++ b/import_tracker/__init__.py @@ -251,6 +251,11 @@ class GirderPlugin(plugin.GirderPlugin): def load(self, info): plugin.getPlugin('jobs').load(info) + try: + import large_image_source_dicom # noqa + plugin.getPlugin('dicomweb').load(info) + except ImportError: + pass ModelImporter.registerModel( 'assetstoreImport', AssetstoreImport, 'import_tracker' ) diff --git a/import_tracker/web_client/main.js b/import_tracker/web_client/main.js index 2e5c407..e51fea2 100644 --- a/import_tracker/web_client/main.js +++ b/import_tracker/web_client/main.js @@ -1,3 +1,5 @@ +import $ from 'jquery'; + import AssetstoreView from '@girder/core/views/body/AssetstoresView'; import FilesystemImportView from '@girder/core/views/body/FilesystemImportView'; import S3ImportView from '@girder/core/views/body/S3ImportView'; @@ -22,16 +24,23 @@ import './JobStatus'; wrap(AssetstoreView, 'render', function (render) { // Call the underlying render function that we are wrapping render.call(this); - - this.$el.find('.g-current-assetstores-container .g-body-title').after( - 'View all past Imports' - ); - - // Inject new button into each assetstore - const assetstores = this.collection.toArray(); - this.$('.g-assetstore-import-button-container').after( - (i) => importDataButton({ importsPageLink: `#assetstore/${assetstores[i].id}/imports` }) - ); + // defer adding buttons so optional plugins can render first. + window.setTimeout(() => { + this.$el.find('.g-current-assetstores-container .g-body-title').after( + 'View all past Imports' + ); + + // Inject new button into each assetstore + const assetstores = this.collection; + this.$('.g-assetstore-import-button-container').after( + function () { + // we can't just use the index of the after call, since not + // all assetstores will have import buttons. + var assetstore = assetstores.get($(this).closest('.g-assetstore-buttons').find('[cid]').attr('cid')); + return importDataButton({ importsPageLink: `#assetstore/${assetstore.id}/imports` }); + } + ); + }, 0); }); // Add duplicate_files option to Import Asset form From 3330e9693f2e882a61d3b255e834b6e68c56009f Mon Sep 17 00:00:00 2001 From: willdunklin Date: Wed, 8 Nov 2023 11:24:05 -0500 Subject: [PATCH 7/7] Remove var usage in AssetstoreView render code --- import_tracker/web_client/main.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/import_tracker/web_client/main.js b/import_tracker/web_client/main.js index e51fea2..94b072a 100644 --- a/import_tracker/web_client/main.js +++ b/import_tracker/web_client/main.js @@ -36,7 +36,7 @@ wrap(AssetstoreView, 'render', function (render) { function () { // we can't just use the index of the after call, since not // all assetstores will have import buttons. - var assetstore = assetstores.get($(this).closest('.g-assetstore-buttons').find('[cid]').attr('cid')); + const assetstore = assetstores.get($(this).closest('.g-assetstore-buttons').find('[cid]').attr('cid')); return importDataButton({ importsPageLink: `#assetstore/${assetstore.id}/imports` }); } );