| 1 |  |  | import os | 
            
                                                                                                            
                            
            
                                    
            
            
                | 2 |  |  | from operator import itemgetter | 
            
                                                                                                            
                            
            
                                    
            
            
                | 3 |  |  | from niprov.dependencies import Dependencies | 
            
                                                                                                            
                            
            
                                    
            
            
                | 4 |  |  |  | 
            
                                                                                                            
                                                                
            
                                    
            
            
                | 5 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 6 |  |  | class JsonFile(object): | 
            
                                                        
            
                                    
            
            
                | 7 |  |  |     """Stores provenance in a local text file encoded as json. | 
            
                                                        
            
                                    
            
            
                | 8 |  |  |     """ | 
            
                                                        
            
                                    
            
            
                | 9 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 10 |  |  |     def __init__(self, dependencies=Dependencies()): | 
            
                                                        
            
                                    
            
            
                | 11 |  |  |         self.filesys = dependencies.getFilesystem() | 
            
                                                        
            
                                    
            
            
                | 12 |  |  |         self.json = dependencies.getSerializer() | 
            
                                                        
            
                                    
            
            
                | 13 |  |  |         self.factory = dependencies.getFileFactory() | 
            
                                                        
            
                                    
            
            
                | 14 |  |  |         self.pictureCache = dependencies.getPictureCache() | 
            
                                                        
            
                                    
            
            
                | 15 |  |  |         url = dependencies.getConfiguration().database_url | 
            
                                                        
            
                                    
            
            
                | 16 |  |  |         self.datafile = os.path.expanduser(url) | 
            
                                                        
            
                                    
            
            
                | 17 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 18 |  |  |     def serializeAndWrite(self, images): | 
            
                                                        
            
                                    
            
            
                | 19 |  |  |         jsonstr = self.json.serializeList(images) | 
            
                                                        
            
                                    
            
            
                | 20 |  |  |         self.filesys.write(self.datafile, jsonstr) | 
            
                                                        
            
                                    
            
            
                | 21 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 22 |  |  |     def add(self, image): | 
            
                                                        
            
                                    
            
            
                | 23 |  |  |         """Add the provenance for one file to storage. | 
            
                                                        
            
                                    
            
            
                | 24 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 25 |  |  |         Args: | 
            
                                                        
            
                                    
            
            
                | 26 |  |  |             image (:class:`.BaseFile`): Image file to store. | 
            
                                                        
            
                                    
            
            
                | 27 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 28 |  |  |         current = self.all() | 
            
                                                        
            
                                    
            
            
                | 29 |  |  |         current.append(image) | 
            
                                                        
            
                                    
            
            
                | 30 |  |  |         self.serializeAndWrite(current) | 
            
                                                        
            
                                    
            
            
                | 31 |  |  |         self.pictureCache.saveToDisk(for_=image) | 
            
                                                        
            
                                    
            
            
                | 32 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 33 |  |  |     def update(self, image): | 
            
                                                        
            
                                    
            
            
                | 34 |  |  |         """Save changed provenance for this file.. | 
            
                                                        
            
                                    
            
            
                | 35 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 36 |  |  |         Args: | 
            
                                                        
            
                                    
            
            
                | 37 |  |  |             image (:class:`.BaseFile`): Image file that has changed. | 
            
                                                        
            
                                    
            
            
                | 38 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 39 |  |  |         current = self.all() | 
            
                                                        
            
                                    
            
            
                | 40 |  |  |         for r in range(len(current)): | 
            
                                                        
            
                                    
            
            
                | 41 |  |  |             if current[r].location.toString() == image.location.toString(): | 
            
                                                        
            
                                    
            
            
                | 42 |  |  |                 current[r] = image | 
            
                                                        
            
                                    
            
            
                | 43 |  |  |         self.serializeAndWrite(current) | 
            
                                                        
            
                                    
            
            
                | 44 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 45 |  |  |     def all(self): | 
            
                                                        
            
                                    
            
            
                | 46 |  |  |         """Retrieve all known provenance from storage. | 
            
                                                        
            
                                    
            
            
                | 47 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 48 |  |  |         Returns: | 
            
                                                        
            
                                    
            
            
                | 49 |  |  |             list: List of provenance for known files. | 
            
                                                        
            
                                    
            
            
                | 50 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 51 |  |  |         try: | 
            
                                                        
            
                                    
            
            
                | 52 |  |  |             jsonstr = self.filesys.read(self.datafile) | 
            
                                                        
            
                                    
            
            
                | 53 |  |  |         except IOError: | 
            
                                                        
            
                                    
            
            
                | 54 |  |  |             return [] | 
            
                                                        
            
                                    
            
            
                | 55 |  |  |         return self.json.deserializeList(jsonstr) | 
            
                                                        
            
                                    
            
            
                | 56 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 57 |  |  |     def knowsByLocation(self, locationString): | 
            
                                                        
            
                                    
            
            
                | 58 |  |  |         """Whether the file at this path has provenance associated with it. | 
            
                                                        
            
                                    
            
            
                | 59 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 60 |  |  |         Returns: | 
            
                                                        
            
                                    
            
            
                | 61 |  |  |             bool: True if provenance is available for that path. | 
            
                                                        
            
                                    
            
            
                | 62 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 63 |  |  |         try: | 
            
                                                        
            
                                    
            
            
                | 64 |  |  |             self.byLocation(locationString) | 
            
                                                        
            
                                    
            
            
                | 65 |  |  |         except IndexError: | 
            
                                                        
            
                                    
            
            
                | 66 |  |  |             return False | 
            
                                                        
            
                                    
            
            
                | 67 |  |  |         return True | 
            
                                                        
            
                                    
            
            
                | 68 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 69 |  |  |     def knows(self, image): | 
            
                                                        
            
                                    
            
            
                | 70 |  |  |         """Whether this file has provenance associated with it. | 
            
                                                        
            
                                    
            
            
                | 71 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 72 |  |  |         Returns: | 
            
                                                        
            
                                    
            
            
                | 73 |  |  |             bool: True if provenance is available for this image. | 
            
                                                        
            
                                    
            
            
                | 74 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 75 |  |  |         try: | 
            
                                                        
            
                                    
            
            
                | 76 |  |  |             self.byLocation(image.path) | 
            
                                                        
            
                                    
            
            
                | 77 |  |  |         except IndexError: | 
            
                                                        
            
                                    
            
            
                | 78 |  |  |             return False | 
            
                                                        
            
                                    
            
            
                | 79 |  |  |         return True | 
            
                                                        
            
                                    
            
            
                | 80 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 81 |  |  |     def knowsSeries(self, image): | 
            
                                                        
            
                                    
            
            
                | 82 |  |  |         """Whether the series that this file is part of has provenance  | 
            
                                                        
            
                                    
            
            
                | 83 |  |  |         associated with it. | 
            
                                                        
            
                                    
            
            
                | 84 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 85 |  |  |         Args: | 
            
                                                        
            
                                    
            
            
                | 86 |  |  |             image (:class:`.BaseFile`): File for which the series is sought. | 
            
                                                        
            
                                    
            
            
                | 87 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 88 |  |  |         Returns: | 
            
                                                        
            
                                    
            
            
                | 89 |  |  |             bool: True if provenance is available for this series. | 
            
                                                        
            
                                    
            
            
                | 90 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 91 |  |  |         try: | 
            
                                                        
            
                                    
            
            
                | 92 |  |  |             self.getSeries(image) | 
            
                                                        
            
                                    
            
            
                | 93 |  |  |         except IndexError: | 
            
                                                        
            
                                    
            
            
                | 94 |  |  |             return False | 
            
                                                        
            
                                    
            
            
                | 95 |  |  |         return True | 
            
                                                        
            
                                    
            
            
                | 96 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 97 |  |  |     def byLocation(self, locationString): | 
            
                                                        
            
                                    
            
            
                | 98 |  |  |         """Get the provenance for a file at the given location.  | 
            
                                                        
            
                                    
            
            
                | 99 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 100 |  |  |         In the case of a dicom series, this returns the provenance for the  | 
            
                                                        
            
                                    
            
            
                | 101 |  |  |         series. | 
            
                                                        
            
                                    
            
            
                | 102 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 103 |  |  |         Args: | 
            
                                                        
            
                                    
            
            
                | 104 |  |  |             locationString (str): Location of the image file. | 
            
                                                        
            
                                    
            
            
                | 105 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 106 |  |  |         Returns: | 
            
                                                        
            
                                    
            
            
                | 107 |  |  |             dict: Provenance for one image file. | 
            
                                                        
            
                                    
            
            
                | 108 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 109 |  |  |         for image in self.all(): | 
            
                                                        
            
                                    
            
            
                | 110 |  |  |             if image.location.toString() == locationString: | 
            
                                                        
            
                                    
            
            
                | 111 |  |  |                 return image | 
            
                                                        
            
                                    
            
            
                | 112 |  |  |             elif 'filesInSeries' in image.provenance and ( | 
            
                                                        
            
                                    
            
            
                | 113 |  |  |                 locationString in image.provenance['filesInSeries']): | 
            
                                                        
            
                                    
            
            
                | 114 |  |  |                 return image | 
            
                                                        
            
                                    
            
            
                | 115 |  |  |         else: | 
            
                                                        
            
                                    
            
            
                | 116 |  |  |             raise IndexError('No file with that path known.') | 
            
                                                        
            
                                    
            
            
                | 117 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 118 |  |  |     def byLocations(self, listOfLocations): | 
            
                                                        
            
                                    
            
            
                | 119 |  |  |         return [f for f in self.all() if f.location.toString() in listOfLocations] | 
            
                                                        
            
                                    
            
            
                | 120 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 121 |  |  |     def bySubject(self, subject): | 
            
                                                        
            
                                    
            
            
                | 122 |  |  |         """Get the provenance for all files of a given participant.  | 
            
                                                        
            
                                    
            
            
                | 123 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 124 |  |  |         Args: | 
            
                                                        
            
                                    
            
            
                | 125 |  |  |             subject (str): The name or other ID string. | 
            
                                                        
            
                                    
            
            
                | 126 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 127 |  |  |         Returns: | 
            
                                                        
            
                                    
            
            
                | 128 |  |  |             list: List of provenance for known files imaging this subject. | 
            
                                                        
            
                                    
            
            
                | 129 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 130 |  |  |         all = self.all() | 
            
                                                        
            
                                    
            
            
                | 131 |  |  |         imagesWithSubject = [f for f in all if 'subject' in f.provenance] | 
            
                                                        
            
                                    
            
            
                | 132 |  |  |         return [f for f in imagesWithSubject if f.provenance['subject']==subject] | 
            
                                                        
            
                                    
            
            
                | 133 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 134 |  |  |     def getSeries(self, image): | 
            
                                                        
            
                                    
            
            
                | 135 |  |  |         """Get the object that carries provenance for the series that the image  | 
            
                                                        
            
                                    
            
            
                | 136 |  |  |         passed is in.  | 
            
                                                        
            
                                    
            
            
                | 137 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 138 |  |  |         Args: | 
            
                                                        
            
                                    
            
            
                | 139 |  |  |             image (:class:`.DicomFile`): File that is part of a series. | 
            
                                                        
            
                                    
            
            
                | 140 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 141 |  |  |         Returns: | 
            
                                                        
            
                                    
            
            
                | 142 |  |  |             :class:`.DicomFile`: Image object that caries provenance for the series. | 
            
                                                        
            
                                    
            
            
                | 143 |  |  |         """ | 
            
                                                        
            
                                    
            
            
                | 144 |  |  |         if image.getSeriesId() is None: | 
            
                                                        
            
                                    
            
            
                | 145 |  |  |             raise IndexError('Image has no series id.') | 
            
                                                        
            
                                    
            
            
                | 146 |  |  |         seriesId = image.getSeriesId() | 
            
                                                        
            
                                    
            
            
                | 147 |  |  |         for image in self.all(): | 
            
                                                        
            
                                    
            
            
                | 148 |  |  |             if 'seriesuid' in image.provenance and ( | 
            
                                                        
            
                                    
            
            
                | 149 |  |  |                 image.provenance['seriesuid'] == seriesId): | 
            
                                                        
            
                                    
            
            
                | 150 |  |  |                 return image | 
            
                                                        
            
                                    
            
            
                | 151 |  |  |         else: | 
            
                                                        
            
                                    
            
            
                | 152 |  |  |             raise IndexError('No provenance record for that series.') | 
            
                                                        
            
                                    
            
            
                | 153 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 154 |  |  |     def byApproval(self, approvalStatus): | 
            
                                                        
            
                                    
            
            
                | 155 |  |  |         matches = [] | 
            
                                                        
            
                                    
            
            
                | 156 |  |  |         for image in self.all(): | 
            
                                                        
            
                                    
            
            
                | 157 |  |  |             if 'approval' in image.provenance: | 
            
                                                        
            
                                    
            
            
                | 158 |  |  |                 if image.provenance['approval'] == approvalStatus: | 
            
                                                        
            
                                    
            
            
                | 159 |  |  |                     matches.append(image) | 
            
                                                        
            
                                    
            
            
                | 160 |  |  |         return matches | 
            
                                                        
            
                                    
            
            
                | 161 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 162 |  |  |     def updateApproval(self, fpath, approvalStatus): | 
            
                                                        
            
                                    
            
            
                | 163 |  |  |         img = self.byLocation(fpath) | 
            
                                                        
            
                                    
            
            
                | 164 |  |  |         img.provenance['approval'] = approvalStatus | 
            
                                                        
            
                                    
            
            
                | 165 |  |  |         self.update(img) | 
            
                                                        
            
                                    
            
            
                | 166 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 167 |  |  |     def latest(self, n=20): | 
            
                                                        
            
                                    
            
            
                | 168 |  |  |         def dateTimeAdded(img): | 
            
                                                        
            
                                    
            
            
                | 169 |  |  |             return img.provenance.get('added') | 
            
                                                        
            
                                    
            
            
                | 170 |  |  |         sortedImages = sorted(self.all(), key=dateTimeAdded, reverse=True) | 
            
                                                        
            
                                    
            
            
                | 171 |  |  |         return sortedImages[:n] | 
            
                                                        
            
                                    
            
            
                | 172 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 173 |  |  |     def statistics(self): | 
            
                                                        
            
                                    
            
            
                | 174 |  |  |         stats = {} | 
            
                                                        
            
                                    
            
            
                | 175 |  |  |         images = self.all() | 
            
                                                        
            
                                    
            
            
                | 176 |  |  |         stats['count'] = len(images) | 
            
                                                        
            
                                    
            
            
                | 177 |  |  |         sizes = [img.provenance['size'] for img in images if 'size' in img.provenance] | 
            
                                                        
            
                                    
            
            
                | 178 |  |  |         stats['totalsize'] = sum(sizes) | 
            
                                                        
            
                                    
            
            
                | 179 |  |  |         return stats | 
            
                                                        
            
                                    
            
            
                | 180 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 181 |  |  |     def byId(self, uid): | 
            
                                                        
            
                                    
            
            
                | 182 |  |  |         for image in self.all(): | 
            
                                                        
            
                                    
            
            
                | 183 |  |  |             if image.provenance['id'] == uid: | 
            
                                                        
            
                                    
            
            
                | 184 |  |  |                 return image | 
            
                                                        
            
                                    
            
            
                | 185 |  |  |         else: | 
            
                                                        
            
                                    
            
            
                | 186 |  |  |             raise IndexError('No file with that path known.') | 
            
                                                        
            
                                    
            
            
                | 187 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 188 |  |  |     def byParents(self, listOfParentLocations): | 
            
                                                        
            
                                    
            
            
                | 189 |  |  |         return [f for f in self.all() if set(listOfParentLocations).intersection( | 
            
                                                        
            
                                    
            
            
                | 190 |  |  |             f.provenance.get('parents',[]))] | 
            
                                                        
            
                                    
            
            
                | 191 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 192 |  |  |     def inquire(self, query): | 
            
                                                        
            
                                    
            
            
                | 193 |  |  |         field = query.getFields()[0] | 
            
                                                        
            
                                    
            
            
                | 194 |  |  |         matches = [] | 
            
                                                        
            
                                    
            
            
                | 195 |  |  |         for image in self.all(): | 
            
                                                        
            
                                    
            
            
                | 196 |  |  |             if field.name in image.provenance: | 
            
                                                        
            
                                    
            
            
                | 197 |  |  |                 if image.provenance[field.name] == field.value: | 
            
                                                        
            
                                    
            
            
                | 198 |  |  |                     matches.append(image) | 
            
                                                        
            
                                    
            
            
                | 199 |  |  |         return matches | 
            
                                                        
            
                                    
            
            
                | 200 |  |  |  | 
            
                                                        
            
                                    
            
            
                | 201 |  |  |  |