Skip to content
dataset.py 19.4 KiB
Newer Older
Christoph.Knote's avatar
sdf
Christoph.Knote committed
import datetime
import sys

class Variable:
Christoph.Knote's avatar
Christoph.Knote committed
    '''
Christoph.Knote's avatar
Christoph.Knote committed
    A Variable is a ICARTT variable description with name, units, scale and missing value.
Christoph.Knote's avatar
Christoph.Knote committed
    '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
    @property
    def desc(self):
        '''
        Return variable description string as it appears in an ICARTT file
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        return self.splitChar.join( [self.name, self.units, self.units ] )
    def __init__(self, name, units, scale=1.0, miss=-9999999):
Christoph.Knote's avatar
Christoph.Knote committed
        #: Name
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        self.name = name
Christoph.Knote's avatar
Christoph.Knote committed
        #: Units
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        self.units = units
Christoph.Knote's avatar
Christoph.Knote committed
        #: Scale factor
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        self.scale = scale
Christoph.Knote's avatar
Christoph.Knote committed
        #: Missing value (string, just as it appears in the ICARTT file)
Christoph.Knote's avatar
Christoph.Knote committed
        self.miss = str(miss)
Christoph.Knote's avatar
Christoph.Knote committed
        #: Split character for description string
        self.splitChar    = ','
Christoph.Knote's avatar
sdf
Christoph.Knote committed

class Dataset:
Christoph.Knote's avatar
Christoph.Knote committed
    '''
    An ICARTT dataset that can be created from scratch or read from a file,
    manipulated, and then written to a file.
    '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
    @property
    def nheader(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Header line count
        '''
Christoph.Knote's avatar
kj  
Christoph.Knote committed
        total = 12 + self.ndvar + 1 + self.nscom + 1 + self.nncom
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        if self.format == 2110:
            total += self.nauxvar + 5
        return total
    @property
    def ndvar(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Dependent variable count
        '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        return len(self.DVAR)
    @property
    def nauxvar(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Auxiliary variables count
        '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        return len(self.AUXVAR)
    @property
    def nvar(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Variable count (independent + dependent)
        '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        return self.ndvar + 1
    @property
    def nscom(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Special comments count
        '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        return len(self.SCOM)
    @property
    def nncom(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Normal comments count
        '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        return len(self.NCOM)
Christoph.Knote's avatar
Christoph.Knote committed
    @property
    def VAR(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Variables (independent and dependent)
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        return [ self.IVAR ] + self.DVAR
    @property
    def varnames(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Names of variables (independent and dependent)
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        return [ x.name for x in self.VAR ]
Christoph.Knote's avatar
Christoph.Knote committed
    @property
    def times(self):
        '''
        Time steps of the data contained.
        '''
        return [ self.dateValid + datetime.timedelta(seconds=x) for x in self[self.IVAR.name] ]

    def __getitem__(self, name):
        '''
        Convenience function to access variable data by name::

           ict = icartt.Dataset(<fname>)
           ict['O3']
        '''
        idx = self.index(name)
        if idx == -1:
            raise Exception("{:s} not found in data".format(name))
        return [ x[idx] for x in self.data ]
Christoph.Knote's avatar
Christoph.Knote committed

    def units(self, name):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Units of variable <name>
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        res = [ x.units for x in self.VAR if x.name == name ]
        if len(res) is 0:
            res = [ '' ]
        return res[0]

    def index(self, name):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Index of variable <name> in data array
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        res = [ i for i, x in enumerate(self.VAR) if x.name == name ]
        if len(res) is 0:
            res = [ -1 ]
        return res[0]
Christoph.Knote's avatar
sdf
Christoph.Knote committed

    def write(self, f=sys.stdout):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Write to file handle <f>
        '''
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        def prnt(txt):
            f.write(str(txt) + "\n")

        # Number of lines in header, file format index (most files use 1001) - comma delimited.
        prnt("{:d}, {:d}".format(self.nheader, self.format))
        # PI last name, first name/initial.
        prnt(self.PI)
        # Organization/affiliation of PI.
        prnt(self.organization)
        # Data source description (e.g., instrument name, platform name, model name, etc.).
        prnt(self.dataSource)
        # Mission name (usually the mission acronym).
        prnt(self.mission)
        # File volume number, number of file volumes (these integer values are used when the data require more than one file per day; for data that require only one file these values are set to 1, 1) - comma delimited.
Christoph.Knote's avatar
Christoph.Knote committed
        prnt(self.splitChar.join([ str(self.VOL), str(self.NVOL) ]))
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        # UTC date when data begin, UTC date of data reduction or revision - comma delimited (yyyy, mm, dd, yyyy, mm, dd).
Christoph.Knote's avatar
Christoph.Knote committed
        prnt(self.splitChar.join([ datetime.datetime.strftime(x, "%Y, %m, %d") for x in [ self.dateValid, self.dateRevised ] ]))
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        # Data Interval (This value describes the time spacing (in seconds) between consecutive data records. It is the (constant) interval between values of the independent variable. For 1 Hz data the data interval value is 1 and for 10 Hz data the value is 0.1. All intervals longer than 1 second must be reported as Start and Stop times, and the Data Interval value is set to 0. The Mid-point time is required when it is not at the average of Start and Stop times. For additional information see Section 2.5 below.).
        prnt("0")
        # Description or name of independent variable (This is the name chosen for the start time. It always refers to the number of seconds UTC from the start of the day on which measurements began. It should be noted here that the independent variable should monotonically increase even when crossing over to a second day.).
        prnt(self.IVAR.desc)
        if self.format == 2110:
            # Description or name of independent (bound) variable (This is the name chosen for the start time. It always refers to the number of seconds UTC from the start of the day on which measurements began. It should be noted here that the independent variable should monotonically increase even when crossing over to a second day.).
            prnt(self.IBVAR.desc)
        # Number of variables (Integer value showing the number of dependent variables: the total number of columns of data is this value plus one.).
        prnt(self.ndvar)
        # Scale factors (1 for most cases, except where grossly inconvenient) - comma delimited.
Christoph.Knote's avatar
Christoph.Knote committed
        prnt(self.splitChar.join( [ "{:6.3f}".format(x.scale) for x in self.DVAR ]))
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        # Missing data indicators (This is -9999 (or -99999, etc.) for any missing data condition, except for the main time (independent) variable which is never missing) - comma delimited.
Christoph.Knote's avatar
Christoph.Knote committed
        prnt(self.splitChar.join( [ str(x.miss) for x in self.DVAR ]))
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        # Variable names and units (Short variable name and units are required, and optional long descriptive name, in that order, and separated by commas. If the variable is unitless, enter the keyword "none" for its units. Each short variable name and units (and optional long name) are entered on one line. The short variable name must correspond exactly to the name used for that variable as a column header, i.e., the last header line prior to start of data.).
        nul = [ prnt(x.desc) for x in self.DVAR ]
        if self.format == 2110:
            # Number of variables (Integer value showing the number of dependent variables: the total number of columns of data is this value plus one.).
            prnt(self.nauxvar)
            # Scale factors (1 for most cases, except where grossly inconvenient) - comma delimited.
Christoph.Knote's avatar
Christoph.Knote committed
            prnt(self.splitChar.join( [ "{:6.3f}".format(x.scale) for x in self.AUXVAR ]))
Christoph.Knote's avatar
sdf
Christoph.Knote committed
            # Missing data indicators (This is -9999 (or -99999, etc.) for any missing data condition, except for the main time (independent) variable which is never missing) - comma delimited.
Christoph.Knote's avatar
Christoph.Knote committed
            prnt(self.splitChar.join( [ str(x.miss) for x in self.AUXVAR ]))
Christoph.Knote's avatar
sdf
Christoph.Knote committed
            # Variable names and units (Short variable name and units are required, and optional long descriptive name, in that order, and separated by commas. If the variable is unitless, enter the keyword "none" for its units. Each short variable name and units (and optional long name) are entered on one line. The short variable name must correspond exactly to the name used for that variable as a column header, i.e., the last header line prior to start of data.).
            nul = [ prnt(x.desc) for x in self.AUXVAR ]

        # Number of SPECIAL comment lines (Integer value indicating the number of lines of special comments, NOT including this line.).
        prnt("{:d}".format(self.nscom))
        # Special comments (Notes of problems or special circumstances unique to this file. An example would be comments/problems associated with a particular flight.).
        nul = [ prnt(x) for x in self.SCOM ]
        # Number of Normal comments (i.e., number of additional lines of SUPPORTING information: Integer value indicating the number of lines of additional information, NOT including this line.).
        prnt("{:d}".format(self.nncom))
        # Normal comments (SUPPORTING information: This is the place for investigators to more completely describe the data and measurement parameters. The supporting information structure is described below as a list of key word: value pairs. Specifically include here information on the platform used, the geo-location of data, measurement technique, and data revision comments. Note the non-optional information regarding uncertainty, the upper limit of detection (ULOD) and the lower limit of detection (LLOD) for each measured variable. The ULOD and LLOD are the values, in the same units as the measurements that correspond to the flags -7777s and -8888s within the data, respectively. The last line of this section should contain all the short variable names on one line. The key words in this section are written in BOLD below and must appear in this section of the header along with the relevant data listed after the colon. For key words where information is not needed or applicable, simply enter N/A.).
        nul = [ prnt(x) for x in self.NCOM ]
Christoph.Knote's avatar
Christoph.Knote committed
        # data!
Christoph.Knote's avatar
Christoph.Knote committed
        nul = [ prnt(self.splitChar.join( [ str(y) for y in x ] )) for x in self.data ]
Christoph.Knote's avatar
sdf
Christoph.Knote committed

Christoph.Knote's avatar
Christoph.Knote committed
    def make_filename(self):
        '''
        Create ICARTT-compliant file name based on the information contained in the dataset
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        return self.dataID + "_" +self.locationID + "_" +datetime.datetime.strftime(self.dateValid, '%Y%m%d') + "_" +"R" + self.revision + ".ict"
Christoph.Knote's avatar
sdf
Christoph.Knote committed

Christoph.Knote's avatar
Christoph.Knote committed
    # sanitize function
    def __readline(self, do_split=True):
        dmp = self.input_fhandle.readline().replace('\n', '').replace('\r','')
        if do_split:
            dmp = [word.strip(' ') for word in dmp.split(self.splitChar) ]
Christoph.Knote's avatar
Christoph.Knote committed
        return dmp

    def read_header(self):
        '''
        Read the ICARTT header (from file)
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        if self.input_fhandle.closed:
            self.input_fhandle = open(self.input_fhandle.name)

        # line 1 - Number of lines in header, file format index (most files use
        # 1001) - comma delimited.
        self.format = int(self.__readline()[1])

        # line 2 - PI last name, first name/initial.
        self.PI = self.__readline(do_split=False)

        # line 3 - Organization/affiliation of PI.
        self.organization = self.__readline(do_split=False)

        # line 4 - Data source description (e.g., instrument name, platform name,
        # model name, etc.).
        self.dataSource = self.__readline(do_split=False)

        # line 5 - Mission name (usually the mission acronym).
        self.mission = self.__readline(do_split=False)

        # line 6 - File volume number, number of file volumes (these integer values
        # are used when the data require more than one file per day; for data that
        # require only one file these values are set to 1, 1) - comma delimited.
        dmp = self.__readline()
        self.VOL  = int(dmp[0])
        self.NVOL = int(dmp[1])

        # line 7 - UTC date when data begin, UTC date of data reduction or revision
        # - comma delimited (yyyy, mm, dd, yyyy, mm, dd).
        dmp = self.__readline()
        self.dateValid   = datetime.datetime.strptime("".join([ "{:s}".format(x) for x in dmp[0:3] ]), '%Y%m%d')
        self.dateRevised = datetime.datetime.strptime("".join([ "{:s}".format(x) for x in dmp[3:6] ]), '%Y%m%d')

        # line 8 - Data Interval (This value describes the time spacing (in seconds)
        # between consecutive data records. It is the (constant) interval between
        # values of the independent variable. For 1 Hz data the data interval value
        # is 1 and for 10 Hz data the value is 0.1. All intervals longer than 1
        # second must be reported as Start and Stop times, and the Data Interval
        # value is set to 0. The Mid-point time is required when it is not at the
        # average of Start and Stop times. For additional information see Section
        # 2.5 below.).
        self.dataInterval = int(self.__readline()[0])


        # line 9 - Description or name of independent variable (This is the name
        # chosen for the start time. It always refers to the number of seconds UTC
        # from the start of the day on which measurements began. It should be noted
        # here that the independent variable should monotonically increase even when
        # crossing over to a second day.
        dmp = self.__readline()
        self.IVAR = Variable(dmp[0], dmp[1])

        # line 10 - Number of variables (Integer value showing the number of
        # dependent variables: the total number of columns of data is this value
        # plus one.).
        ndvar = int(self.__readline()[0])

        # line 11- Scale factors (1 for most cases, except where grossly
        # inconvenient) - comma delimited.
        dvscale = [ float(x) for x in self.__readline() ]

        # line 12 - Missing data indicators (This is -9999 (or -99999, etc.) for
        # any missing data condition, except for the main time (independent)
        # variable which is never missing) - comma delimited.
        dvmiss = [ x for x in self.__readline() ]
        # no float casting here, as we need to do string comparison lateron when reading data...

        # line 13 - Variable names and units (Short variable name and units are
        # required, and optional long descriptive name, in that order, and separated
        # by commas. If the variable is unitless, enter the keyword "none" for its
        # units. Each short variable name and units (and optional long name) are
        # entered on one line. The short variable name must correspond exactly to
        # the name used for that variable as a column header, i.e., the last header
        # line prior to start of data.).
        dmp = self.__readline()
        dvname     = [ dmp[0] ]
        dvunits    = [ dmp[1] ]

        for i in range(1, ndvar):
            dmp = self.__readline()
            dvname   += [ dmp[0] ]
            dvunits  += [ dmp[1] ]

        self.DVAR = [ Variable(name, unit, scale, miss) for name, unit, scale, miss in zip(dvname, dvunits, dvscale, dvmiss) ]

        # line 14 + nvar - Number of SPECIAL comment lines (Integer value
        # indicating the number of lines of special comments, NOT including this
        # line.).
        nscom = int(self.__readline()[0])

        # line 15 + nvar - Special comments (Notes of problems or special
        # circumstances unique to this file. An example would be comments/problems
        # associated with a particular flight.).
        self.SCOM          = [ self.__readline(do_split=False) for i in range(0, nscom) ]

        # line 16 + nvar + nscom - Number of Normal comments (i.e., number of
        # additional lines of SUPPORTING information: Integer value indicating the
        # number of lines of additional information, NOT including this line.).
        nncom = int(self.__readline()[0])

        # line 17 + nvar + nscom - Normal comments (SUPPORTING information: This is
        # the place for investigators to more completely describe the data and
        # measurement parameters. The supporting information structure is described
        # below as a list of key word: value pairs. Specifically include here
        # information on the platform used, the geo-location of data, measurement
        # technique, and data revision comments. Note the non-optional information
        # regarding uncertainty, the upper limit of detection (ULOD) and the lower
        # limit of detection (LLOD) for each measured variable. The ULOD and LLOD
        # are the values, in the same units as the measurements that correspond to
        # the flags -7777's and -8888's within the data, respectively. The last line
        # of this section should contain all the "short" variable names on one line.
        # The key words in this section are written in BOLD below and must appear in
        # this section of the header along with the relevant data listed after the
        # colon. For key words where information is not needed or applicable, simply
        # enter N/A.).
        self.NCOM          = [ self.__readline(do_split=False) for i in range(0, nncom) ]

        self.input_fhandle.close()

    def __nan_miss_float(self, raw):
        return [ float(x.replace(self.VAR[i].miss, 'NaN')) for i, x in enumerate(raw) ]

    def read_data(self):
        '''
        Read ICARTT data (from file)
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        if self.input_fhandle.closed:
            self.input_fhandle = open(self.input_fhandle.name)

        nul = [ self.input_fhandle.readline() for i in range(self.nheader) ]
Christoph.Knote's avatar
Christoph.Knote committed

        self.data = [ self.__nan_miss_float(line.split(self.splitChar)) for line in self.input_fhandle ]

        self.input_fhandle.close()

    def read_first_and_last(self):
        '''
        Read first and last ICARTT data line (from file). Useful for quick estimates e.g. of the time extent
        of big ICARTT files, without having to read the whole thing, which would be slow.
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        if self.input_fhandle.closed:
            self.input_fhandle = open(self.input_fhandle.name)

        nul = [ self.input_fhandle.readline() for i in range(self.nheader) ]
Christoph.Knote's avatar
Christoph.Knote committed

        first = self.input_fhandle.readline()
        self.data  = [ self.__nan_miss_float(first.split(self.splitChar)) ]
        for line in self.input_fhandle:
            pass
        last = line
        self.data += [ self.__nan_miss_float(last.split(',')) ]

        self.input_fhandle.close()

    def read(self):
Christoph.Knote's avatar
Christoph.Knote committed
        '''
        Read ICARTT data and header
        '''
Christoph.Knote's avatar
Christoph.Knote committed
        self.read_header()
        self.read_data()

Christoph.Knote's avatar
Christoph.Knote committed
    def __init__(self, f=None, loadData=True):
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        self.format       = 1001
Christoph.Knote's avatar
Christoph.Knote committed

        self.revision     = '0'
        self.dataID       = 'dataID'
        self.locationID   = 'locationID'

Christoph.Knote's avatar
sdf
Christoph.Knote committed
        self.PI           = 'Mustermann, Martin'
        self.organization = 'Musterinstitut'
        self.dataSource   = 'Musterdatenprodukt'
        self.mission      = 'MUSTEREX'
Christoph.Knote's avatar
Christoph.Knote committed
        self.VOL          = 1
        self.NVOL         = 1
Christoph.Knote's avatar
sdf
Christoph.Knote committed
        self.dateValid    = datetime.datetime.today()
        self.dateRevised  = datetime.datetime.today()
        self.dataInterval = 0
        self.IVAR         = Variable('Time_Start',
                                     'seconds_from_0_hours_on_valid_date',
                                     1.0, -9999999)
        self.DVAR         = [
                            Variable('Time_Stop',
                                     'seconds_from_0_hours_on_valid_date',
Christoph.Knote's avatar
Christoph.Knote committed
                                     1.0, -9999999),
                            Variable('Some_Variable',
                                     'ppbv',
Christoph.Knote's avatar
sdf
Christoph.Knote committed
                                     1.0, -9999999)
                            ]
        self.SCOM         = []
        self.NCOM         = []
Christoph.Knote's avatar
Christoph.Knote committed

        self.data         = [ [1.0, 2.0, 45.0], [2.0, 3.0, 36.0] ]

Christoph.Knote's avatar
sdf
Christoph.Knote committed
        # for 2210
        self.IBVAR        = None
        self.AUXVAR       = []

Christoph.Knote's avatar
Christoph.Knote committed
        self.splitChar    = ','

        # read data if f is not None
        if f is not None:
            if isinstance(f, (str, unicode)):
                self.input_fhandle = open(f, 'r')
            else:
                self.input_fhandle = f

            self.read_header()
            if loadData:
                self.read_data()