parseDICOM.py 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. import DICOMLib
  2. import sys
  3. import json
  4. import numpy
  5. import zipfile
  6. import shutil
  7. import os
  8. def main(configFile=None):
  9. print('Imported!')
  10. with open(configFile) as f:
  11. config=json.load(f)
  12. config.update(connectDB(config))
  13. parseData(config,getMeanHeartDose)
  14. def connectDB(setup):
  15. nixSuite=os.path.join(os.path.expanduser('~'),'software','src','nixsuite')
  16. sys.path.append(os.path.join(nixSuite,'wrapper'))
  17. import nixWrapper
  18. nixWrapper.loadLibrary('labkeyInterface')
  19. import labkeyInterface
  20. import labkeyDatabaseBrowser
  21. import labkeyFileBrowser
  22. nixWrapper.loadLibrary('orthancInterface')
  23. import orthancInterface
  24. import orthancDatabaseBrowser
  25. import orthancFileBrowser
  26. importlib.reload(orthancFileBrowser)
  27. net=labkeyInterface.labkeyInterface()
  28. qfile='{}.json'.format(setup['server'])
  29. fconfig=os.path.join(os.path.expanduser('~'),'.labkey',qfile)
  30. net.init(fconfig)
  31. net.getCSRF()
  32. onet=orthancInterface.orthancInterface()
  33. onet.init(fconfig)
  34. return {"db":labkeyDatabaseBrowser.labkeyDB(net),
  35. "fb":labkeyFileBrowser.labkeyFileBrowser(net),
  36. "odb":orthancDatabaseBrowser.orthancDB(onet),
  37. "ofb":orthancFileBrowser.orthancFileBrowser(onet)}
  38. #explicit template
  39. def _updateRow(config,r):
  40. print(r)
  41. return False
  42. def parseData(config,updateRow=_updateRow):
  43. #iterates over data appliying updateRow function to every row
  44. #updateRow is an implementation of a generic function
  45. #with arguments
  46. # def updateRow(config,r)
  47. #returning True if row needs to be updated on the server
  48. #and False otherwise
  49. #update values are stored in the r dict
  50. db=config['db']
  51. qFilter=config.get('qFilter',[])
  52. debug=config.get('debug',False)
  53. #get dataset
  54. ds=db.selectRows(config['project'],config['schema'],config['query'],qFilter)
  55. rows=ds['rows']
  56. #shorten list in debug mode
  57. if debug:
  58. rows=rows[0:3]
  59. for r in rows:
  60. #this could be a generic function either as parameter of config or an argument to parseData
  61. update=updateRow(config,r)
  62. #print(r)
  63. if not update:
  64. continue
  65. db.modifyRows('update',config['project'],config['schema'],config['query'],[r])
  66. def getMeanHeartDose(config,r):
  67. #calculates mean heart dose
  68. #stores it as doseHeart to row r
  69. #return True if r needs to be updated on the server
  70. #and False if r is unchanged
  71. sid=r['orthancStudyId']
  72. if not sid:
  73. print('No study for {}'.format(r['ParticipantId']))
  74. return False
  75. doseHeart=r['doseHeart']
  76. if doseHeart:
  77. #no need to update
  78. return False
  79. #outDir=getDicomZip(config,sid)
  80. outDir=getDicomInstances(config,sid)
  81. nodes=loadVolumes(outDir)
  82. msg=checkNodes(config,nodes)
  83. if len(msg)>0:
  84. r['comments']=msg
  85. else:
  86. r['doseHeart']=getMeanDose(nodes,'Heart')
  87. clearDir(outDir)
  88. #needs updating
  89. return True
  90. def loadVolumes(dataDir):
  91. nodeNames=[]
  92. with DICOMLib.DICOMUtils.TemporaryDICOMDatabase() as db:
  93. DICOMLib.DICOMUtils.importDicom(dataDir, db)
  94. patientUIDs = db.patients()
  95. for patientUID in patientUIDs:
  96. print(patientUID)
  97. nodeNames.extend(DICOMLib.DICOMUtils.loadPatientByUID(patientUID))
  98. #print(nodes)
  99. nodes=[slicer.util.getNode(pattern=n) for n in nodeNames]
  100. volumeNodes=[n for n in nodes if n.GetClassName()=='vtkMRMLScalarVolumeNode']
  101. doseNodes=[n for n in volumeNodes if n.GetName().find('RTDOSE')>-1]
  102. segmentationNodes=[n for n in nodes if n.GetClassName()=='vtkMRMLSegmentationNode']
  103. nv=len(volumeNodes)
  104. ns=len(segmentationNodes)
  105. nd=len(doseNodes)
  106. print(f'vol:{nv} seg:{ns} dose: {nd}')
  107. return {'vol':volumeNodes,'dose':doseNodes,'seg':segmentationNodes}
  108. def checkNodes(config,nodes):
  109. msg=''
  110. nD=len(nodes['dose'])
  111. if nD>1:
  112. msg+=f'DOSE[{nD}]'
  113. nS=len(nodes['seg'])
  114. if nS>1:
  115. if len(msg)>0:
  116. msg+='/'
  117. msg+=f'SEG[{nS}]'
  118. return msg
  119. def getMeanDose(nodes,target):
  120. segNode=nodes['seg'][0]
  121. seg=segNode.GetSegmentation()
  122. segmentIds=seg.GetSegmentIDs()
  123. #[seg.GetNthSegment(i) for i in range(seg.GetNumberOfSegments())]
  124. targetSegmentIds=[s for s in segmentIds if seg.GetSegment(s).GetName()==target]
  125. #labelmapVolumeNode = slicer.mrmlScene.AddNewNodeByClass("vtkMRMLLabelMapVolumeNode")
  126. #export=slicer.vtkSlicerSegmentationsModuleLogic.ExportSegmentsToLabelmapNode
  127. #export(segNode, targetSegmentIds, labelmapVolumeNode, nodes['dose'][0])
  128. #nodes.update({'labelMap':labelmapVolumeNode})
  129. doseNode=nodes['dose'][0]
  130. doseArray = slicer.util.arrayFromVolume(doseNode)
  131. segmentArray = slicer.util.arrayFromSegmentBinaryLabelmap(segNode, targetSegmentIds[0], doseNode)
  132. doseVoxels = doseArray[segmentArray != 0]
  133. print(numpy.mean(doseVoxels))
  134. #add a float() to avoid JSON complaining about float32 converion
  135. return float(numpy.mean(doseVoxels))
  136. def getDicomInstances(config,sid):
  137. odb=config['odb']
  138. ofb=config['ofb']
  139. sd=odb.getStudyData(sid)
  140. series=sd['Series']
  141. instances=[]
  142. for s in series:
  143. sed=odb.getSeriesData(s)
  144. instances.extend(sed['Instances'])
  145. #download instances one by one
  146. baseDir=config.get('baseDir',os.path.join(os.path.expanduser('~'),'temp'))
  147. outDir=os.path.join(baseDir,sid)
  148. clearDir(outDir)
  149. os.mkdir(outDir)
  150. for oid in instances:
  151. local=os.path.join(outDir,f'{oid}.dcm')
  152. ofb.getInstance(oid,local)
  153. return outDir
  154. def getDicomZip(config,sid):
  155. ofb=config['ofb']
  156. baseDir=config.get('baseDir',os.path.join(os.path.expanduser('~'),'temp'))
  157. fname=f'{sid}.zip'
  158. path=os.path.join(baseDir,fname)
  159. if not os.path.isfile(path):
  160. ofb.getZip('studies',sid,path,'archive')
  161. print(f'Using {path}')
  162. #unzip path
  163. outDir=extractZip(config,path)
  164. os.remove(path)
  165. return outDir
  166. def extractZip(config,fname):
  167. #flattens the zip files in the baseDir/bname directory
  168. #where bname is the basename of the file without the .zip suffix
  169. fzip=zipfile.ZipFile(fname)
  170. names=fzip.namelist()
  171. bname=os.path.basename(fname)
  172. bname=bname.replace('.zip','')
  173. baseDir=config['baseDir']
  174. outDir=os.path.join(baseDir,bname)
  175. #clean
  176. clearDir(outDir)
  177. os.mkdir(outDir)
  178. outnames=[os.path.join(outDir,f'out{i:03d}.dcm') for i in range(len(names))]
  179. #extracts and renames (avoids *nix and win directory separator confusion)
  180. for (member,out) in zip(names,outnames):
  181. with fzip.open(member) as zf, open(out, 'wb') as f:
  182. shutil.copyfileobj(zf, f)
  183. return outDir
  184. def clearDir(outDir):
  185. if os.path.isdir(outDir):
  186. shutil.rmtree(outDir)
  187. if __name__=='__main__':
  188. try:
  189. main(sys.argv[1])
  190. except IndexError:
  191. main()
  192. print('Succesful completion')
  193. quit()