source: sasview/src/sas/sascalc/dataloader/readers/sesans_reader.py @ 2b310602

ESS_GUIESS_GUI_DocsESS_GUI_batch_fittingESS_GUI_bumps_abstractionESS_GUI_iss1116ESS_GUI_iss879ESS_GUI_iss959ESS_GUI_openclESS_GUI_orderingESS_GUI_sync_sascalccostrafo411magnetic_scattrelease-4.2.2ticket-1009ticket-1094-headlessticket-1242-2d-resolutionticket-1243ticket-1249ticket885unittest-saveload
Last change on this file since 2b310602 was 2b310602, checked in by Adam Washington <adam.washington@…>, 7 years ago

Update SESANS reader to SESANS format v1.0

  • Property mode set to 100644
File size: 4.7 KB
Line 
1"""
2    SESANS reader (based on ASCII reader)
3
4    Reader for .ses or .sesans file format
5
6    Jurrian Bakker
7"""
8import numpy as np
9import os
10from sas.sascalc.dataloader.data_info import Data1D
11
12# Check whether we have a converter available
13has_converter = True
14try:
15    from sas.sascalc.data_util.nxsunit import Converter
16except:
17    has_converter = False
18_ZERO = 1e-16
19
20
21class Reader:
22    """
23    Class to load sesans files (6 columns).
24    """
25    # File type
26    type_name = "SESANS"
27
28    # Wildcards
29    type = ["SESANS files (*.ses)|*.ses",
30            "SESANS files (*..sesans)|*.sesans"]
31    # List of allowed extensions
32    ext = ['.ses', '.SES', '.sesans', '.SESANS']
33
34    # Flag to bypass extension check
35    allow_all = True
36
37    def read(self, path):
38        """
39        Load data file
40
41        :param path: file path
42
43        :return: SESANSData1D object, or None
44
45        :raise RuntimeError: when the file can't be opened
46        :raise ValueError: when the length of the data vectors are inconsistent
47        """
48        if os.path.isfile(path):
49            basename = os.path.basename(path)
50            _, extension = os.path.splitext(basename)
51            if not (self.allow_all or extension.lower() in self.ext):
52                raise RuntimeError("{} has an unrecognized file extension".format(path))
53        else:
54            raise RunetimeError("{} is not a file".format(path))
55        with open(path, 'r') as input_f:
56            # Read in binary mode since GRASP frequently has no-ascii
57            # characters that brakes the open operation
58            line = input_f.readline()
59            params = {}
60            while not line.startswith("BEGIN_DATA"):
61                terms = line.split()
62                if len(terms) >= 2:
63                    params[terms[0]] = " ".join(terms[1:])
64                line = input_f.readline()
65            self.params = params
66            headers = input_f.readline().split()
67
68            data = np.loadtxt(input_f)
69            if data.size < 1:
70                raise RuntimeError("{} is empty".format(path))
71            x = data[:, headers.index("SpinEchoLength")]
72            dx = data[:, headers.index("SpinEchoLength_error")]
73            lam = data[:, headers.index("Wavelength")]
74            dlam = data[:, headers.index("Wavelength_error")]
75            y = data[:, headers.index("Depolarisation")]
76            dy = data[:, headers.index("Depolarisation_error")]
77
78            lam_unit = self._unit_fetch("Wavelength")
79            x, x_unit = self._unit_conversion(x, "A", self._unit_fetch("SpinEchoLength"))
80            dx, dx_unit = self._unit_conversion(
81                dx, lam_unit,
82                self._unit_fetch("SpinEchoLength"))
83            dlam, dlam_unit = self._unit_conversion(
84                dlam, lam_unit,
85                self._unit_fetch("Wavelength"))
86            y_unit = self._unit_fetch("Depolarisation")
87
88            output = Data1D(x=x, y=y, lam=lam, dy=dy, dx=dx, dlam=dlam,
89                            isSesans=True)
90
91            output.y_unit = y_unit
92            output.x_unit = x_unit
93            self.filename = output.filename = basename
94            output.xaxis(r"\rm{z}", x_unit)
95            # Adjust label to ln P/(lam^2 t), remove lam column refs
96            output.yaxis(r"\rm{ln(P)/(t \lambda^2)}", y_unit)
97            # Store loading process information
98            output.meta_data['loader'] = self.type_name
99            output.sample.name = params["Sample"]
100            output.sample.ID = params["DataFileTitle"]
101            output.sample.thickness = float(
102                self._unit_conversion(
103                    params["Thickness"], "cm", self._unit_fetch("Thickness"))[0])
104
105            output.sample.zacceptance = (
106                float(params["Theta_zmax"]),
107                self._unit_fetch("Theta_zmax"))
108
109            output.sample.yacceptance = (
110                float(params["Theta_ymax"]),
111                self._unit_fetch("Theta_ymax"))
112            return output
113
114    @staticmethod
115    def _unit_conversion(value, value_unit, default_unit):
116        """
117        Performs unit conversion on a measurement.
118
119        :param value: The magnitude of the measurement
120        :param value_unit: a string containing the final desired unit
121        :param default_unit: a string containing the units of the original measurement
122        :return: The magnitude of the measurement in the new units
123        """
124        # (float, string, string) -> float
125        if has_converter and value_unit != default_unit:
126            data_conv_q = Converter(value_unit)
127            value = data_conv_q(value, units=default_unit)
128            new_unit = default_unit
129        else:
130            new_unit = value_unit
131        return value, new_unit
132
133    def _unit_fetch(self, unit):
134        return self.params[unit+"_unit"]
Note: See TracBrowser for help on using the repository browser.