[Git][debian-gis-team/pygac][upstream] New upstream version 1.7.3

Antonio Valentino (@antonio.valentino) gitlab at salsa.debian.org
Sun Mar 24 17:01:54 GMT 2024



Antonio Valentino pushed to branch upstream at Debian GIS Project / pygac


Commits:
0b5163ec by Antonio Valentino at 2024-03-24T16:30:12+00:00
New upstream version 1.7.3
- - - - -


11 changed files:

- .github/workflows/ci.yaml
- CHANGELOG.md
- bin/pygac-convert-patmosx-coefficients
- pygac/calibration.py
- pygac/data/calibration.json
- pygac/pod_reader.py
- pygac/reader.py
- pygac/tests/test_calibrate_klm.py
- pygac/tests/test_calibrate_pod.py
- pygac/tests/test_klm.py
- pygac/tests/test_reader.py


Changes:

=====================================
.github/workflows/ci.yaml
=====================================
@@ -10,7 +10,7 @@ jobs:
       fail-fast: true
       matrix:
         os: ["ubuntu-latest"]
-        python-version: ["3.8", "3.9", "3.10"]
+        python-version: ["3.10", "3.11", "3.12"]
         experimental: [false]
 
     env:


=====================================
CHANGELOG.md
=====================================
@@ -1,3 +1,15 @@
+## Version 1.7.3 (2024/03/19)
+
+
+### Pull Requests Merged
+
+#### Features added
+
+* [PR 125](https://github.com/pytroll/pygac/pull/125) - Support reading EO-SIP LAC data
+
+In this release 1 pull request was closed.
+
+
 ## Version 1.7.2 (2023/06/26)
 
 


=====================================
bin/pygac-convert-patmosx-coefficients
=====================================
@@ -64,10 +64,10 @@ class PatmosxReader:
         r'\s*(?P<ch1_gain_switches_count>[eE0-9\.-]+)[^\n]*\n'
         r'\s*(?P<ch2_gain_switches_count>[eE0-9\.-]+)[^\n]*\n'
         r'\s*(?P<ch3a_gain_switches_count>[eE0-9\.-]+)[^\n]*\n'
-        r'\s*(?P<PRT1_0>[eE0-9\.-]+)\s*(?P<PRT1_1>[eE0-9\.-]+)\s*(?P<PRT1_2>[eE0-9\.-]+)\s*(?P<PRT1_3>[eE0-9\.-]+)\s*(?P<PRT1_4>[eE0-9\.-]+)[^\n]*\n'
-        r'\s*(?P<PRT2_0>[eE0-9\.-]+)\s*(?P<PRT2_1>[eE0-9\.-]+)\s*(?P<PRT2_2>[eE0-9\.-]+)\s*(?P<PRT2_3>[eE0-9\.-]+)\s*(?P<PRT2_4>[eE0-9\.-]+)[^\n]*\n'
-        r'\s*(?P<PRT3_0>[eE0-9\.-]+)\s*(?P<PRT3_1>[eE0-9\.-]+)\s*(?P<PRT3_2>[eE0-9\.-]+)\s*(?P<PRT3_3>[eE0-9\.-]+)\s*(?P<PRT3_4>[eE0-9\.-]+)[^\n]*\n'
-        r'\s*(?P<PRT4_0>[eE0-9\.-]+)\s*(?P<PRT4_1>[eE0-9\.-]+)\s*(?P<PRT4_2>[eE0-9\.-]+)\s*(?P<PRT4_3>[eE0-9\.-]+)\s*(?P<PRT4_4>[eE0-9\.-]+)[^\n]*\n'
+        r'\s*(?P<PRT1_0>[eE0-9\.-]+)\,*\s*(?P<PRT1_1>[eE0-9\.-]+)\,*\s*(?P<PRT1_2>[eE0-9\.-]+)\,*\s*(?P<PRT1_3>[eE0-9\.-]+)\,*\s*(?P<PRT1_4>[eE0-9\.-]+)[^\n]*\n'
+        r'\s*(?P<PRT2_0>[eE0-9\.-]+)\,*\s*(?P<PRT2_1>[eE0-9\.-]+)\,*\s*(?P<PRT2_2>[eE0-9\.-]+)\,*\s*(?P<PRT2_3>[eE0-9\.-]+)\,*\s*(?P<PRT2_4>[eE0-9\.-]+)[^\n]*\n'
+        r'\s*(?P<PRT3_0>[eE0-9\.-]+)\,*\s*(?P<PRT3_1>[eE0-9\.-]+)\,*\s*(?P<PRT3_2>[eE0-9\.-]+)\,*\s*(?P<PRT3_3>[eE0-9\.-]+)\,*\s*(?P<PRT3_4>[eE0-9\.-]+)[^\n]*\n'
+        r'\s*(?P<PRT4_0>[eE0-9\.-]+)\,*\s*(?P<PRT4_1>[eE0-9\.-]+)\,*\s*(?P<PRT4_2>[eE0-9\.-]+)\,*\s*(?P<PRT4_3>[eE0-9\.-]+)\,*\s*(?P<PRT4_4>[eE0-9\.-]+)[^\n]*\n'
         r'\s*(?P<PRT_weight_1>[eE0-9\.-]+)\s*(?P<PRT_weight_2>[eE0-9\.-]+)\s*(?P<PRT_weight_3>[eE0-9\.-]+)\s*(?P<PRT_weight_4>[eE0-9\.-]+)[^\n]*\n'
         r'(?:\s*(?P<sst_mask_day_0>[eE0-9\.-]+),\s*(?P<sst_mask_day_1>[eE0-9\.-]+),\s*(?P<sst_mask_day_2>[eE0-9\.-]+),\s*(?P<sst_mask_day_3>[eE0-9\.-]+)[^\n]*\n)?'
         r'(?:\s*(?P<sst_mask_night_0>[eE0-9\.-]+),\s*(?P<sst_mask_night_1>[eE0-9\.-]+),\s*(?P<sst_mask_night_2>[eE0-9\.-]+),\s*(?P<sst_mask_night_3>[eE0-9\.-]+)[^\n]*\n)?'


=====================================
pygac/calibration.py
=====================================
@@ -62,7 +62,11 @@ class Calibrator(object):
         '689386c822de18a07194ac7fd71652ea': {
             'name': 'PATMOS-x, v2017r1, with provisional coefficients for MetOp-C',
             'status': CoeffStatus.PROVISIONAL
-        }
+        },
+        'e8735ec394ecdb87b7edcd261e72d2eb': {
+            'name': 'PATMOS-x, v2023',
+            'status': CoeffStatus.PROVISIONAL
+        },
     }
     fields = [
         "dark_count", "gain_switch", "s0", "s1", "s2", "b",  # "b0", "b1", "b2",


=====================================
pygac/data/calibration.json
=====================================
@@ -43,23 +43,23 @@
         "channel_1": {
             "dark_count": 40.43,
             "gain_switch": 501.0,
-            "s0": 0.11266666666666668,
-            "s1": 0.609,
-            "s2": -0.029
+            "s0": 0.11133333333333334,
+            "s1": 0.887,
+            "s2": -0.033
         },
         "channel_2": {
             "dark_count": 39.75,
             "gain_switch": 500.0,
-            "s0": 0.13266666666666668,
-            "s1": 0.98,
-            "s2": -0.016
+            "s0": 0.13333333333333333,
+            "s1": 0.807,
+            "s2": 0.006
         },
         "channel_3a": {
             "dark_count": 41.8,
             "gain_switch": 502.0,
-            "s0": 0.1217142857142857,
-            "s1": 1.224,
-            "s2": -0.033
+            "s0": 0.12457142857142857,
+            "s1": 1.358,
+            "s2": -0.035
         },
         "channel_3b": {
             "b0": 0.0,
@@ -123,22 +123,22 @@
             "dark_count": 39.7,
             "gain_switch": 501.12,
             "s0": 0.11066666666666668,
-            "s1": 2.019,
-            "s2": -0.201
+            "s1": 1.893,
+            "s2": -0.14
         },
         "channel_2": {
             "dark_count": 40.0,
             "gain_switch": 500.82,
-            "s0": 0.122,
-            "s1": 1.476,
-            "s2": -0.137
+            "s0": 0.12266666666666666,
+            "s1": 1.392,
+            "s2": -0.08
         },
         "channel_3a": {
             "dark_count": 40.3,
             "gain_switch": 501.32,
-            "s0": 0.11485714285714287,
-            "s1": 1.748,
-            "s2": -0.033
+            "s0": 0.1142857142857143,
+            "s1": 2.605,
+            "s2": -0.189
         },
         "channel_3b": {
             "b0": 0.0,
@@ -201,23 +201,23 @@
         "channel_1": {
             "dark_count": 40.41,
             "gain_switch": 498.68,
-            "s0": 0.10866666666666668,
-            "s1": 7.424,
-            "s2": -2.579
+            "s0": 0.11,
+            "s1": 1.497,
+            "s2": -0.086
         },
         "channel_2": {
             "dark_count": 40.94,
             "gain_switch": 500.01,
-            "s0": 0.13,
-            "s1": -0.9,
-            "s2": 1.278
+            "s0": 0.12866666666666668,
+            "s1": 3.982,
+            "s2": -0.51
         },
         "channel_3a": {
             "dark_count": 40.57,
             "gain_switch": 498.72,
-            "s0": 0.16228571428571428,
-            "s1": -45.535,
-            "s2": 20.122
+            "s0": 0.12457142857142857,
+            "s1": 5.208,
+            "s2": -0.91
         },
         "channel_3b": {
             "b0": 0.0,
@@ -281,20 +281,20 @@
             "dark_count": 39.44,
             "gain_switch": null,
             "s0": 0.111,
-            "s1": 6.087,
-            "s2": -1.039
+            "s1": 6.031,
+            "s2": -1.089
         },
         "channel_2": {
             "dark_count": 39.4,
             "gain_switch": null,
             "s0": 0.137,
-            "s1": 0.119,
-            "s2": 0.123
+            "s1": -0.006,
+            "s2": 0.179
         },
         "channel_3a": {
             "dark_count": 37.51,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -360,15 +360,15 @@
             "dark_count": 40.0,
             "gain_switch": null,
             "s0": 0.11,
-            "s1": 0.632,
-            "s2": -0.044
+            "s1": 0.563,
+            "s2": -0.031
         },
         "channel_2": {
             "dark_count": 40.0,
             "gain_switch": null,
-            "s0": 0.118,
-            "s1": -0.037,
-            "s2": 0.072
+            "s0": 0.117,
+            "s1": 0.164,
+            "s2": 0.039
         },
         "channel_3a": {
             "dark_count": 40.0,
@@ -438,21 +438,21 @@
         "channel_1": {
             "dark_count": 41.0,
             "gain_switch": null,
-            "s0": 0.121,
-            "s1": 2.032,
-            "s2": -0.032
+            "s0": 0.12,
+            "s1": 2.184,
+            "s2": -0.051
         },
         "channel_2": {
             "dark_count": 40.0,
             "gain_switch": null,
-            "s0": 0.148,
-            "s1": 1.323,
-            "s2": -0.008
+            "s0": 0.151,
+            "s1": 0.505,
+            "s2": 0.118
         },
         "channel_3a": {
             "dark_count": 40.0,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -518,20 +518,20 @@
             "dark_count": 41.0,
             "gain_switch": null,
             "s0": 0.121,
-            "s1": 3.555,
-            "s2": -0.339
+            "s1": 3.559,
+            "s2": -0.334
         },
         "channel_2": {
             "dark_count": 41.0,
             "gain_switch": null,
-            "s0": 0.152,
-            "s1": 0.254,
-            "s2": 0.201
+            "s0": 0.148,
+            "s1": 1.342,
+            "s2": 0.096
         },
         "channel_3a": {
             "dark_count": 39.0,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -596,20 +596,20 @@
         "channel_1": {
             "dark_count": 39.0,
             "gain_switch": 500.0,
-            "s0": 0.11933333333333333,
-            "s1": -0.069,
-            "s2": 0.002
+            "s0": 0.12,
+            "s1": -0.241,
+            "s2": 0.012
         },
         "channel_2": {
             "dark_count": 40.0,
             "gain_switch": 500.0,
-            "s0": 0.13733333333333334,
-            "s1": 0.339,
-            "s2": -0.01
+            "s0": 0.138,
+            "s1": 0.095,
+            "s2": 0.008
         },
         "channel_3a": {
             "dark_count": 39.0,
-            "gain_switch": 500.0,
+            "gain_switch": null,
             "s0": 0.1,
             "s1": 0.0,
             "s2": 0.0
@@ -676,22 +676,22 @@
             "dark_count": 39.3,
             "gain_switch": 498.96,
             "s0": 0.11,
-            "s1": 0.839,
-            "s2": -0.051
+            "s1": 1.268,
+            "s2": -0.126
         },
         "channel_2": {
             "dark_count": 38.9,
             "gain_switch": 500.17,
             "s0": 0.11933333333333333,
-            "s1": 0.786,
-            "s2": -0.031
+            "s1": 0.758,
+            "s2": -0.06
         },
         "channel_3a": {
             "dark_count": 38.4,
             "gain_switch": 499.43,
-            "s0": 0.10685714285714286,
-            "s1": 0.29,
-            "s2": -0.294
+            "s0": 0.108,
+            "s1": -0.146,
+            "s2": -0.27
         },
         "channel_3b": {
             "b0": 0.0,
@@ -754,23 +754,23 @@
         "channel_1": {
             "dark_count": 39.99,
             "gain_switch": 501.12,
-            "s0": 0.11466666666666665,
-            "s1": 1.007,
-            "s2": -0.044
+            "s0": 0.116,
+            "s1": 0.517,
+            "s2": 0.028
         },
         "channel_2": {
             "dark_count": 39.09,
             "gain_switch": 500.73,
-            "s0": 0.14,
-            "s1": 1.474,
-            "s2": -0.118
+            "s0": 0.14133333333333334,
+            "s1": 0.739,
+            "s2": 0.026
         },
         "channel_3a": {
             "dark_count": 42.09,
             "gain_switch": 501.37,
-            "s0": 0.11942857142857143,
-            "s1": 2.787,
-            "s2": -0.292
+            "s0": 0.12,
+            "s1": 3.086,
+            "s2": -0.301
         },
         "channel_3b": {
             "b0": 0.0,
@@ -833,21 +833,21 @@
         "channel_1": {
             "dark_count": 39.44,
             "gain_switch": 500.54,
-            "s0": 0.114,
-            "s1": 0.603,
-            "s2": -0.0
+            "s0": 0.11133333333333334,
+            "s1": 1.13,
+            "s2": -0.017
         },
         "channel_2": {
             "dark_count": 39.4,
             "gain_switch": 500.4,
-            "s0": 0.128,
-            "s1": 0.632,
-            "s2": 0.045
+            "s0": 0.124,
+            "s1": 1.39,
+            "s2": 0.011
         },
         "channel_3a": {
             "dark_count": 37.51,
             "gain_switch": 500.56,
-            "s0": 0.1,
+            "s0": 0.22350000000000014,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -912,21 +912,21 @@
         "channel_1": {
             "dark_count": 38.8,
             "gain_switch": 496.43,
-            "s0": 0.108,
-            "s1": 0.626,
-            "s2": -0.044
+            "s0": 0.10866666666666668,
+            "s1": 0.286,
+            "s2": 0.012
         },
         "channel_2": {
             "dark_count": 39.0,
             "gain_switch": 500.37,
             "s0": 0.122,
-            "s1": 0.95,
-            "s2": -0.039
+            "s1": 0.478,
+            "s2": 0.052
         },
         "channel_3a": {
             "dark_count": 39.4,
             "gain_switch": 496.11,
-            "s0": 0.1,
+            "s0": 0.10771428571385998,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -1005,7 +1005,7 @@
         "channel_3a": {
             "dark_count": 37.51,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -1084,7 +1084,7 @@
         "channel_3a": {
             "dark_count": 39.0,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -1149,21 +1149,21 @@
         "channel_1": {
             "dark_count": 39.44,
             "gain_switch": null,
-            "s0": 0.119,
-            "s1": 6.065,
+            "s0": 0.126,
+            "s1": 2.974,
             "s2": 0.0
         },
         "channel_2": {
             "dark_count": 39.4,
             "gain_switch": null,
-            "s0": 0.136,
-            "s1": 7.248,
+            "s0": 0.138,
+            "s1": 5.958,
             "s2": 0.0
         },
         "channel_3a": {
             "dark_count": 37.51,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -1228,21 +1228,21 @@
         "channel_1": {
             "dark_count": 38.0,
             "gain_switch": null,
-            "s0": 0.108,
-            "s1": 4.255,
-            "s2": 0.64
+            "s0": 0.107,
+            "s1": 4.694,
+            "s2": 0.51
         },
         "channel_2": {
             "dark_count": 40.0,
             "gain_switch": null,
-            "s0": 0.122,
-            "s1": 0.31,
-            "s2": 0.642
+            "s0": 0.121,
+            "s1": 1.147,
+            "s2": 0.428
         },
         "channel_3a": {
             "dark_count": 38.0,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -1321,7 +1321,7 @@
         "channel_3a": {
             "dark_count": 37.51,
             "gain_switch": null,
-            "s0": 0.1,
+            "s0": 0.0,
             "s1": 0.0,
             "s2": 0.0
         },
@@ -1382,5 +1382,4 @@
             "d4": 0.0
         }
     }
-}
-
+}
\ No newline at end of file


=====================================
pygac/pod_reader.py
=====================================
@@ -279,7 +279,7 @@ class PODReader(Reader):
         # choose the right header depending on the date
         with file_opener(fileobj or filename) as fd_:
             self.tbm_head, self.head = self.read_header(
-                filename, fileobj=fd_)
+                filename, fileobj=fd_, header_date=self.header_date)
             if self.tbm_head:
                 tbm_offset = tbm_header.itemsize
             else:
@@ -302,12 +302,14 @@ class PODReader(Reader):
         return self.head, self.scans
 
     @classmethod
-    def read_header(cls, filename, fileobj=None):
+    def read_header(cls, filename, fileobj=None, header_date="auto"):
         """Read the file header.
 
         Args:
             filename (str): Path to GAC/LAC file
             fileobj: An open file object to read from. (optional)
+            header_date: date to use to choose the header.
+                Defaults to "auto" to use the data to pick the header corresponding to the date of the file.
 
         Returns:
             archive_header (struct): archive header
@@ -332,19 +334,7 @@ class PODReader(Reader):
                 fd_.seek(0)
                 tbm_head = None
                 tbm_offset = 0
-            # read header
-            head0, = np.frombuffer(
-                fd_.read(header0.itemsize),
-                dtype=header0, count=1)
-            year, jday, _ = cls.decode_timestamps(head0["start_time"])
-            start_date = (datetime.date(year, 1, 1) +
-                          datetime.timedelta(days=int(jday) - 1))
-            if start_date < datetime.date(1992, 9, 8):
-                header = header1
-            elif start_date <= datetime.date(1994, 11, 15):
-                header = header2
-            else:
-                header = header3
+            header = cls.choose_header_based_on_timestamp(header_date, fd_)
             fd_.seek(tbm_offset, 0)
             # need to copy frombuffer to have write access on head
             head, = np.frombuffer(
@@ -354,6 +344,31 @@ class PODReader(Reader):
         cls._validate_header(head)
         return tbm_head, head
 
+    @classmethod
+    def choose_header_based_on_timestamp(cls, header_date, fd_):
+        """Choose the header dtype based on the timestamp."""
+        if header_date == "auto":
+            header_date = cls.get_start_date(fd_)
+        if header_date < datetime.date(1992, 9, 8):
+            header = header1
+        elif header_date <= datetime.date(1994, 11, 15):
+            header = header2
+        else:
+            header = header3
+        return header
+
+    @classmethod
+    def get_start_date(cls, fd_):
+        """Get the start time from the filestream."""
+        head0, = np.frombuffer(
+                    fd_.read(header0.itemsize),
+                    dtype=header0, count=1)
+        year, jday, _ = cls.decode_timestamps(head0["start_time"])
+        header_date = (datetime.date(year, 1, 1) +
+                       datetime.timedelta(days=int(jday) - 1))
+
+        return header_date
+
     @classmethod
     def _validate_header(cls, header):
         """Check if the header belongs to this reader."""


=====================================
pygac/reader.py
=====================================
@@ -86,6 +86,10 @@ class NoTLEData(IndexError):
     """Raised if no TLE data available within time range."""
 
 
+class DecodingError(ValueError):
+    """Raised when decoding of some value fails."""
+
+
 class Reader(six.with_metaclass(ABCMeta)):
     """Reader for GAC and LAC format, POD and KLM platforms."""
 
@@ -95,7 +99,7 @@ class Reader(six.with_metaclass(ABCMeta)):
 
     def __init__(self, interpolate_coords=True, adjust_clock_drift=True,
                  tle_dir=None, tle_name=None, tle_thresh=7, creation_site=None,
-                 custom_calibration=None, calibration_file=None):
+                 custom_calibration=None, calibration_file=None, header_date="auto"):
         """Init the reader.
 
         Args:
@@ -111,6 +115,7 @@ class Reader(six.with_metaclass(ABCMeta)):
             custom_calibration: dictionary with a subset of user defined satellite specific
                                 calibration coefficients
             calibration_file: path to json file containing default calibrations
+            header_date: the date to use for pod header choice. Defaults to "auto".
 
         """
         self.meta_data = {}
@@ -122,6 +127,7 @@ class Reader(six.with_metaclass(ABCMeta)):
         self.creation_site = (creation_site or 'NSS').encode('utf-8')
         self.custom_calibration = custom_calibration
         self.calibration_file = calibration_file
+        self.header_date = header_date
         self.head = None
         self.scans = None
         self.spacecraft_name = None
@@ -205,22 +211,38 @@ class Reader(six.with_metaclass(ABCMeta)):
             filename (str): path to file
         """
         filename = str(filename)
-        data_set_name = header['data_set_name'].decode(errors='ignore')
-        if not cls.data_set_pattern.match(data_set_name):
-            LOG.debug('The data_set_name in header %s does not match.'
-                      ' Use filename instead.' % header['data_set_name'])
+        for encoding in "utf-8", "cp500":
+            data_set_name = header['data_set_name']
+            try:
+                data_set_name = cls._decode_data_set_name(data_set_name, encoding)
+            except DecodingError as err:
+                LOG.debug(str(err))
+            else:
+                header["data_set_name"] = data_set_name
+                break
+        else:
+            LOG.debug(f'The data_set_name in header {header["data_set_name"]} does not match.'
+                      ' Use filename instead.')
             match = cls.data_set_pattern.search(filename)
             if match:
                 data_set_name = match.group()
-                LOG.debug("Set data_set_name, to filename %s"
-                          % data_set_name)
+                LOG.debug(f"Set data_set_name, to filename {data_set_name}")
                 header['data_set_name'] = data_set_name.encode()
             else:
-                LOG.debug("header['data_set_name']=%s; filename='%s'"
-                          % (header['data_set_name'], filename))
+                LOG.debug(f"header['data_set_name']={header['data_set_name']}; filename='{filename}'")
                 raise ReaderError('Cannot determine data_set_name!')
         return header
 
+    @classmethod
+    def _decode_data_set_name(cls, data_set_name, encoding):
+        data_set_name = data_set_name.decode(encoding, errors='ignore')
+        if not cls.data_set_pattern.match(data_set_name):
+            raise DecodingError(f'The data_set_name in header {data_set_name} '
+                                f'does not seem correct using encoding {encoding}.')
+        else:
+            data_set_name = data_set_name.encode()
+        return data_set_name
+
     @classmethod
     def _validate_header(cls, header):
         """Check if the header belongs to this reader.
@@ -274,7 +296,7 @@ class Reader(six.with_metaclass(ABCMeta)):
                 "Expected %d scan lines, but found %d!"
                 % (count, line_count))
             warnings.warn("Unexpected number of scanlines!",
-                          category=RuntimeWarning)
+                          category=RuntimeWarning, stacklevel=2)
         self.scans = np.frombuffer(
             buffer, dtype=self.scanline_type, count=line_count)
 


=====================================
pygac/tests/test_calibrate_klm.py
=====================================
@@ -63,12 +63,12 @@ class TestGenericCalibration(unittest.TestCase):
 
         ref3 = calibrate_solar(data, channel, year, jday, cal, corr)
 
-        expected = (np.array([[np.nan, 27.37909518, 110.60103456],
-                              [0.11943135, 6.03671211, 57.99695154]]),
-                    np.array([[np.nan, 3.05229160e+01, 1.24811455e+02],
-                              [1.23011792e-01, 6.82715447e+00, 6.52122414e+01]]),
-                    np.array([[0., 523.41775, 1034.41775],
-                              [41., 150., 711.41775]]))
+        expected = (np.array([[np.nan, 27.32328509, 110.84050459],
+                              [0.1191198, 6.02096454, 58.0497768]]),
+                    np.array([[np.nan, 3.04160070e+01, 1.24374292e+02],
+                              [1.22580933e-01, 6.80324179e+00, 6.49838301e+01]]),
+                    np.array([[0., 524.33117, 1035.33117],
+                              [41., 150., 712.33117]]))
         np.testing.assert_allclose(ref1, expected[0])
         np.testing.assert_allclose(ref2, expected[1])
         np.testing.assert_allclose(ref3, expected[2])
@@ -87,9 +87,9 @@ class TestGenericCalibration(unittest.TestCase):
         ict_counts = np.array([[745.3, 397.9, 377.8],
                                [744.8, 398.1, 378.4],
                                [745.7, 398., 378.3]])
-        space_counts = np.array([[987.3,  992.5,  989.4],
-                                 [986.9,  992.8,  989.6],
-                                 [986.3,  992.3,  988.9]])
+        space_counts = np.array([[987.3, 992.5, 989.4],
+                                 [986.9, 992.8, 989.6],
+                                 [986.3, 992.3, 988.9]])
 
         spacecraft_id = "noaa19"
         cal = Calibrator(spacecraft_id)


=====================================
pygac/tests/test_calibrate_pod.py
=====================================
@@ -64,10 +64,10 @@ class TestGenericCalibration(unittest.TestCase):
 
         ref3 = calibrate_solar(data, channel, year, jday, cal, corr)
 
-        expected = (np.array([[np.nan, 60.891074, 126.953364],
-                              [0., 14.091565, 85.195791]]),
-                    np.array([[np.nan, 72.98262, 152.16334],
-                              [0., 16.889821, 102.113687]]),
+        expected = (np.array([[np.nan, 60.91525491, 127.00377987],
+                              [0., 14.0971609, 85.22962417]]),
+                    np.array([[np.nan, 72.51635437, 151.19121018],
+                              [0., 16.7819164, 101.46131111]]),
                     np.array([[-32001., -32001., -32001.],
                               [-32001., -32001., -32001.]]))
 
@@ -89,9 +89,9 @@ class TestGenericCalibration(unittest.TestCase):
         ict_counts = np.array([[745.3, 397.9, 377.8],
                                [744.8, 398.1, 378.4],
                                [745.7, 398., 378.3]])
-        space_counts = np.array([[987.3,  992.5,  989.4],
-                                 [986.9,  992.8,  989.6],
-                                 [986.3,  992.3,  988.9]])
+        space_counts = np.array([[987.3, 992.5, 989.4],
+                                 [986.9, 992.8, 989.6],
+                                 [986.3, 992.3, 988.9]])
 
         spacecraft_id = "noaa14"
         cal = Calibrator(spacecraft_id)


=====================================
pygac/tests/test_klm.py
=====================================
@@ -38,7 +38,7 @@ from pygac.tests.utils import CalledWithArray
 class TestKLM:
     """Test the klm reader."""
 
-    def setup(self):
+    def setup_method(self):
         """Set up the tests."""
         self.reader = GACKLMReader()
 
@@ -126,7 +126,7 @@ class TestKLM:
 class TestGACKLM:
     """Tests for gac klm."""
 
-    def setup(self):
+    def setup_method(self):
         """Set up the tests."""
         self.reader = GACKLMReader()
 
@@ -150,7 +150,7 @@ class TestGACKLM:
 class TestLACKLM:
     """Tests for lac klm."""
 
-    def setup(self):
+    def setup_method(self):
         """Set up the tests."""
         self.reader = LACKLMReader()
         self.reader.scans = np.ones(100, dtype=scanline)


=====================================
pygac/tests/test_reader.py
=====================================
@@ -24,11 +24,9 @@ import datetime
 import os
 import sys
 import unittest
+import pytest
 
-try:
-    import mock
-except ImportError:
-    from unittest import mock
+from unittest import mock
 import numpy as np
 import numpy.testing
 from pygac.gac_reader import GACReader, ReaderError
@@ -36,6 +34,10 @@ from pygac.lac_reader import LACReader
 from pygac.pod_reader import POD_QualityIndicator
 from pygac.gac_pod import scanline
 from pygac.reader import NoTLEData
+from pygac.lac_pod import LACPODReader
+
+from pygac.pod_reader import tbm_header as tbm_header_dtype, header3
+from pygac.lac_pod import scanline as lacpod_scanline
 
 
 class TestPath(os.PathLike):
@@ -127,9 +129,6 @@ class TestGacReader(unittest.TestCase):
         """Set up the tests."""
         self.interpolator = interpolator
         self.reader = GACReader()
-        # python 2 compatibility
-        if sys.version_info.major < 3:
-            self.assertRaisesRegex = self.assertRaisesRegexp
 
     def test_filename(self):
         """Test the setter of the filename property."""
@@ -174,29 +173,62 @@ class TestGacReader(unittest.TestCase):
             head = {'data_set_name': b'\xea\xf8'}
             self.reader._validate_header(head)
 
-    def test__correct_data_set_name(self):
+    def test__correct_data_set_name_ebcdic_encoded_header_invalid_path(self):
+        """Test the data_set_name correction in file header."""
+        inv_filename = 'InvalidFileName'
+        inv_filepath = 'path/to/' + inv_filename
+
+        expected_data_set_name = 'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'
+        val_head = {'data_set_name': 'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'.encode("cp500")}
+        head = self.reader._correct_data_set_name(val_head.copy(), inv_filepath)
+        assert head['data_set_name'] == expected_data_set_name.encode()
+
+    def test__correct_data_set_name_valid_header_and_file(self):
         """Test the data_set_name correction in file header."""
         val_filename = 'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'
         val_filepath = 'path/to/' + val_filename
         val_head = {'data_set_name': b'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'}
-        inv_filename = 'InvalidFileName'
-        inv_filepath = 'path/to/' + inv_filename
-        inv_head = {'data_set_name': b'InvalidDataSetName'}
         # Note: always pass a copy to _correct_data_set_name, because
         #       the input header is modified in place.
         # enter a valid data_set_name and filepath
         head = self.reader._correct_data_set_name(val_head.copy(), val_filepath)
+        assert head['data_set_name'] == val_filename.encode()
+
+    def test__correct_data_set_name_invalid_header_and_valid_file(self):
+        """Test the data_set_name correction in file header."""
+        val_filename = 'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'
+        val_filepath = 'path/to/' + val_filename
+        inv_head = {'data_set_name': b'InvalidDataSetName'}
+
         # enter an invalid data_set_name, but valid filepath
         head = self.reader._correct_data_set_name(inv_head.copy(), val_filepath)
-        self.assertEqual(head['data_set_name'], val_filename.encode())
-        # enter an invalid data_set_name, and invalid filepath
+        assert head['data_set_name'] == val_filename.encode()
+
+    def test__correct_data_set_name_invalid_header_and_file(self):
+        """Test the data_set_name correction in file header."""
+        inv_filename = 'InvalidFileName'
+        inv_filepath = 'path/to/' + inv_filename
+        inv_head = {'data_set_name': b'InvalidDataSetName'}
         with self.assertRaisesRegex(ReaderError, 'Cannot determine data_set_name!'):
-            head = self.reader._correct_data_set_name(inv_head.copy(), inv_filepath)
+            _ = self.reader._correct_data_set_name(inv_head.copy(), inv_filepath)
+
+    def test__correct_data_set_name_valid_header_invalid_file(self):
+        """Test the data_set_name correction in file header."""
+        val_head = {'data_set_name': b'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'}
+        inv_filename = 'InvalidFileName'
+        inv_filepath = 'path/to/' + inv_filename
+
         # enter a valid data_set_name, and an invalid filepath
         # should be fine, because the data_set_name is the pefered source
         head = self.reader._correct_data_set_name(val_head.copy(), inv_filepath)
-        self.assertEqual(head['data_set_name'], val_head['data_set_name'])
-        # enter a valid data_set_name, and an FSFile/pathlib object as filepath
+        assert head['data_set_name'] == val_head['data_set_name']
+
+    def test__correct_data_set_name_valid_header_pathlib_file(self):
+        """Test the data_set_name correction in file header."""
+        val_filename = 'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'
+        val_filepath = 'path/to/' + val_filename
+        val_head = {'data_set_name': b'NSS.GHRR.TN.D80001.S0332.E0526.B0627173.WI'}
+
         fs_filepath = TestPath(val_filepath)
         head = self.reader._correct_data_set_name(val_head.copy(), fs_filepath)
         self.assertEqual(head['data_set_name'], val_filename.encode())
@@ -648,3 +680,105 @@ class TestLacReader(unittest.TestCase):
         self.reader.correct_scan_line_numbers()
         numpy.testing.assert_array_equal(self.reader.scans['scan_line_number'],
                                          expected)
+
+
+ at pytest.fixture
+def pod_file_with_tbm_header(tmp_path):
+    """Create a pod file with a tbm header, and header, and some scanlines."""
+    number_of_scans = 3
+
+    tbm_header = np.zeros(1, dtype=tbm_header_dtype)
+    tbm_header["data_set_name"] = b"BRN.HRPT.NJ.D00322.S0334.E0319.B3031919.BL  "
+    tbm_header["select_flag"] = b"S"
+    tbm_header["beginning_latitude"] = b"+77"
+    tbm_header["ending_latitude"] = b"+22"
+    tbm_header["beginning_longitude"] = b"-004"
+    tbm_header["ending_longitude"] = b"+032"
+    tbm_header["start_hour"] = b'AL'
+    tbm_header["start_minute"] = b'L '
+    tbm_header["number_of_minutes"] = b'ALL'
+    tbm_header["appended_data_flag"] = b'Y'
+    tbm_header["channel_select_flag"][0, :5] = b'\x01'
+    tbm_header["sensor_data_word_size"] = b'10'
+
+    header = np.zeros(1, dtype=header3)
+    header["noaa_spacecraft_identification_code"] = 3
+    header["data_type_code"] = 48
+    header["start_time"] = [51522, 181, 62790]
+    header["number_of_scans"] = number_of_scans
+    header["end_time"] = [51522, 195, 42286]
+    header["processing_block_id"] = b'3031919'
+    header["ramp_auto_calibration"] = 0
+    header["number_of_data_gaps"] = 0
+    header["dacs_quality"] = [21, 7, 0, 0, 0, 0]
+    header["calibration_parameter_id"] = 12336
+    header["dacs_status"] = 24
+    header["nadir_earth_location_tolerance"] = 20
+    header["start_of_data_set_year"] = 2000
+    # EBCDIC, aka cp500 encoding
+    header["data_set_name"] = (b"\xc2\xd9\xd5K\xc8\xd9\xd7\xe3K\xd5\xd1K\xc4\xf0\xf0\xf3\xf2\xf2K\xe2\xf0\xf3\xf1\xf9K"
+                               b"\xc5\xf0\xf3\xf3\xf4K\xc2\xf3\xf0\xf3\xf1\xf9\xf1\xf9K\xc2\xd3@@")
+    header["year_of_epoch"] = 2000
+    header["julian_day_of_epoch"] = 322
+    header["millisecond_utc_epoch_time_of_day"] = 11864806
+    header["semi_major_axis"] = 7226239
+    header["eccentricity"] = 100496
+    header["inclination"] = 9915900
+    header["argument_of_perigee"] = 2511798
+    header["right_ascension"] = 30366227
+    header["mean_anomaly"] = 7341437
+    header["x_component_of_position_vector"] = 40028760
+    header["y_component_of_position_vector"] = -60151283
+    header["z_component_of_position_vector"] = 0,
+    header["x_dot_component_of_position_vector"] = -990271
+    header["y_dot_component_of_position_vector"] = -646066
+    header["z_dot_component_of_position_vector"] = 7337861
+    header["yaw_fixed_error_correction"] = 0
+    header["roll_fixed_error_correction"] = 0
+    header["pitch_fixed_error_correction"] = 0
+
+    scanlines = np.zeros(number_of_scans, dtype=lacpod_scanline)
+    scanlines["scan_line_number"] = np.arange(number_of_scans) + 1
+    scanlines["time_code"][:, :2] = [51522, 181]
+    scanlines["time_code"][:, 2] = np.arange(62790, 62790 + 166 * number_of_scans, 166)
+    scanlines["quality_indicators"] = 1073741824
+    scanlines["calibration_coefficients"] = [149722896, -23983736, 173651248, -27815402, -1803673, 6985664, -176321840,
+                                             666680320, -196992576, 751880640]
+    scanlines["number_of_meaningful_zenith_angles_and_earth_location_appended"] = 51
+    scanlines["solar_zenith_angles"] = [-24, -26, -28, -30, -32, -33, -34, -35, -37, -37, -38, -39, -40, -41, -41, -42,
+                                        -43, -43, -44, -45, -45, -46, -46, -47, -48, -48, -49, -49, -50, -50, -51, -52,
+                                        -52, -53, -53, -54, -55, -55, -56, -57, -58, -59, -60, -61, -62, -63, -64, -66,
+                                        -68, -70, -73]
+    scanlines["earth_location"] = [9929, -36, 9966, 747, 9983, 1415, 9988, 1991, 9984, 2492, 9975, 2931,
+                                   9963, 3320, 9948, 3667, 9931, 3979, 9913, 4262, 9895, 4519, 9875, 4755,
+                                   9856, 4972, 9836, 5174, 9816, 5362, 9796, 5538, 9775, 5703, 9755, 5860,
+                                   9734, 6009, 9713, 6150, 9692, 6286, 9671, 6416, 9650, 6542, 9628, 6663,
+                                   9606, 6781, 9583, 6896, 9560, 7009, 9537, 7119, 9513, 7227, 9488, 7334,
+                                   9463, 7440, 9437, 7545, 9409, 7650, 9381, 7755, 9351, 7860, 9320,
+                                   7966, 9288, 8073, 9253, 8181, 9216, 8291, 9177, 8403, 9135, 8518,
+                                   9090, 8637, 9040, 8759, 8986, 8886, 8926, 9019, 8859, 9159, 8784,
+                                   9307, 8697, 9466, 8596, 9637, 8476, 9824, 8327, 10033]
+    scanlines["telemetry"] = 0
+    scanlines["sensor_data"] = 99
+    scanlines["add_on_zenith"] = 0
+    scanlines["clock_drift_delta"] = 0
+
+    pod_filename = tmp_path / "image.l1b"
+    offset = 14800
+    fill = np.zeros(offset - header3.itemsize, dtype=np.uint8)
+
+    with open(pod_filename, "wb") as fd_:
+        fd_.write(tbm_header.tobytes())
+        fd_.write(header.tobytes())
+        fd_.write(fill.tobytes())
+        fd_.write(scanlines.tobytes())
+    return pod_filename
+
+
+def test_podlac_eosip(pod_file_with_tbm_header):
+    """Test reading a real podlac file."""
+    reader = LACPODReader(interpolate_coords=False)
+    reader.read(pod_file_with_tbm_header)
+    assert reader.head.itemsize == header3.itemsize
+    # this is broken in eosip pod data, tbm data set name has start and end times reversed.
+    # assert reader.head["data_set_name"] == reader.tbm_head["data_set_name"]



View it on GitLab: https://salsa.debian.org/debian-gis-team/pygac/-/commit/0b5163ec9c81ce1abb812b976ae292f043348ba3

-- 
View it on GitLab: https://salsa.debian.org/debian-gis-team/pygac/-/commit/0b5163ec9c81ce1abb812b976ae292f043348ba3
You're receiving this email because of your account on salsa.debian.org.


-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://alioth-lists.debian.net/pipermail/pkg-grass-devel/attachments/20240324/5f63138f/attachment-0001.htm>


More information about the Pkg-grass-devel mailing list