source: sasview/src/sas/sascalc/dataloader/loader.py @ 7ad12e9f

ticket-1094-headless
Last change on this file since 7ad12e9f was 8c9e65c, checked in by GitHub <noreply@…>, 6 years ago

py37 support for sascalc. Refs #888 an #1233.

  • Property mode set to 100644
File size: 15.8 KB
RevLine 
[4749514]1"""
2    File handler to support different file extensions.
[270c882b]3    Uses reflectometer registry utility.
[f06d7fc]4
[4749514]5    The default readers are found in the 'readers' sub-module
6    and registered by default at initialization time.
[f06d7fc]7
[4749514]8    To add a new default reader, one must register it in
9    the register_readers method found in readers/__init__.py.
[f06d7fc]10
[4749514]11    A utility method (find_plugins) is available to inspect
12    a directory (for instance, a user plug-in directory) and
13    look for new readers/writers.
14"""
15#####################################################################
[5d8f9b3]16# This software was developed by the University of Tennessee as part of the
17# Distributed Data Analysis of Neutron Scattering Experiments (DANSE)
18# project funded by the US National Science Foundation.
19# See the license text in license.txt
20# copyright 2008, University of Tennessee
[4749514]21######################################################################
22
23import os
24import sys
25import logging
26import time
27from zipfile import ZipFile
[574adc7]28
[b699768]29from sas.sascalc.data_util.registry import ExtensionRegistry
[574adc7]30
[4749514]31# Default readers are defined in the readers sub-module
[574adc7]32from . import readers
33from .loader_exceptions import NoKnownLoaderException, FileContentsException,\
[da8bb53]34    DefaultReaderException
[574adc7]35from .readers import ascii_reader
36from .readers import cansas_reader
37from .readers import cansas_reader_HDF5
[4749514]38
[463e7ffc]39logger = logging.getLogger(__name__)
[c155a16]40
[5d8f9b3]41
[4749514]42class Registry(ExtensionRegistry):
43    """
44    Registry class for file format extensions.
45    Readers and writers are supported.
46    """
47    def __init__(self):
48        super(Registry, self).__init__()
[f06d7fc]49
[270c882b]50        # Writers
[4749514]51        self.writers = {}
[f06d7fc]52
[270c882b]53        # List of wildcards
[4749514]54        self.wildcards = ['All (*.*)|*.*']
[f06d7fc]55
[270c882b]56        # Creation time, for testing
[4749514]57        self._created = time.time()
[f06d7fc]58
[4749514]59        # Register default readers
60        readers.read_associations(self)
61
62    def load(self, path, format=None):
63        """
64        Call the loader for the file type of path.
65
66        :param path: file path
67        :param format: explicit extension, to force the use
68            of a particular reader
69
[b9b612a]70        Defaults to the ascii (multi-column), cansas XML, and cansas NeXuS
71        readers if no reader was registered for the file's extension.
[4749514]72        """
[dcb91cf]73        # Gets set to a string if the file has an associated reader that fails
74        msg_from_reader = None
[4749514]75        try:
76            return super(Registry, self).load(path, format=format)
[dc8d1c2]77        #except Exception: raise  # for debugging, don't use fallback loader
[371b9e2]78        except NoKnownLoaderException as nkl_e:
[dcb91cf]79            pass  # Try the ASCII reader
[3ece5dd]80        except FileContentsException as fc_exc:
[dcb91cf]81            # File has an associated reader but it failed.
82            # Save the error message to display later, but try the 3 default loaders
83            msg_from_reader = fc_exc.message
[3ece5dd]84        except Exception:
[371b9e2]85            pass
[3eb7bf2]86
[dcb91cf]87        # File has no associated reader, or the associated reader failed.
88        # Try the ASCII reader
[b9b612a]89        try:
90            ascii_loader = ascii_reader.Reader()
91            return ascii_loader.read(path)
[4a8d55c]92        except NoKnownLoaderException:
93            pass  # Try the Cansas XML reader
[da8bb53]94        except DefaultReaderException:
[3ece5dd]95            pass  # Loader specific error to try the cansas XML reader
96        except FileContentsException as e:
[dcb91cf]97            if msg_from_reader is None:
98                raise RuntimeError(e.message)
[3eb7bf2]99
100        # ASCII reader failed - try CanSAS xML reader
[b9b612a]101        try:
102            cansas_loader = cansas_reader.Reader()
103            return cansas_loader.read(path)
[4a8d55c]104        except NoKnownLoaderException:
105            pass  # Try the NXcanSAS reader
[da8bb53]106        except DefaultReaderException:
[3eb7bf2]107            pass  # Loader specific error to try the NXcanSAS reader
[3ece5dd]108        except FileContentsException as e:
[dcb91cf]109            if msg_from_reader is None:
110                raise RuntimeError(e.message)
111        except Exception:
[da8bb53]112            pass
[3eb7bf2]113
114        # CanSAS XML reader failed - try NXcanSAS reader
[b9b612a]115        try:
116            cansas_nexus_loader = cansas_reader_HDF5.Reader()
117            return cansas_nexus_loader.read(path)
[8dec7e7]118        except DefaultReaderException as e:
119            logging.error("No default loader can load the data")
[b9b612a]120            # No known reader available. Give up and throw an error
[dcb91cf]121            if msg_from_reader is None:
122                msg = "\nUnknown data format: {}.\nThe file is not a ".format(path)
123                msg += "known format that can be loaded by SasView.\n"
124                raise NoKnownLoaderException(msg)
125            else:
126                # Associated reader and default readers all failed.
127                # Show error message from associated reader
128                raise RuntimeError(msg_from_reader)
[8dec7e7]129        except FileContentsException as e:
[dcb91cf]130            err_msg = msg_from_reader if msg_from_reader is not None else e.message
131            raise RuntimeError(err_msg)
[f06d7fc]132
[4749514]133    def find_plugins(self, dir):
134        """
135        Find readers in a given directory. This method
136        can be used to inspect user plug-in directories to
137        find new readers/writers.
[f06d7fc]138
[4749514]139        :param dir: directory to search into
140        :return: number of readers found
141        """
142        readers_found = 0
143        temp_path = os.path.abspath(dir)
144        if not os.path.isdir(temp_path):
145            temp_path = os.path.join(os.getcwd(), dir)
146        if not os.path.isdir(temp_path):
147            temp_path = os.path.join(os.path.dirname(__file__), dir)
148        if not os.path.isdir(temp_path):
149            temp_path = os.path.join(os.path.dirname(sys.path[0]), dir)
[f06d7fc]150
[4749514]151        dir = temp_path
152        # Check whether the directory exists
153        if not os.path.isdir(dir):
154            msg = "DataLoader couldn't locate DataLoader plugin folder."
155            msg += """ "%s" does not exist""" % dir
[c155a16]156            logger.warning(msg)
[4749514]157            return readers_found
[f06d7fc]158
[4749514]159        for item in os.listdir(dir):
160            full_path = os.path.join(dir, item)
161            if os.path.isfile(full_path):
[f06d7fc]162
[4749514]163                # Process python files
164                if item.endswith('.py'):
165                    toks = os.path.splitext(os.path.basename(item))
166                    try:
167                        sys.path.insert(0, os.path.abspath(dir))
168                        module = __import__(toks[0], globals(), locals())
169                        if self._identify_plugin(module):
170                            readers_found += 1
[e090ba90]171                    except Exception as exc:
[4749514]172                        msg = "Loader: Error importing "
[e090ba90]173                        msg += "%s\n  %s" % (item, exc)
[c155a16]174                        logger.error(msg)
[f06d7fc]175
[4749514]176                # Process zip files
177                elif item.endswith('.zip'):
178                    try:
179                        # Find the modules in the zip file
180                        zfile = ZipFile(item)
181                        nlist = zfile.namelist()
[f06d7fc]182
[4749514]183                        sys.path.insert(0, item)
184                        for mfile in nlist:
185                            try:
186                                # Change OS path to python path
187                                fullname = mfile.replace('/', '.')
188                                fullname = os.path.splitext(fullname)[0]
189                                module = __import__(fullname, globals(),
[f06d7fc]190                                                    locals(), [""])
[4749514]191                                if self._identify_plugin(module):
192                                    readers_found += 1
[e090ba90]193                            except Exception as exc:
[4749514]194                                msg = "Loader: Error importing"
[e090ba90]195                                msg += " %s\n  %s" % (mfile, exc)
[c155a16]196                                logger.error(msg)
[f06d7fc]197
[e090ba90]198                    except Exception as exc:
[4749514]199                        msg = "Loader: Error importing "
[e090ba90]200                        msg += " %s\n  %s" % (item, exc)
[c155a16]201                        logger.error(msg)
[f06d7fc]202
[4749514]203        return readers_found
[f06d7fc]204
[4749514]205    def associate_file_type(self, ext, module):
206        """
207        Look into a module to find whether it contains a
208        Reader class. If so, APPEND it to readers and (potentially)
209        to the list of writers for the given extension
[f06d7fc]210
[4749514]211        :param ext: file extension [string]
212        :param module: module object
213        """
214        reader_found = False
[f06d7fc]215
[4749514]216        if hasattr(module, "Reader"):
217            try:
218                # Find supported extensions
219                loader = module.Reader()
220                if ext not in self.loaders:
221                    self.loaders[ext] = []
222                # Append the new reader to the list
223                self.loaders[ext].append(loader.read)
224
225                reader_found = True
[f06d7fc]226
[4749514]227                # Keep track of wildcards
228                type_name = module.__name__
229                if hasattr(loader, 'type_name'):
230                    type_name = loader.type_name
[f06d7fc]231
[4749514]232                wcard = "%s files (*%s)|*%s" % (type_name, ext.lower(),
[f06d7fc]233                                                ext.lower())
[4749514]234                if wcard not in self.wildcards:
235                    self.wildcards.append(wcard)
[f06d7fc]236
[4749514]237                # Check whether writing is supported
238                if hasattr(loader, 'write'):
239                    if ext not in self.writers:
240                        self.writers[ext] = []
241                    # Append the new writer to the list
242                    self.writers[ext].append(loader.write)
[f06d7fc]243
[e090ba90]244            except Exception as exc:
[4749514]245                msg = "Loader: Error accessing"
[e090ba90]246                msg += " Reader in %s\n  %s" % (module.__name__, exc)
[c155a16]247                logger.error(msg)
[4749514]248        return reader_found
249
250    def associate_file_reader(self, ext, loader):
251        """
252        Append a reader object to readers
[f06d7fc]253
[4749514]254        :param ext: file extension [string]
255        :param module: reader object
256        """
257        reader_found = False
[f06d7fc]258
[4749514]259        try:
260            # Find supported extensions
261            if ext not in self.loaders:
262                self.loaders[ext] = []
263            # Append the new reader to the list
264            self.loaders[ext].append(loader.read)
265
266            reader_found = True
[f06d7fc]267
[4749514]268            # Keep track of wildcards
269            if hasattr(loader, 'type_name'):
270                type_name = loader.type_name
[f06d7fc]271
[4749514]272                wcard = "%s files (*%s)|*%s" % (type_name, ext.lower(),
273                                                ext.lower())
274                if wcard not in self.wildcards:
275                    self.wildcards.append(wcard)
[f06d7fc]276
[e090ba90]277        except Exception as exc:
[4749514]278            msg = "Loader: Error accessing Reader "
[e090ba90]279            msg += "in %s\n  %s" % (loader.__name__, exc)
[c155a16]280            logger.error(msg)
[4749514]281        return reader_found
282
283    def _identify_plugin(self, module):
284        """
[f06d7fc]285        Look into a module to find whether it contains a
[4749514]286        Reader class. If so, add it to readers and (potentially)
287        to the list of writers.
288        :param module: module object
[f06d7fc]289
[4749514]290        """
291        reader_found = False
[f06d7fc]292
[4749514]293        if hasattr(module, "Reader"):
294            try:
295                # Find supported extensions
296                loader = module.Reader()
297                for ext in loader.ext:
298                    if ext not in self.loaders:
299                        self.loaders[ext] = []
300                    # When finding a reader at run time,
301                    # treat this reader as the new default
302                    self.loaders[ext].insert(0, loader.read)
303
304                    reader_found = True
[f06d7fc]305
[4749514]306                    # Keep track of wildcards
307                    type_name = module.__name__
308                    if hasattr(loader, 'type_name'):
309                        type_name = loader.type_name
310                    wcard = "%s files (*%s)|*%s" % (type_name, ext.lower(),
[f06d7fc]311                                                    ext.lower())
[4749514]312                    if wcard not in self.wildcards:
313                        self.wildcards.append(wcard)
[f06d7fc]314
[4749514]315                # Check whether writing is supported
316                if hasattr(loader, 'write'):
317                    for ext in loader.ext:
318                        if ext not in self.writers:
319                            self.writers[ext] = []
320                        self.writers[ext].insert(0, loader.write)
[f06d7fc]321
[e090ba90]322            except Exception as exc:
[4749514]323                msg = "Loader: Error accessing Reader"
[e090ba90]324                msg += " in %s\n  %s" % (module.__name__, exc)
[c155a16]325                logger.error(msg)
[4749514]326        return reader_found
327
328    def lookup_writers(self, path):
329        """
330        :return: the loader associated with the file type of path.
331        :Raises ValueError: if file type is not known.
332        """
333        # Find matching extensions
334        extlist = [ext for ext in self.extensions() if path.endswith(ext)]
335        # Sort matching extensions by decreasing order of length
[dc8d1c2]336        extlist.sort(key=len)
[4749514]337        # Combine loaders for matching extensions into one big list
338        writers = []
339        for L in [self.writers[ext] for ext in extlist]:
340            writers.extend(L)
341        # Remove duplicates if they exist
342        if len(writers) != len(set(writers)):
343            result = []
344            for L in writers:
345                if L not in result:
346                    result.append(L)
347            writers = L
348        # Raise an error if there are no matching extensions
349        if len(writers) == 0:
[574adc7]350            raise ValueError("Unknown file type for " + path)
[4749514]351        # All done
352        return writers
353
354    def save(self, path, data, format=None):
355        """
356        Call the writer for the file type of path.
357
358        Raises ValueError if no writer is available.
359        Raises KeyError if format is not available.
360        May raise a writer-defined exception if writer fails.
361        """
362        if format is None:
363            writers = self.lookup_writers(path)
364        else:
365            writers = self.writers[format]
366        for fn in writers:
367            try:
368                return fn(path, data)
[b1ec23d]369            except Exception as exc:
370                msg = "Saving file {} using the {} writer failed.\n".format(
371                    path, type(fn).__name__)
372                msg += str(exc)
373                logger.exception(msg)  # give other loaders a chance to succeed
[4749514]374
[f06d7fc]375
[4749514]376class Loader(object):
377    """
378    Utility class to use the Registry as a singleton.
379    """
380    ## Registry instance
381    __registry = Registry()
[f06d7fc]382
[4749514]383    def associate_file_type(self, ext, module):
384        """
385        Look into a module to find whether it contains a
386        Reader class. If so, append it to readers and (potentially)
387        to the list of writers for the given extension
[f06d7fc]388
[4749514]389        :param ext: file extension [string]
390        :param module: module object
391        """
392        return self.__registry.associate_file_type(ext, module)
393
394    def associate_file_reader(self, ext, loader):
395        """
396        Append a reader object to readers
[f06d7fc]397
[4749514]398        :param ext: file extension [string]
399        :param module: reader object
400        """
401        return self.__registry.associate_file_reader(ext, loader)
402
403    def load(self, file, format=None):
404        """
405        Load a file
[f06d7fc]406
[4749514]407        :param file: file name (path)
408        :param format: specified format to use (optional)
409        :return: DataInfo object
410        """
411        return self.__registry.load(file, format)
[f06d7fc]412
[4749514]413    def save(self, file, data, format):
414        """
415        Save a DataInfo object to file
416        :param file: file name (path)
417        :param data: DataInfo object
[f06d7fc]418        :param format: format to write the data in
[4749514]419        """
420        return self.__registry.save(file, data, format)
[f06d7fc]421
[4749514]422    def _get_registry_creation_time(self):
423        """
424        Internal method used to test the uniqueness
425        of the registry object
426        """
427        return self.__registry._created
[f06d7fc]428
429    def find_plugins(self, directory):
[4749514]430        """
431        Find plugins in a given directory
[f06d7fc]432
[4749514]433        :param dir: directory to look into to find new readers/writers
434        """
[f06d7fc]435        return self.__registry.find_plugins(directory)
436
[4749514]437    def get_wildcards(self):
[f06d7fc]438        """
439        Return the list of wildcards
440        """
[4749514]441        return self.__registry.wildcards
Note: See TracBrowser for help on using the repository browser.