dataExplorer.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. import os
  2. import unittest
  3. import vtk, qt, ctk, slicer
  4. from slicer.ScriptedLoadableModule import *
  5. import logging
  6. import slicerNetwork
  7. import json
  8. import loadPatient
  9. #
  10. # dataExplorer
  11. #
  12. class dataExplorer(ScriptedLoadableModule):
  13. """Uses ScriptedLoadableModule base class, available at:
  14. https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
  15. """
  16. def __init__(self, parent):
  17. ScriptedLoadableModule.__init__(self, parent)
  18. self.parent.title = "dataExplorer" # TODO make this more human readable by adding spaces
  19. self.parent.categories = ["Examples"]
  20. self.parent.dependencies = []
  21. self.parent.contributors = ["Andrej Studen (University of Ljubljana)"] # replace with "Firstname Lastname (Organization)"
  22. self.parent.helpText = """
  23. This is an example of scripted loadable module bundled in an extension.
  24. It performs a simple thresholding on the input volume and optionally captures a screenshot.
  25. """
  26. self.parent.helpText += self.getDefaultModuleDocumentationLink()
  27. self.parent.acknowledgementText = """
  28. This extension developed within Medical Physics research programe of ARRS
  29. """ # replace with organization, grant and thanks.
  30. #
  31. # dataExplorerWidget
  32. #
  33. class dataExplorerWidget(ScriptedLoadableModuleWidget):
  34. """Uses ScriptedLoadableModuleWidget base class, available at:
  35. https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
  36. """
  37. def setup(self):
  38. ScriptedLoadableModuleWidget.setup(self)
  39. self.loadPatientLogic=loadPatient.loadPatientLogic(self)
  40. # Instantiate and connect widgets ...
  41. #
  42. # Parameters Area
  43. #
  44. try:
  45. self.startDir=os.path.join(os.environ['HOME'],"temp/crt")
  46. except:
  47. fhome=os.environ['HOMEDRIVE']+os.environ['HOMEPATH']
  48. self.startDir=os.path.join(fhome,"temp")
  49. self.sNet=slicerNetwork.labkeyURIHandler()
  50. self.loadPatientLogic.setURIHandler(self.sNet)
  51. configCollapsibleButton = ctk.ctkCollapsibleButton()
  52. configCollapsibleButton.text = "Configuration"
  53. self.layout.addWidget(configCollapsibleButton)
  54. # Layout within the dummy collapsible button
  55. configFormLayout = qt.QFormLayout(configCollapsibleButton)
  56. self.configDir='/afs/f9.ijs.si/home/studen/software/src/embraceParse'
  57. self.loadConfigButton=qt.QPushButton("Load Config")
  58. self.loadConfigButton.connect('clicked(bool)',self.onLoadConfigButtonClicked)
  59. configFormLayout.addRow("Configuration:",self.loadConfigButton)
  60. ## connection sub-area
  61. connectionCollapsibleButton = ctk.ctkCollapsibleButton()
  62. connectionCollapsibleButton.text = "Connection"
  63. self.layout.addWidget(connectionCollapsibleButton)
  64. # Layout within the dummy collapsible button
  65. connectionFormLayout = qt.QFormLayout(connectionCollapsibleButton)
  66. #
  67. # input volume selector
  68. #
  69. self.server=qt.QLineEdit("https://merlin.fmf.uni-lj.si")
  70. connectionFormLayout.addRow("Server: ", self.server)
  71. self.userCertButton=qt.QPushButton("Load")
  72. self.userCertButton.toolTip="Load user certificate (crt)"
  73. self.userCertButton.connect('clicked(bool)',self.onUserCertButtonClicked)
  74. connectionFormLayout.addRow("User certificate:",self.userCertButton)
  75. self.userKeyButton=qt.QPushButton("Load")
  76. self.userKeyButton.toolTip="Load user key (key)"
  77. self.userKeyButton.connect('clicked(bool)',self.onUserKeyButtonClicked)
  78. connectionFormLayout.addRow("User key:",self.userKeyButton)
  79. self.userCAButton=qt.QPushButton("Load")
  80. self.userCAButton.toolTip="Load CA certificate (crt)"
  81. self.userCAButton.connect('clicked(bool)',self.onUserCAButtonClicked)
  82. connectionFormLayout.addRow("User certificate:",self.userCAButton)
  83. self.authName=qt.QLineEdit("email")
  84. #self.authName.textChanged.connect(self.updateAuthName)
  85. connectionFormLayout.addRow("Labkey username: ", self.authName)
  86. self.authPass=qt.QLineEdit("email")
  87. self.authPass.setEchoMode(qt.QLineEdit.Password)
  88. #self.authPass.textChanged.connect(self.updateAuthPass)
  89. connectionFormLayout.addRow("Labkey password: ", self.authPass)
  90. self.initButton=qt.QPushButton("Init")
  91. self.initButton.toolTip="Init connection"
  92. self.initButton.connect('clicked(bool)',self.onInitButtonClicked)
  93. connectionFormLayout.addRow("Connection:",self.initButton)
  94. # Add vertical spacer
  95. self.layout.addStretch(1)
  96. datasetCollapsibleButton = ctk.ctkCollapsibleButton()
  97. datasetCollapsibleButton.text = "Dataset"
  98. self.layout.addWidget(datasetCollapsibleButton)
  99. # Layout within the dummy collapsible button
  100. datasetFormLayout = qt.QFormLayout(datasetCollapsibleButton)
  101. self.datasetName=qt.QLineEdit("datasetName")
  102. datasetFormLayout.addRow("Dataset:",self.datasetName)
  103. self.datasetProject=qt.QLineEdit("datasetProject")
  104. datasetFormLayout.addRow("Project:",self.datasetProject)
  105. self.datasetButton=qt.QPushButton("Load")
  106. self.datasetButton.connect('clicked(bool)',self.onDatasetLoadButtonClicked)
  107. datasetFormLayout.addRow("Project:",self.datasetButton)
  108. dataCollapsibleButton = ctk.ctkCollapsibleButton()
  109. dataCollapsibleButton.text = "Data"
  110. self.layout.addWidget(dataCollapsibleButton)
  111. # Layout within the dummy collapsible button
  112. dataFormLayout = qt.QVBoxLayout(dataCollapsibleButton)
  113. self.data=qt.QTableWidget(3,3)
  114. dataFormLayout.addWidget(self.data)
  115. patientsCollapsibleButton = ctk.ctkCollapsibleButton()
  116. patientsCollapsibleButton.text = "Patients"
  117. self.layout.addWidget(patientsCollapsibleButton)
  118. # Layout within the dummy collapsible button
  119. self.patientsFormLayout = qt.QVBoxLayout(patientsCollapsibleButton)
  120. self.signalMapper=qt.QSignalMapper()
  121. self.signalMapper.connect(self.signalMapper, qt.SIGNAL("mapped(const QString &)"), self.onPatientButtonClicked)
  122. def cleanup(self):
  123. pass
  124. def onLoadConfigButtonClicked(self):
  125. filename=qt.QFileDialog.getOpenFileName(None,'Open configuration file (JSON)',
  126. self.configDir, '*.json')
  127. with open(filename,'r') as f:
  128. dt=json.load(f)
  129. if dt.has_key('host'):
  130. self.server.setText(dt['host'])
  131. if dt.has_key('dataset'):
  132. self.datasetName.setText(dt['dataset'])
  133. if dt.has_key('project'):
  134. self.datasetProject.setText(dt['project'])
  135. if dt.has_key('SSL'):
  136. if dt['SSL'].has_key('user'):
  137. self.userCertButton.setText(dt['SSL']['user'])
  138. if dt['SSL'].has_key('key'):
  139. self.userKeyButton.setText(dt['SSL']['key'])
  140. if dt['SSL'].has_key('keyPwd'):
  141. self.keyPwd=dt['SSL']['keyPwd']
  142. if dt['SSL'].has_key('ca'):
  143. self.userCAButton.setText(dt['SSL']['ca'])
  144. if dt.has_key('labkey'):
  145. if dt['labkey'].has_key('user'):
  146. self.authName.setText(dt['labkey']['user'])
  147. if dt['labkey'].has_key('password'):
  148. self.authPass.setText(dt['labkey']['password'])
  149. self.loadConfigButton.setText(os.path.basename(filename))
  150. def onUserCertButtonClicked(self):
  151. filename=qt.QFileDialog.getOpenFileName(None,'Open user certificate',
  152. self.startDir, '*.crt')
  153. #pwd=qt.QInputDialog.getText(None,'Certificate password',
  154. # 'Enter certificate password',qt.QLineEdit.Password)
  155. if not(filename) :
  156. print "No file selected"
  157. return
  158. f=qt.QFile(filename)
  159. if not (f.open(qt.QIODevice.ReadOnly)) :
  160. print "Could not open file"
  161. return
  162. certList=qt.QSslCertificate.fromPath(filename)
  163. if len(certList) < 1:
  164. print "Troubles parsing {0}".format(filename)
  165. return
  166. cert=qt.QSslCertificate(f)
  167. print "cert.isNull()={0}".format(cert.isNull())
  168. self.userCertButton.setText(filename)
  169. self.authName.setText(cert.subjectInfo("emailAddress"))
  170. def onUserKeyButtonClicked(self):
  171. filename=qt.QFileDialog.getOpenFileName(None,'Open private key',
  172. self.startDir, '*.key')
  173. if not (filename) :
  174. print "No file selected"
  175. return
  176. f=qt.QFile(filename)
  177. if not (f.open(qt.QIODevice.ReadOnly)) :
  178. print "Could not open file"
  179. return
  180. self.keyPwd=qt.QInputDialog.getText(None,'Private key password',
  181. 'Enter key password',qt.QLineEdit.Password)
  182. key=qt.QSslKey(f,qt.QSsl.Rsa,qt.QSsl.Pem,qt.QSsl.PrivateKey,
  183. str(self.keyPwd))
  184. self.userKeyButton.setText(filename)
  185. def onUserCAButtonClicked(self):
  186. filename=qt.QFileDialog.getOpenFileName(None,'Open authority certificate',
  187. self.startDir, '*.crt')
  188. if not(filename) :
  189. print "No file selected"
  190. return
  191. f=qt.QFile(filename)
  192. if not (f.open(qt.QIODevice.ReadOnly)) :
  193. print "Could not open file"
  194. return
  195. certList=qt.QSslCertificate.fromPath(filename)
  196. if len(certList) < 1:
  197. print "Troubles parsing {0}".format(filename)
  198. return
  199. #self.logic.caCert=qt.QSslCertificate(f)#certList[0]
  200. self.userCAButton.setText(filename)
  201. def onInitButtonClicked(self):
  202. self.sNet.configureSSL(
  203. self.userCertButton.text,
  204. self.userKeyButton.text,
  205. self.keyPwd,
  206. self.userCAButton.text
  207. )
  208. self.sNet.hostname=self.server.text
  209. self.sNet.auth_name=self.authName.text
  210. self.sNet.auth_pass=self.authPass.text
  211. self.sNet.initRemote()
  212. self.initButton.setText("Active")
  213. def onDatasetLoadButtonClicked(self):
  214. ds=self.sNet.loadDataset(self.datasetProject.text,self.datasetName.text)
  215. #loaded a JSON object -> convert to suitable display
  216. columns=ds['columnModel']
  217. self.data.setColumnCount(len(columns))
  218. self.data.setRowCount(len(ds['rows']))
  219. columnNames=[f['header'] for f in columns]
  220. self.data.setHorizontalHeaderLabels(columnNames)
  221. columnID=[f['dataIndex'] for f in columns]
  222. insertRowID=0
  223. for row in ds['rows']:
  224. insertColumnID=0
  225. for c in columnID:
  226. item=qt.QTableWidgetItem(str(row[c]))
  227. self.data.setItem(insertRowID,insertColumnID,item)
  228. insertColumnID+=1
  229. insertRowID+=1
  230. #clear patient list
  231. while self.patientsFormLayout.count():
  232. child=self.patientsFormLayout.takeAt(0)
  233. if child.widget():
  234. self.signalMapper.removeMapping(child.widget())
  235. child.widget().deleteLater()
  236. #populate patient list
  237. patientList=[f['EMBRACE_ID'] for f in ds['rows']]
  238. patientSet=set(patientList)
  239. for p in patientSet:
  240. patientButton=qt.QPushButton(p)
  241. patientButton.connect(patientButton, qt.SIGNAL("clicked()"),
  242. self.signalMapper, qt.SLOT("map()"))
  243. self.patientsFormLayout.addWidget(patientButton)
  244. self.signalMapper.setMapping(patientButton,p)
  245. def onPatientButtonClicked(self, label):
  246. print "onPatientButtonClicked() for {}".format(label)
  247. self.loadPatientLogic.load(label)
  248. #
  249. # dataExplorerLogic
  250. #
  251. class dataExplorerLogic(ScriptedLoadableModuleLogic):
  252. """This class should implement all the actual
  253. computation done by your module. The interface
  254. should be such that other python code can import
  255. this class and make use of the functionality without
  256. requiring an instance of the Widget.
  257. Uses ScriptedLoadableModuleLogic base class, available at:
  258. https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
  259. """
  260. def hasImageData(self,volumeNode):
  261. """This is an example logic method that
  262. returns true if the passed in volume
  263. node has valid image data
  264. """
  265. if not volumeNode:
  266. logging.debug('hasImageData failed: no volume node')
  267. return False
  268. if volumeNode.GetImageData() is None:
  269. logging.debug('hasImageData failed: no image data in volume node')
  270. return False
  271. return True
  272. def isValidInputOutputData(self, inputVolumeNode, outputVolumeNode):
  273. """Validates if the output is not the same as input
  274. """
  275. if not inputVolumeNode:
  276. logging.debug('isValidInputOutputData failed: no input volume node defined')
  277. return False
  278. if not outputVolumeNode:
  279. logging.debug('isValidInputOutputData failed: no output volume node defined')
  280. return False
  281. if inputVolumeNode.GetID()==outputVolumeNode.GetID():
  282. logging.debug('isValidInputOutputData failed: input and output volume is the same. Create a new volume for output to avoid this error.')
  283. return False
  284. return True
  285. def takeScreenshot(self,name,description,type=-1):
  286. # show the message even if not taking a screen shot
  287. slicer.util.delayDisplay('Take screenshot: '+description+'.\nResult is available in the Annotations module.', 3000)
  288. lm = slicer.app.layoutManager()
  289. # switch on the type to get the requested window
  290. widget = 0
  291. if type == slicer.qMRMLScreenShotDialog.FullLayout:
  292. # full layout
  293. widget = lm.viewport()
  294. elif type == slicer.qMRMLScreenShotDialog.ThreeD:
  295. # just the 3D window
  296. widget = lm.threeDWidget(0).threeDView()
  297. elif type == slicer.qMRMLScreenShotDialog.Red:
  298. # red slice window
  299. widget = lm.sliceWidget("Red")
  300. elif type == slicer.qMRMLScreenShotDialog.Yellow:
  301. # yellow slice window
  302. widget = lm.sliceWidget("Yellow")
  303. elif type == slicer.qMRMLScreenShotDialog.Green:
  304. # green slice window
  305. widget = lm.sliceWidget("Green")
  306. else:
  307. # default to using the full window
  308. widget = slicer.util.mainWindow()
  309. # reset the type so that the node is set correctly
  310. type = slicer.qMRMLScreenShotDialog.FullLayout
  311. # grab and convert to vtk image data
  312. qimage = ctk.ctkWidgetsUtils.grabWidget(widget)
  313. imageData = vtk.vtkImageData()
  314. slicer.qMRMLUtils().qImageToVtkImageData(qimage,imageData)
  315. annotationLogic = slicer.modules.annotations.logic()
  316. annotationLogic.CreateSnapShot(name, description, type, 1, imageData)
  317. def run(self, inputVolume, outputVolume, imageThreshold, enableScreenshots=0):
  318. """
  319. Run the actual algorithm
  320. """
  321. if not self.isValidInputOutputData(inputVolume, outputVolume):
  322. slicer.util.errorDisplay('Input volume is the same as output volume. Choose a different output volume.')
  323. return False
  324. logging.info('Processing started')
  325. # Compute the thresholded output volume using the Threshold Scalar Volume CLI module
  326. cliParams = {'InputVolume': inputVolume.GetID(), 'OutputVolume': outputVolume.GetID(), 'ThresholdValue' : imageThreshold, 'ThresholdType' : 'Above'}
  327. cliNode = slicer.cli.run(slicer.modules.thresholdscalarvolume, None, cliParams, wait_for_completion=True)
  328. # Capture screenshot
  329. if enableScreenshots:
  330. self.takeScreenshot('dataExplorerTest-Start','MyScreenshot',-1)
  331. logging.info('Processing completed')
  332. return True
  333. class dataExplorerTest(ScriptedLoadableModuleTest):
  334. """
  335. This is the test case for your scripted module.
  336. Uses ScriptedLoadableModuleTest base class, available at:
  337. https://github.com/Slicer/Slicer/blob/master/Base/Python/slicer/ScriptedLoadableModule.py
  338. """
  339. def setUp(self):
  340. """ Do whatever is needed to reset the state - typically a scene clear will be enough.
  341. """
  342. slicer.mrmlScene.Clear(0)
  343. def runTest(self):
  344. """Run as few or as many tests as needed here.
  345. """
  346. self.setUp()
  347. self.test_dataExplorer1()
  348. def test_dataExplorer1(self):
  349. """ Ideally you should have several levels of tests. At the lowest level
  350. tests should exercise the functionality of the logic with different inputs
  351. (both valid and invalid). At higher levels your tests should emulate the
  352. way the user would interact with your code and confirm that it still works
  353. the way you intended.
  354. One of the most important features of the tests is that it should alert other
  355. developers when their changes will have an impact on the behavior of your
  356. module. For example, if a developer removes a feature that you depend on,
  357. your test should break so they know that the feature is needed.
  358. """
  359. self.delayDisplay("Starting the test")
  360. #
  361. # first, get some data
  362. #
  363. import urllib
  364. downloads = (
  365. ('http://slicer.kitware.com/midas3/download?items=5767', 'FA.nrrd', slicer.util.loadVolume),
  366. )
  367. for url,name,loader in downloads:
  368. filePath = slicer.app.temporaryPath + '/' + name
  369. if not os.path.exists(filePath) or os.stat(filePath).st_size == 0:
  370. logging.info('Requesting download %s from %s...\n' % (name, url))
  371. urllib.urlretrieve(url, filePath)
  372. if loader:
  373. logging.info('Loading %s...' % (name,))
  374. loader(filePath)
  375. self.delayDisplay('Finished with download and loading')
  376. volumeNode = slicer.util.getNode(pattern="FA")
  377. logic = dataExplorerLogic()
  378. self.assertIsNotNone( logic.hasImageData(volumeNode) )
  379. self.delayDisplay('Test passed!')