123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458 |
- import os
- import unittest
- import vtk, qt, ctk, slicer
- from slicer.ScriptedLoadableModule import *
- import logging
- import slicerNetwork
- import json
- import loadPatient
- #
- # dataExplorer
- #
- class dataExplorer(ScriptedLoadableModule):
- """Uses ScriptedLoadableModule base class, available at:
- https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
- """
- def __init__(self, parent):
- ScriptedLoadableModule.__init__(self, parent)
- self.parent.title = "dataExplorer" # TODO make this more human readable by adding spaces
- self.parent.categories = ["Examples"]
- self.parent.dependencies = []
- self.parent.contributors = ["Andrej Studen (University of Ljubljana)"] # replace with "Firstname Lastname (Organization)"
- self.parent.helpText = """
- This is an example of scripted loadable module bundled in an extension.
- It performs a simple thresholding on the input volume and optionally captures a screenshot.
- """
- self.parent.helpText += self.getDefaultModuleDocumentationLink()
- self.parent.acknowledgementText = """
- This extension developed within Medical Physics research programe of ARRS
- """ # replace with organization, grant and thanks.
- #
- # dataExplorerWidget
- #
- class dataExplorerWidget(ScriptedLoadableModuleWidget):
- """Uses ScriptedLoadableModuleWidget base class, available at:
- https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
- """
- def setup(self):
- ScriptedLoadableModuleWidget.setup(self)
- self.loadPatientLogic=loadPatient.loadPatientLogic(self)
- # Instantiate and connect widgets ...
- #
- # Parameters Area
- #
- try:
- self.startDir=os.path.join(os.environ['HOME'],"temp/crt")
- except:
- fhome=os.environ['HOMEDRIVE']+os.environ['HOMEPATH']
- self.startDir=os.path.join(fhome,"temp")
- self.sNet=slicerNetwork.labkeyURIHandler()
- self.loadPatientLogic.setURIHandler(self.sNet)
- configCollapsibleButton = ctk.ctkCollapsibleButton()
- configCollapsibleButton.text = "Configuration"
- self.layout.addWidget(configCollapsibleButton)
- # Layout within the dummy collapsible button
- configFormLayout = qt.QFormLayout(configCollapsibleButton)
- self.configDir='/afs/f9.ijs.si/home/studen/software/src/embraceParse'
- self.loadConfigButton=qt.QPushButton("Load Config")
- self.loadConfigButton.connect('clicked(bool)',self.onLoadConfigButtonClicked)
- configFormLayout.addRow("Configuration:",self.loadConfigButton)
- ## connection sub-area
- connectionCollapsibleButton = ctk.ctkCollapsibleButton()
- connectionCollapsibleButton.text = "Connection"
- self.layout.addWidget(connectionCollapsibleButton)
- # Layout within the dummy collapsible button
- connectionFormLayout = qt.QFormLayout(connectionCollapsibleButton)
- #
- # input volume selector
- #
- self.server=qt.QLineEdit("https://merlin.fmf.uni-lj.si")
- connectionFormLayout.addRow("Server: ", self.server)
- self.userCertButton=qt.QPushButton("Load")
- self.userCertButton.toolTip="Load user certificate (crt)"
- self.userCertButton.connect('clicked(bool)',self.onUserCertButtonClicked)
- connectionFormLayout.addRow("User certificate:",self.userCertButton)
- self.userKeyButton=qt.QPushButton("Load")
- self.userKeyButton.toolTip="Load user key (key)"
- self.userKeyButton.connect('clicked(bool)',self.onUserKeyButtonClicked)
- connectionFormLayout.addRow("User key:",self.userKeyButton)
- self.userCAButton=qt.QPushButton("Load")
- self.userCAButton.toolTip="Load CA certificate (crt)"
- self.userCAButton.connect('clicked(bool)',self.onUserCAButtonClicked)
- connectionFormLayout.addRow("User certificate:",self.userCAButton)
- self.authName=qt.QLineEdit("email")
- #self.authName.textChanged.connect(self.updateAuthName)
- connectionFormLayout.addRow("Labkey username: ", self.authName)
- self.authPass=qt.QLineEdit("email")
- self.authPass.setEchoMode(qt.QLineEdit.Password)
- #self.authPass.textChanged.connect(self.updateAuthPass)
- connectionFormLayout.addRow("Labkey password: ", self.authPass)
- self.initButton=qt.QPushButton("Init")
- self.initButton.toolTip="Init connection"
- self.initButton.connect('clicked(bool)',self.onInitButtonClicked)
- connectionFormLayout.addRow("Connection:",self.initButton)
- # Add vertical spacer
- self.layout.addStretch(1)
- datasetCollapsibleButton = ctk.ctkCollapsibleButton()
- datasetCollapsibleButton.text = "Dataset"
- self.layout.addWidget(datasetCollapsibleButton)
- # Layout within the dummy collapsible button
- datasetFormLayout = qt.QFormLayout(datasetCollapsibleButton)
- self.datasetName=qt.QLineEdit("datasetName")
- datasetFormLayout.addRow("Dataset:",self.datasetName)
- self.datasetProject=qt.QLineEdit("datasetProject")
- datasetFormLayout.addRow("Project:",self.datasetProject)
- self.datasetButton=qt.QPushButton("Load")
- self.datasetButton.connect('clicked(bool)',self.onDatasetLoadButtonClicked)
- datasetFormLayout.addRow("Project:",self.datasetButton)
- dataCollapsibleButton = ctk.ctkCollapsibleButton()
- dataCollapsibleButton.text = "Data"
- self.layout.addWidget(dataCollapsibleButton)
- # Layout within the dummy collapsible button
- dataFormLayout = qt.QVBoxLayout(dataCollapsibleButton)
- self.data=qt.QTableWidget(3,3)
- dataFormLayout.addWidget(self.data)
- patientsCollapsibleButton = ctk.ctkCollapsibleButton()
- patientsCollapsibleButton.text = "Patients"
- self.layout.addWidget(patientsCollapsibleButton)
- # Layout within the dummy collapsible button
- self.patientsFormLayout = qt.QVBoxLayout(patientsCollapsibleButton)
- self.signalMapper=qt.QSignalMapper()
- self.signalMapper.connect(self.signalMapper, qt.SIGNAL("mapped(const QString &)"), self.onPatientButtonClicked)
- def cleanup(self):
- pass
- def onLoadConfigButtonClicked(self):
- filename=qt.QFileDialog.getOpenFileName(None,'Open configuration file (JSON)',
- self.configDir, '*.json')
- with open(filename,'r') as f:
- dt=json.load(f)
- if dt.has_key('host'):
- self.server.setText(dt['host'])
- if dt.has_key('dataset'):
- self.datasetName.setText(dt['dataset'])
- if dt.has_key('project'):
- self.datasetProject.setText(dt['project'])
- if dt.has_key('SSL'):
- if dt['SSL'].has_key('user'):
- self.userCertButton.setText(dt['SSL']['user'])
- if dt['SSL'].has_key('key'):
- self.userKeyButton.setText(dt['SSL']['key'])
- if dt['SSL'].has_key('keyPwd'):
- self.keyPwd=dt['SSL']['keyPwd']
- if dt['SSL'].has_key('ca'):
- self.userCAButton.setText(dt['SSL']['ca'])
- if dt.has_key('labkey'):
- if dt['labkey'].has_key('user'):
- self.authName.setText(dt['labkey']['user'])
- if dt['labkey'].has_key('password'):
- self.authPass.setText(dt['labkey']['password'])
- self.loadConfigButton.setText(os.path.basename(filename))
- def onUserCertButtonClicked(self):
- filename=qt.QFileDialog.getOpenFileName(None,'Open user certificate',
- self.startDir, '*.crt')
- #pwd=qt.QInputDialog.getText(None,'Certificate password',
- # 'Enter certificate password',qt.QLineEdit.Password)
- if not(filename) :
- print "No file selected"
- return
- f=qt.QFile(filename)
- if not (f.open(qt.QIODevice.ReadOnly)) :
- print "Could not open file"
- return
- certList=qt.QSslCertificate.fromPath(filename)
- if len(certList) < 1:
- print "Troubles parsing {0}".format(filename)
- return
- cert=qt.QSslCertificate(f)
- print "cert.isNull()={0}".format(cert.isNull())
- self.userCertButton.setText(filename)
- self.authName.setText(cert.subjectInfo("emailAddress"))
- def onUserKeyButtonClicked(self):
- filename=qt.QFileDialog.getOpenFileName(None,'Open private key',
- self.startDir, '*.key')
- if not (filename) :
- print "No file selected"
- return
- f=qt.QFile(filename)
- if not (f.open(qt.QIODevice.ReadOnly)) :
- print "Could not open file"
- return
- self.keyPwd=qt.QInputDialog.getText(None,'Private key password',
- 'Enter key password',qt.QLineEdit.Password)
- key=qt.QSslKey(f,qt.QSsl.Rsa,qt.QSsl.Pem,qt.QSsl.PrivateKey,
- str(self.keyPwd))
- self.userKeyButton.setText(filename)
- def onUserCAButtonClicked(self):
- filename=qt.QFileDialog.getOpenFileName(None,'Open authority certificate',
- self.startDir, '*.crt')
- if not(filename) :
- print "No file selected"
- return
- f=qt.QFile(filename)
- if not (f.open(qt.QIODevice.ReadOnly)) :
- print "Could not open file"
- return
- certList=qt.QSslCertificate.fromPath(filename)
- if len(certList) < 1:
- print "Troubles parsing {0}".format(filename)
- return
- #self.logic.caCert=qt.QSslCertificate(f)#certList[0]
- self.userCAButton.setText(filename)
- def onInitButtonClicked(self):
- self.sNet.configureSSL(
- self.userCertButton.text,
- self.userKeyButton.text,
- self.keyPwd,
- self.userCAButton.text
- )
- self.sNet.hostname=self.server.text
- self.sNet.auth_name=self.authName.text
- self.sNet.auth_pass=self.authPass.text
- self.sNet.initRemote()
- self.initButton.setText("Active")
- def onDatasetLoadButtonClicked(self):
- ds=self.sNet.loadDataset(self.datasetProject.text,self.datasetName.text)
- #loaded a JSON object -> convert to suitable display
- columns=ds['columnModel']
- self.data.setColumnCount(len(columns))
- self.data.setRowCount(len(ds['rows']))
- columnNames=[f['header'] for f in columns]
- self.data.setHorizontalHeaderLabels(columnNames)
- columnID=[f['dataIndex'] for f in columns]
- insertRowID=0
- for row in ds['rows']:
- insertColumnID=0
- for c in columnID:
- item=qt.QTableWidgetItem(str(row[c]))
- self.data.setItem(insertRowID,insertColumnID,item)
- insertColumnID+=1
- insertRowID+=1
- #clear patient list
- while self.patientsFormLayout.count():
- child=self.patientsFormLayout.takeAt(0)
- if child.widget():
- self.signalMapper.removeMapping(child.widget())
- child.widget().deleteLater()
- #populate patient list
- patientList=[f['EMBRACE_ID'] for f in ds['rows']]
- patientSet=set(patientList)
- for p in patientSet:
- patientButton=qt.QPushButton(p)
- patientButton.connect(patientButton, qt.SIGNAL("clicked()"),
- self.signalMapper, qt.SLOT("map()"))
- self.patientsFormLayout.addWidget(patientButton)
- self.signalMapper.setMapping(patientButton,p)
- def onPatientButtonClicked(self, label):
- print "onPatientButtonClicked() for {}".format(label)
- self.loadPatientLogic.load(label)
- #
- # dataExplorerLogic
- #
- class dataExplorerLogic(ScriptedLoadableModuleLogic):
- """This class should implement all the actual
- computation done by your module. The interface
- should be such that other python code can import
- this class and make use of the functionality without
- requiring an instance of the Widget.
- Uses ScriptedLoadableModuleLogic base class, available at:
- https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
- """
- def hasImageData(self,volumeNode):
- """This is an example logic method that
- returns true if the passed in volume
- node has valid image data
- """
- if not volumeNode:
- logging.debug('hasImageData failed: no volume node')
- return False
- if volumeNode.GetImageData() is None:
- logging.debug('hasImageData failed: no image data in volume node')
- return False
- return True
- def isValidInputOutputData(self, inputVolumeNode, outputVolumeNode):
- """Validates if the output is not the same as input
- """
- if not inputVolumeNode:
- logging.debug('isValidInputOutputData failed: no input volume node defined')
- return False
- if not outputVolumeNode:
- logging.debug('isValidInputOutputData failed: no output volume node defined')
- return False
- if inputVolumeNode.GetID()==outputVolumeNode.GetID():
- logging.debug('isValidInputOutputData failed: input and output volume is the same. Create a new volume for output to avoid this error.')
- return False
- return True
- def takeScreenshot(self,name,description,type=-1):
- # show the message even if not taking a screen shot
- slicer.util.delayDisplay('Take screenshot: '+description+'.\nResult is available in the Annotations module.', 3000)
- lm = slicer.app.layoutManager()
- # switch on the type to get the requested window
- widget = 0
- if type == slicer.qMRMLScreenShotDialog.FullLayout:
- # full layout
- widget = lm.viewport()
- elif type == slicer.qMRMLScreenShotDialog.ThreeD:
- # just the 3D window
- widget = lm.threeDWidget(0).threeDView()
- elif type == slicer.qMRMLScreenShotDialog.Red:
- # red slice window
- widget = lm.sliceWidget("Red")
- elif type == slicer.qMRMLScreenShotDialog.Yellow:
- # yellow slice window
- widget = lm.sliceWidget("Yellow")
- elif type == slicer.qMRMLScreenShotDialog.Green:
- # green slice window
- widget = lm.sliceWidget("Green")
- else:
- # default to using the full window
- widget = slicer.util.mainWindow()
- # reset the type so that the node is set correctly
- type = slicer.qMRMLScreenShotDialog.FullLayout
- # grab and convert to vtk image data
- qimage = ctk.ctkWidgetsUtils.grabWidget(widget)
- imageData = vtk.vtkImageData()
- slicer.qMRMLUtils().qImageToVtkImageData(qimage,imageData)
- annotationLogic = slicer.modules.annotations.logic()
- annotationLogic.CreateSnapShot(name, description, type, 1, imageData)
- def run(self, inputVolume, outputVolume, imageThreshold, enableScreenshots=0):
- """
- Run the actual algorithm
- """
- if not self.isValidInputOutputData(inputVolume, outputVolume):
- slicer.util.errorDisplay('Input volume is the same as output volume. Choose a different output volume.')
- return False
- logging.info('Processing started')
- # Compute the thresholded output volume using the Threshold Scalar Volume CLI module
- cliParams = {'InputVolume': inputVolume.GetID(), 'OutputVolume': outputVolume.GetID(), 'ThresholdValue' : imageThreshold, 'ThresholdType' : 'Above'}
- cliNode = slicer.cli.run(slicer.modules.thresholdscalarvolume, None, cliParams, wait_for_completion=True)
- # Capture screenshot
- if enableScreenshots:
- self.takeScreenshot('dataExplorerTest-Start','MyScreenshot',-1)
- logging.info('Processing completed')
- return True
- class dataExplorerTest(ScriptedLoadableModuleTest):
- """
- This is the test case for your scripted module.
- Uses ScriptedLoadableModuleTest base class, available at:
- https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
- """
- def setUp(self):
- """ Do whatever is needed to reset the state - typically a scene clear will be enough.
- """
- slicer.mrmlScene.Clear(0)
- def runTest(self):
- """Run as few or as many tests as needed here.
- """
- self.setUp()
- self.test_dataExplorer1()
- def test_dataExplorer1(self):
- """ Ideally you should have several levels of tests. At the lowest level
- tests should exercise the functionality of the logic with different inputs
- (both valid and invalid). At higher levels your tests should emulate the
- way the user would interact with your code and confirm that it still works
- the way you intended.
- One of the most important features of the tests is that it should alert other
- developers when their changes will have an impact on the behavior of your
- module. For example, if a developer removes a feature that you depend on,
- your test should break so they know that the feature is needed.
- """
- self.delayDisplay("Starting the test")
- #
- # first, get some data
- #
- import urllib
- downloads = (
- ('http://slicer.kitware.com/midas3/download?items=5767', 'FA.nrrd', slicer.util.loadVolume),
- )
- for url,name,loader in downloads:
- filePath = slicer.app.temporaryPath + '/' + name
- if not os.path.exists(filePath) or os.stat(filePath).st_size == 0:
- logging.info('Requesting download %s from %s...\n' % (name, url))
- urllib.urlretrieve(url, filePath)
- if loader:
- logging.info('Loading %s...' % (name,))
- loader(filePath)
- self.delayDisplay('Finished with download and loading')
- volumeNode = slicer.util.getNode(pattern="FA")
- logic = dataExplorerLogic()
- self.assertIsNotNone( logic.hasImageData(volumeNode) )
- self.delayDisplay('Test passed!')
|