diff --git a/src/io/import/processors/restoreStateFile.ts b/src/io/import/processors/restoreStateFile.ts index 840b3120c..4d4502c81 100644 --- a/src/io/import/processors/restoreStateFile.ts +++ b/src/io/import/processors/restoreStateFile.ts @@ -26,7 +26,10 @@ import { makeImageSelection, useDatasetStore, } from '@/src/store/datasets'; -import { useSegmentGroupStore } from '@/src/store/segmentGroups'; +import { + makeDefaultSegmentGroupName, + useSegmentGroupStore, +} from '@/src/store/segmentGroups'; import { useToolStore } from '@/src/store/tools'; import { useLayersStore } from '@/src/store/datasets-layers'; import { extractFilesFromZip } from '@/src/io/zip'; @@ -34,6 +37,114 @@ import downloadUrl from '@/src/io/import/processors/downloadUrl'; import updateFileMimeType from '@/src/io/import/processors/updateFileMimeType'; import extractArchiveTarget from '@/src/io/import/processors/extractArchiveTarget'; +const LABELMAP_PALETTE_2_1_0 = { + '0': { + value: 0, + name: 'Segment 0', + color: [0, 0, 0, 0], + }, + '1': { + value: 1, + name: 'Segment 1', + color: [153, 153, 0, 255], + }, + '2': { + value: 2, + name: 'Segment 2', + color: [76, 76, 0, 255], + }, + '3': { + value: 3, + name: 'Segment 3', + color: [255, 255, 0, 255], + }, + '4': { + value: 4, + name: 'Segment 4', + color: [0, 76, 0, 255], + }, + '5': { + value: 5, + name: 'Segment 5', + color: [0, 153, 0, 255], + }, + '6': { + value: 6, + name: 'Segment 6', + color: [0, 255, 0, 255], + }, + '7': { + value: 7, + name: 'Segment 7', + color: [76, 0, 0, 255], + }, + '8': { + value: 8, + name: 'Segment 8', + color: [153, 0, 0, 255], + }, + '9': { + value: 9, + name: 'Segment 9', + color: [255, 0, 0, 255], + }, + '10': { + value: 10, + name: 'Segment 10', + color: [0, 76, 76, 255], + }, + '11': { + value: 11, + name: 'Segment 11', + color: [0, 153, 153, 255], + }, + '12': { + value: 12, + name: 'Segment 12', + color: [0, 255, 255, 255], + }, + '13': { + value: 13, + name: 'Segment 13', + color: [0, 0, 76, 255], + }, + '14': { + value: 14, + name: 'Segment 14', + color: [0, 0, 153, 255], + }, +}; + +const migrateManifest = (manifestString: string) => { + const inputManifest = JSON.parse(manifestString); + + if (inputManifest.version === '2.1.0') { + inputManifest.tools.paint.activeSegmentGroupID = + inputManifest.tools.paint.activeLabelmapID; + delete inputManifest.tools.paint.activeLabelmapID; + + const order = Object.keys(LABELMAP_PALETTE_2_1_0).map((key) => Number(key)); + inputManifest.labelMaps = inputManifest.labelMaps.map( + (labelMap: any, index: number) => ({ + id: labelMap.id, + path: labelMap.path, + metadata: { + parentImage: labelMap.parent, + name: makeDefaultSegmentGroupName('My Image', index), + segments: { + order, + byValue: LABELMAP_PALETTE_2_1_0, + }, + }, + }) + ); + + inputManifest.version = '3.0.0'; + } + + return inputManifest; +}; + const resolveUriSource: ImportHandler = async (dataSource, { extra, done }) => { const { uriSrc } = dataSource; @@ -234,9 +345,9 @@ const restoreStateFile: ImportHandler = async ( throw new Error('State file does not have exactly 1 manifest'); } - const manifest = ManifestSchema.parse( - JSON.parse(await manifests[0].file.text()) - ); + const manifestString = await manifests[0].file.text(); + const migrated = migrateManifest(manifestString); + const manifest = ManifestSchema.parse(migrated); // We restore the view first, so that the appropriate watchers are triggered // in the views as the data is loaded diff --git a/tests/fixtures/toolsProstate.2-1-0.volview.json b/tests/fixtures/toolsProstate.2-1-0.volview.json new file mode 100644 index 000000000..00d8a36dd --- /dev/null +++ b/tests/fixtures/toolsProstate.2-1-0.volview.json @@ -0,0 +1,353 @@ +{ + "version": "2.1.0", + "datasets": [ + { + "id": "1.3.6.1.4.1.14519.5.2.1.7311.5101.206828891270520544417996275680.5tse2d1254.538438420111018.1D000000SN0D000000S0D000000S0D000000S0D970296SN0D241922", + "path": "data/1.3.6.1.4.1.14519.5.2.1.7311.5101.206828891270520544417996275680.5tse2d1254.538438420111018.1D000000SN0D000000S0D000000S0D000000S0D970296SN0D241922/", + "type": "dicom" + } + ], + "remoteFiles": { + "1.3.6.1.4.1.14519.5.2.1.7311.5101.206828891270520544417996275680.5tse2d1254.538438420111018.1D000000SN0D000000S0D000000S0D000000S0D970296SN0D241922": [ + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-19.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-18.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-17.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-16.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-15.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-14.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-13.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-12.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-11.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-10.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-09.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-08.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-07.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-06.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-05.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-04.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-03.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-02.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + }, + { + "archiveSrc": { "path": "MRI-PROSTATEx-0004/1-01.dcm" }, + "parent": { + "uriSrc": { + "uri": "https://data.kitware.com/api/v1/item/63527c7311dab8142820a338/download", + "name": "MRI-PROSTATEx-0004.zip" + } + } + } + ] + }, + "labelMaps": [], + "tools": { + "crosshairs": { "position": [0, 0, 0] }, + "paint": { + "activeLabelmapID": null, + "brushSize": 4, + "brushValue": 1, + "labelmapOpacity": 1 + }, + "crop": { + "10": { + "Sagittal": [-0.5, 383.5], + "Coronal": [-0.5, 383.5], + "Axial": [-0.5, 18.5] + } + }, + "current": "Ruler", + "polygons": { + "tools": [ + { + "imageID": "1.3.6.1.4.1.14519.5.2.1.7311.5101.206828891270520544417996275680.5tse2d1254.538438420111018.1D000000SN0D000000S0D000000S0D000000S0D970296SN0D241922", + "frameOfReference": { + "planeNormal": [-1, 1.9901095602651587e-10, -4.961898903617917e-11], + "planeOrigin": [ + -21.913253803870006, -85.14991853516106, 12.839991645554424 + ] + }, + "slice": 192, + "placing": false, + "color": "#ffffff", + "strokeWidth": 1, + "name": "Polygon", + "points": [ + [-21.913253793455468, -26.035952658844444, 40.043030426568826], + [-21.913253795879754, -43.50275785498896, 18.845472462845592], + [-21.91325378913905, -13.2182627464645, 4.460887366907217] + ], + "id": "15", + "label": "9", + "labelName": "white", + "movePoint": null + } + ], + "labels": { + "7": { "labelName": "red", "color": "red", "strokeWidth": 1 }, + "8": { "labelName": "green", "color": "#00ff00", "strokeWidth": 1 }, + "9": { "labelName": "white", "color": "#ffffff", "strokeWidth": 1 } + } + }, + "rectangles": { + "tools": [ + { + "imageID": "1.3.6.1.4.1.14519.5.2.1.7311.5101.206828891270520544417996275680.5tse2d1254.538438420111018.1D000000SN0D000000S0D000000S0D000000S0D970296SN0D241922", + "frameOfReference": { + "planeNormal": [-1, 1.9901095602651587e-10, -4.961898903617917e-11], + "planeOrigin": [ + -21.913253803870006, -85.14991853516106, 12.839991645554424 + ] + }, + "slice": 192, + "placing": false, + "color": "red", + "strokeWidth": 1, + "name": "Rectangle", + "firstPoint": [ + -21.913253791326838, -11.01393722969522, 57.39330673229667 + ], + "secondPoint": [ + -21.913253781334035, 26.49523475520717, 6.444097013912222 + ], + "id": "11", + "fillColor": "transparent", + "label": "6", + "labelName": "lesion" + } + ], + "labels": { + "4": { + "labelName": "artifact", + "color": "#888888", + "strokeWidth": 1, + "fillColor": "#ffbf0040" + }, + "5": { + "labelName": "innocuous", + "color": "#00ff00", + "strokeWidth": 1, + "fillColor": "#00ff0020" + }, + "6": { + "labelName": "lesion", + "color": "red", + "strokeWidth": 1, + "fillColor": "transparent" + } + } + }, + "rulers": { + "tools": [ + { + "imageID": "1.3.6.1.4.1.14519.5.2.1.7311.5101.206828891270520544417996275680.5tse2d1254.538438420111018.1D000000SN0D000000S0D000000S0D000000S0D970296SN0D241922", + "frameOfReference": { + "planeNormal": [ + 2.0510340000000028e-10, 0.9702957270336713, -0.24192189256080032 + ], + "planeOrigin": [ + -117.91325378418009, 7.998471279176442, -10.38451004504583 + ] + }, + "slice": 192, + "placing": false, + "color": "#ffffff", + "strokeWidth": 1, + "name": "Ruler", + "firstPoint": [ + -62.811914987613584, 20.39704096666574, 39.34343757084859 + ], + "secondPoint": [ + -23.028116033464777, 17.60723051718813, 28.15411889634344 + ], + "id": "20", + "label": "3", + "labelName": "white" + } + ], + "labels": { + "1": { "labelName": "red", "color": "red", "strokeWidth": 1 }, + "2": { "labelName": "green", "color": "#00ff00", "strokeWidth": 1 }, + "3": { "labelName": "white", "color": "#ffffff", "strokeWidth": 1 } + } + } + }, + "layout": { + "name": "Quad View", + "direction": "H", + "items": [ + { "direction": "V", "items": ["Coronal", "3D"] }, + { "direction": "V", "items": ["Sagittal", "Axial"] } + ] + }, + "views": [ + { + "id": "Coronal", + "type": "2D", + "props": { "viewDirection": "Right", "viewUp": "Superior" }, + "config": {} + }, + { + "id": "Sagittal", + "type": "2D", + "props": { "viewDirection": "Posterior", "viewUp": "Superior" }, + "config": {} + }, + { + "id": "Axial", + "type": "2D", + "props": { "viewDirection": "Superior", "viewUp": "Anterior" }, + "config": {} + }, + { + "id": "3D", + "type": "3D", + "props": { "viewDirection": "Posterior", "viewUp": "Superior" }, + "config": {} + } + ], + "parentToLayers": [], + "primarySelection": "1.3.6.1.4.1.14519.5.2.1.7311.5101.206828891270520544417996275680.5tse2d1254.538438420111018.1D000000SN0D000000S0D000000S0D000000S0D970296SN0D241922" +} diff --git a/tests/fixtures/toolsProstate.volview.json b/tests/fixtures/toolsProstate.3-0-0.volview.json similarity index 100% rename from tests/fixtures/toolsProstate.volview.json rename to tests/fixtures/toolsProstate.3-0-0.volview.json diff --git a/tests/specs/remote-manifest.e2e.ts b/tests/specs/remote-manifest.e2e.ts index c167cec0f..e76f282ee 100644 --- a/tests/specs/remote-manifest.e2e.ts +++ b/tests/specs/remote-manifest.e2e.ts @@ -3,6 +3,7 @@ import * as fs from 'fs'; import { cleanuptotal } from 'wdio-cleanuptotal-service'; import { TEMP_DIR } from '../../wdio.shared.conf'; import { volViewPage } from '../pageobjects/volview.page'; +import { downloadFile } from './utils'; async function writeManifestToFile(manifest: any, fileName: string) { const filePath = path.join(TEMP_DIR, fileName); @@ -32,15 +33,10 @@ describe('VolView loading of remoteManifest.json', () => { it('should load relative URI with no name property', async () => { const dicom = '1-001.dcm'; - const dicomPath = path.join(TEMP_DIR, dicom); - if (!fs.existsSync(dicomPath)) { - const response = await fetch( - 'https://data.kitware.com/api/v1/file/655d42a694ef39bf0a4a8bb3/download' - ); - const data = await response.arrayBuffer(); - const buffer = Buffer.from(data); - fs.writeFileSync(dicomPath, buffer); - } + await downloadFile( + 'https://data.kitware.com/api/v1/file/655d42a694ef39bf0a4a8bb3/download', + dicom + ); const manifest = { resources: [{ url: `/tmp/${dicom}` }], diff --git a/tests/specs/state-manifest.e2e.ts b/tests/specs/state-manifest.e2e.ts index 9242018ef..10606b033 100644 --- a/tests/specs/state-manifest.e2e.ts +++ b/tests/specs/state-manifest.e2e.ts @@ -4,32 +4,54 @@ import { cleanuptotal } from 'wdio-cleanuptotal-service'; import JSZip from 'jszip'; import { FIXTURES, TEMP_DIR } from '../../wdio.shared.conf'; import { volViewPage } from '../pageobjects/volview.page'; -// import toolsProstateJSON from '/fixtures/toolsProstate.volview.json'; +import { downloadFile } from './utils'; + +async function writeManifestToZip(manifestPath: string, fileName: string) { + const filePath = path.join(TEMP_DIR, fileName); + const manifest = fs.readFileSync(manifestPath); + + const zip = new JSZip(); + zip.file('manifest.json', manifest); + const data = await zip.generateAsync({ type: 'nodebuffer' }); + + await fs.promises.writeFile(filePath, data); + cleanuptotal.addCleanup(async () => { + fs.unlinkSync(filePath); + }); + + return filePath; +} + +async function openVolViewPage(fileName: string) { + const urlParams = `?urls=[tmp/${fileName}]`; + await volViewPage.open(urlParams); + await volViewPage.waitForViews(); + + const notifications = await volViewPage.getNotificationsCount(); + expect(notifications).toEqual(0); +} describe('State file manifest.json code', () => { it('has no errors loading version 3.0.0 manifest.json file ', async () => { - // write json to zip file in temp dir - const fileName = 'asdf.zip'; - const filePath = path.join(TEMP_DIR, fileName); - - const manifest = fs.readFileSync( - path.join(FIXTURES, 'toolsProstate.volview.json') + const manifestPath = path.join( + FIXTURES, + 'toolsProstate.3-0-0.volview.json' ); + const fileName = 'temp-session.volview.zip'; + await writeManifestToZip(manifestPath, fileName); + await openVolViewPage(fileName); + }); - const zip = new JSZip(); - zip.file('manifest.json', manifest); - const data = await zip.generateAsync({ type: 'nodebuffer' }); - - await fs.promises.writeFile(filePath, data); - cleanuptotal.addCleanup(async () => { - fs.unlinkSync(filePath); - }); + // Dev test + // http://localhost:8080/?&urls=[http://localhost:9999/session.volview-2-1-0-labelmap-tools.zip] + it('has no errors loading version 2.1.0 manifest.json file ', async () => { + const FILE_NAME = 'session.volview-2-1-0-labelmap-tools.zip'; - const urlParams = `?urls=[tmp/${fileName}]`; - await volViewPage.open(urlParams); - await volViewPage.waitForViews(); + await downloadFile( + 'https://data.kitware.com/api/v1/file/6566acb6c5a2b36857ad1786/download', + FILE_NAME + ); - const notifications = await volViewPage.getNotificationsCount(); - expect(notifications).toEqual(0); + await openVolViewPage(FILE_NAME); }); }); diff --git a/tests/specs/utils.ts b/tests/specs/utils.ts new file mode 100644 index 000000000..f48424353 --- /dev/null +++ b/tests/specs/utils.ts @@ -0,0 +1,15 @@ +import * as path from 'path'; +import * as fs from 'fs'; +import { TEMP_DIR } from '../../wdio.shared.conf'; + +// File is not automatically deleted +export const downloadFile = async (url: string, fileName: string) => { + const savePath = path.join(TEMP_DIR, fileName); + if (!fs.existsSync(savePath)) { + const response = await fetch(url); + const data = await response.arrayBuffer(); + const buffer = Buffer.from(data); + fs.writeFileSync(savePath, buffer); + } + return savePath; +}; diff --git a/wdio.shared.conf.ts b/wdio.shared.conf.ts index aa9e06b52..b2f8bb26e 100644 --- a/wdio.shared.conf.ts +++ b/wdio.shared.conf.ts @@ -11,7 +11,7 @@ export const DOWNLOAD_TIMEOUT = Number(process.env.DOWNLOAD_TIMEOUT ?? 5000); const ROOT = projectRoot(); // TEMP_DIR is also downloads directory -const TMP = '.tmp/'; +const TMP = '.tmp'; export const TEMP_DIR = path.resolve(ROOT, TMP); const FIXTURES_DIR = 'tests/fixtures/'; export const FIXTURES = path.resolve(ROOT, FIXTURES_DIR); @@ -59,7 +59,7 @@ export const config: Options.Testrunner = { mount: '/', path: './dist', }, - { mount: '/tmp', path: `./${TMP}` }, + { mount: `/tmp`, path: `./${TMP}` }, ], port: TEST_PORT, },