Source code for grass.temporal.temporal_extent

"""
Temporal extent classes

Usage:

.. code-block:: python

    >>> import grass.temporal as tgis
    >>> from datetime import datetime
    >>> tgis.init()
    >>> t = tgis.RasterRelativeTime()
    >>> t = tgis.RasterAbsoluteTime()


(C) 2012-2013 by the GRASS Development Team
This program is free software under the GNU General Public
License (>=v2). Read the file COPYING that comes with GRASS
for details.

:authors: Soeren Gebbert
"""

from .base import SQLDatabaseInterface

###############################################################################


[docs]class TemporalExtent(SQLDatabaseInterface): """This is the abstract time base class for relative and absolute time objects. It abstract class implements the interface to absolute and relative time. Absolute time is represented by datetime time stamps, relative time is represented by a unit an integer value. This class implements temporal topology relationships computation after [Allen and Ferguson 1994 Actions and Events in Interval Temporal Logic]. Usage: .. code-block:: python >>> init() >>> A = TemporalExtent( ... table="raster_absolute_time", ... ident="soil@PERMANENT", ... start_time=datetime(2001, 1, 1), ... end_time=datetime(2005, 1, 1), ... ) >>> A.id 'soil@PERMANENT' >>> A.start_time datetime.datetime(2001, 1, 1, 0, 0) >>> A.end_time datetime.datetime(2005, 1, 1, 0, 0) >>> A.print_info() | Start time:................. 2001-01-01 00:00:00 | End time:................... 2005-01-01 00:00:00 >>> A.print_shell_info() start_time='2001-01-01 00:00:00' end_time='2005-01-01 00:00:00' >>> # relative time >>> A = TemporalExtent( ... table="raster_absolute_time", ... ident="soil@PERMANENT", ... start_time=0, ... end_time=1, ... ) >>> A.id 'soil@PERMANENT' >>> A.start_time 0 >>> A.end_time 1 >>> A.print_info() | Start time:................. 0 | End time:................... 1 >>> A.print_shell_info() start_time='0' end_time='1' """ def __init__(self, table=None, ident=None, start_time=None, end_time=None) -> None: SQLDatabaseInterface.__init__(self, table, ident) self.set_id(ident) self.set_start_time(start_time) self.set_end_time(end_time)
[docs] def intersect(self, extent): """Intersect this temporal extent with the provided temporal extent and return a new temporal extent with the new start and end time :param extent: The temporal extent to intersect with :return: The new temporal extent with start and end time, or None in case of no intersection Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=6) >>> inter = A.intersect(A) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> A = TemporalExtent(start_time=5, end_time=6) >>> B = TemporalExtent(start_time=5, end_time=7) >>> inter = A.intersect(B) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> inter = B.intersect(A) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> A = TemporalExtent(start_time=3, end_time=6) >>> B = TemporalExtent(start_time=5, end_time=7) >>> inter = A.intersect(B) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> inter = B.intersect(A) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> A = TemporalExtent(start_time=3, end_time=8) >>> B = TemporalExtent(start_time=5, end_time=6) >>> inter = A.intersect(B) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> inter = B.intersect(A) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=6) >>> inter = A.intersect(B) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> inter = B.intersect(A) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> A = TemporalExtent(start_time=5, end_time=None) >>> B = TemporalExtent(start_time=3, end_time=6) >>> inter = A.intersect(B) >>> inter.print_info() | Start time:................. 5 | End time:................... None >>> inter = B.intersect(A) >>> inter.print_info() | Start time:................. 5 | End time:................... None >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=4) >>> inter = A.intersect(B) >>> print(inter) None >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=None) >>> inter = A.intersect(B) >>> print(inter) None """ relation = self.temporal_relation(extent) if relation in {"after", "before"}: return None if self.D["end_time"] is None: return TemporalExtent(start_time=self.D["start_time"]) if extent.D["end_time"] is None: return TemporalExtent(start_time=extent.D["start_time"]) start = None end = None if self.D["start_time"] > extent.D["start_time"]: start = self.D["start_time"] else: start = extent.D["start_time"] if self.D["end_time"] > extent.D["end_time"]: end = extent.D["end_time"] else: end = self.D["end_time"] if issubclass(type(self), RelativeTemporalExtent): return RelativeTemporalExtent( start_time=start, end_time=end, unit=self.get_unit() ) if issubclass(type(self), AbsoluteTemporalExtent): return AbsoluteTemporalExtent(start_time=start, end_time=end) if issubclass(type(self), TemporalExtent): return TemporalExtent(start_time=start, end_time=end)
[docs] def disjoint_union(self, extent): """Creates a disjoint union with this temporal extent and the provided one. Return a new temporal extent with the new start and end time. :param extent: The temporal extent to create a union with :return: The new temporal extent with start and end time Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=6) >>> inter = A.intersect(A) >>> inter.print_info() | Start time:................. 5 | End time:................... 6 >>> A = TemporalExtent(start_time=5, end_time=6) >>> B = TemporalExtent(start_time=5, end_time=7) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 5 | End time:................... 7 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 5 | End time:................... 7 >>> A = TemporalExtent(start_time=3, end_time=6) >>> B = TemporalExtent(start_time=5, end_time=7) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 7 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 7 >>> A = TemporalExtent(start_time=3, end_time=8) >>> B = TemporalExtent(start_time=5, end_time=6) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=6) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> A = TemporalExtent(start_time=5, end_time=None) >>> B = TemporalExtent(start_time=3, end_time=6) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 6 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 6 >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=4) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=None) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> A = TemporalExtent(start_time=5, end_time=None) >>> B = TemporalExtent(start_time=3, end_time=8) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 8 >>> A = TemporalExtent(start_time=5, end_time=None) >>> B = TemporalExtent(start_time=3, end_time=None) >>> inter = A.disjoint_union(B) >>> inter.print_info() | Start time:................. 3 | End time:................... 5 >>> inter = B.disjoint_union(A) >>> inter.print_info() | Start time:................. 3 | End time:................... 5 >>> A = RelativeTemporalExtent(start_time=5, end_time=None, unit="years") >>> B = RelativeTemporalExtent(start_time=3, end_time=None, unit="years") >>> inter = A.disjoint_union(B) >>> inter.print_info() +-------------------- Relative time -----------------------------------------+ | Start time:................. 3 | End time:................... 5 | Relative time unit:......... years >>> inter = B.disjoint_union(A) >>> inter.print_info() +-------------------- Relative time -----------------------------------------+ | Start time:................. 3 | End time:................... 5 | Relative time unit:......... years >>> from datetime import datetime as dt >>> A = AbsoluteTemporalExtent( ... start_time=dt(2001, 1, 10), end_time=dt(2003, 1, 1) ... ) >>> B = AbsoluteTemporalExtent( ... start_time=dt(2005, 1, 10), end_time=dt(2008, 1, 1) ... ) >>> inter = A.disjoint_union(B) >>> inter.print_info() +-------------------- Absolute time -----------------------------------------+ | Start time:................. 2001-01-10 00:00:00 | End time:................... 2008-01-01 00:00:00 >>> inter = B.disjoint_union(A) >>> inter.print_info() +-------------------- Absolute time -----------------------------------------+ | Start time:................. 2001-01-10 00:00:00 | End time:................... 2008-01-01 00:00:00 """ # noqa: E501 start = None end = None if self.D["start_time"] < extent.D["start_time"]: start = self.D["start_time"] else: start = extent.D["start_time"] # End time handling if self.D["end_time"] is None and extent.D["end_time"] is None: if self.D["start_time"] > extent.D["start_time"]: end = self.D["start_time"] else: end = extent.D["start_time"] elif self.D["end_time"] is None: if self.D["start_time"] > extent.D["end_time"]: end = self.D["start_time"] else: end = extent.D["end_time"] elif extent.D["end_time"] is None: if self.D["end_time"] > extent.D["start_time"]: end = self.D["end_time"] else: end = extent.D["start_time"] elif self.D["end_time"] < extent.D["end_time"]: end = extent.D["end_time"] else: end = self.D["end_time"] if issubclass(type(self), RelativeTemporalExtent): return RelativeTemporalExtent( start_time=start, end_time=end, unit=self.get_unit() ) if issubclass(type(self), AbsoluteTemporalExtent): return AbsoluteTemporalExtent(start_time=start, end_time=end) if issubclass(type(self), TemporalExtent): return TemporalExtent(start_time=start, end_time=end)
[docs] def union(self, extent): """Creates a union with this temporal extent and the provided one. Return a new temporal extent with the new start and end time. :param extent: The temporal extent to create a union with :return: The new temporal extent with start and end time, or None in case the temporal extents are unrelated (before or after) .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=4) >>> inter = A.intersect(B) >>> print(inter) None >>> A = TemporalExtent(start_time=5, end_time=8) >>> B = TemporalExtent(start_time=3, end_time=None) >>> inter = A.intersect(B) >>> print(inter) None """ relation = self.temporal_relation(extent) if relation in {"after", "before"}: return None return self.disjoint_union(extent)
[docs] def starts(self, extent) -> bool: """Return True if this temporal extent (A) starts at the start of the provided temporal extent (B) and finishes within it :: A |-----| B |---------| :param extent: The temporal extent object with which this extent starts Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=6) >>> B = TemporalExtent(start_time=5, end_time=7) >>> A.starts(B) True >>> B.starts(A) False """ if self.D["end_time"] is None or extent.D["end_time"] is None: return False return bool( self.D["start_time"] == extent.D["start_time"] and self.D["end_time"] < extent.D["end_time"] )
[docs] def started(self, extent) -> bool: """Return True if this temporal extent (A) started at the start of the provided temporal extent (B) and finishes after it :: A |---------| B |-----| :param extent: The temporal extent object with which this extent started Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=5, end_time=6) >>> A.started(B) True >>> B.started(A) False """ if self.D["end_time"] is None or extent.D["end_time"] is None: return False return bool( self.D["start_time"] == extent.D["start_time"] and self.D["end_time"] > extent.D["end_time"] )
[docs] def finishes(self, extent) -> bool: """Return True if this temporal extent (A) starts after the start of the provided temporal extent (B) and finishes with it :: A |-----| B |---------| :param extent: The temporal extent object with which this extent finishes Usage: .. code-block:: python >>> A = TemporalExtent(start_time=6, end_time=7) >>> B = TemporalExtent(start_time=5, end_time=7) >>> A.finishes(B) True >>> B.finishes(A) False """ if self.D["end_time"] is None or extent.D["end_time"] is None: return False return bool( self.D["end_time"] == extent.D["end_time"] and self.D["start_time"] > extent.D["start_time"] )
[docs] def finished(self, extent) -> bool: """Return True if this temporal extent (A) starts before the start of the provided temporal extent (B) and finishes with it :: A |---------| B |-----| :param extent: The temporal extent object with which this extent finishes Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=6, end_time=7) >>> A.finished(B) True >>> B.finished(A) False """ if self.D["end_time"] is None or extent.D["end_time"] is None: return False return bool( self.D["end_time"] == extent.D["end_time"] and self.D["start_time"] < extent.D["start_time"] )
[docs] def after(self, extent) -> bool: """Return True if this temporal extent (A) is located after the provided temporal extent (B) :: A |---------| B |---------| :param extent: The temporal extent object that is located before this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=8, end_time=9) >>> B = TemporalExtent(start_time=6, end_time=7) >>> A.after(B) True >>> B.after(A) False """ if extent.D["end_time"] is None: return self.D["start_time"] > extent.D["start_time"] return self.D["start_time"] > extent.D["end_time"]
[docs] def before(self, extent): """Return True if this temporal extent (A) is located before the provided temporal extent (B) :: A |---------| B |---------| :param extent: The temporal extent object that is located after this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=6, end_time=7) >>> B = TemporalExtent(start_time=8, end_time=9) >>> A.before(B) True >>> B.before(A) False """ if self.D["end_time"] is None: return self.D["start_time"] < extent.D["start_time"] return self.D["end_time"] < extent.D["start_time"]
[docs] def adjacent(self, extent) -> bool: """Return True if this temporal extent (A) is a meeting neighbor the provided temporal extent (B) :: A |---------| B |---------| A |---------| B |---------| :param extent: The temporal extent object that is a meeting neighbor of this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=7, end_time=9) >>> A.adjacent(B) True >>> B.adjacent(A) True >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=3, end_time=5) >>> A.adjacent(B) True >>> B.adjacent(A) True """ if self.D["end_time"] is None and extent.D["end_time"] is None: return False return bool( self.D["start_time"] is not None and extent.D["end_time"] is not None and ( self.D["start_time"] == extent.D["end_time"] or self.D["end_time"] == extent.D["start_time"] ) )
[docs] def follows(self, extent) -> bool: """Return True if this temporal extent (A) follows the provided temporal extent (B) :: A |---------| B |---------| :param extent: The temporal extent object that is the predecessor of this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=3, end_time=5) >>> A.follows(B) True >>> B.follows(A) False """ return ( extent.D["end_time"] is not None and self.D["start_time"] == extent.D["end_time"] )
[docs] def precedes(self, extent) -> bool: """Return True if this temporal extent (A) precedes the provided temporal extent (B) :: A |---------| B |---------| :param extent: The temporal extent object that is the successor of this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=7, end_time=9) >>> A.precedes(B) True >>> B.precedes(A) False """ return ( self.D["end_time"] is not None and self.D["end_time"] == extent.D["start_time"] )
[docs] def during(self, extent) -> bool: """Return True if this temporal extent (A) is located during the provided temporal extent (B) :: A |-------| B |---------| :param extent: The temporal extent object that contains this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=4, end_time=9) >>> A.during(B) True >>> B.during(A) False """ # Check single point of time in interval if extent.D["end_time"] is None: return False # Check single point of time in interval if self.D["end_time"] is None: return bool( self.D["start_time"] >= extent.D["start_time"] and self.D["start_time"] < extent.D["end_time"] ) return bool( self.D["start_time"] > extent.D["start_time"] and self.D["end_time"] < extent.D["end_time"] )
[docs] def contains(self, extent) -> bool: """Return True if this temporal extent (A) contains the provided temporal extent (B) :: A |---------| B |-------| :param extent: The temporal extent object that is located during this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=4, end_time=9) >>> B = TemporalExtent(start_time=5, end_time=8) >>> A.contains(B) True >>> B.contains(A) False """ # Check single point of time in interval if self.D["end_time"] is None: return False # Check single point of time in interval if extent.D["end_time"] is None: return bool( self.D["start_time"] <= extent.D["start_time"] and self.D["end_time"] > extent.D["start_time"] ) return bool( self.D["start_time"] < extent.D["start_time"] and self.D["end_time"] > extent.D["end_time"] )
[docs] def equal(self, extent) -> bool: """Return True if this temporal extent (A) is equal to the provided temporal extent (B) :: A |---------| B |---------| :param extent: The temporal extent object that is equal during this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=6) >>> B = TemporalExtent(start_time=5, end_time=6) >>> A.equal(B) True >>> B.equal(A) True """ if self.D["end_time"] is None and extent.D["end_time"] is None: return self.D["start_time"] == extent.D["start_time"] if self.D["end_time"] is None or extent.D["end_time"] is None: return False return bool( self.D["start_time"] == extent.D["start_time"] and self.D["end_time"] == extent.D["end_time"] )
[docs] def overlaps(self, extent) -> bool: """Return True if this temporal extent (A) overlapped the provided temporal extent (B) :: A |---------| B |---------| :param extent: The temporal extent object that is overlaps this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=5, end_time=7) >>> B = TemporalExtent(start_time=6, end_time=8) >>> A.overlaps(B) True >>> B.overlaps(A) False >>> A = TemporalExtent(start_time=5, end_time=6) >>> B = TemporalExtent(start_time=6, end_time=8) >>> A.overlaps(B) False >>> B.overlaps(A) False """ return bool( self.D["end_time"] is not None and extent.D["end_time"] is not None and self.D["start_time"] < extent.D["start_time"] and self.D["end_time"] < extent.D["end_time"] and self.D["end_time"] > extent.D["start_time"] )
[docs] def overlapped(self, extent) -> bool: """Return True if this temporal extent (A) overlaps the provided temporal extent (B) :: A |---------| B |---------| :param extent: The temporal extent object that is overlapped this extent Usage: .. code-block:: python >>> A = TemporalExtent(start_time=6, end_time=8) >>> B = TemporalExtent(start_time=5, end_time=7) >>> A.overlapped(B) True >>> B.overlapped(A) False >>> A = TemporalExtent(start_time=6, end_time=8) >>> B = TemporalExtent(start_time=5, end_time=6) >>> A.overlapped(B) False >>> B.overlapped(A) False """ return bool( self.D["end_time"] is not None and extent.D["end_time"] is not None and self.D["start_time"] > extent.D["start_time"] and self.D["end_time"] > extent.D["end_time"] and self.D["start_time"] < extent.D["end_time"] )
[docs] def temporal_relation(self, extent): """Returns the temporal relation between temporal objects Temporal relationships are implemented after [Allen and Ferguson 1994 Actions and Events in Interval Temporal Logic] The following temporal relationships are supported: - equal - during - contains - overlaps - overlapped - after - before - starts - finishes - started - finished - follows - precedes :param extent: The temporal extent :return: The name of the temporal relation or None if no relation found """ # First check for correct time if "start_time" not in self.D: return None if "end_time" not in self.D: return None if "start_time" not in extent.D: return None if "end_time" not in extent.D: return None # Return None if the start_time is undefined if self.D["start_time"] is None or extent.D["start_time"] is None: return None if self.equal(extent): return "equal" if self.during(extent): return "during" if self.contains(extent): return "contains" if self.overlaps(extent): return "overlaps" if self.overlapped(extent): return "overlapped" if self.after(extent): return "after" if self.before(extent): return "before" if self.starts(extent): return "starts" if self.finishes(extent): return "finishes" if self.started(extent): return "started" if self.finished(extent): return "finished" if self.follows(extent): return "follows" if self.precedes(extent): return "precedes" return None
[docs] def set_id(self, ident) -> None: """Convenient method to set the unique identifier (primary key)""" self.ident = ident self.D["id"] = ident
[docs] def set_start_time(self, start_time) -> None: """Set the valid start time of the extent""" self.D["start_time"] = start_time
[docs] def set_end_time(self, end_time) -> None: """Set the valid end time of the extent""" self.D["end_time"] = end_time
[docs] def get_id(self): """Convenient method to get the unique identifier (primary key) :return: None if not found """ if "id" in self.D: return self.D["id"] return None
[docs] def get_start_time(self): """Get the valid start time of the extent :return: None if not found""" if "start_time" in self.D: return self.D["start_time"] return None
[docs] def get_end_time(self): """Get the valid end time of the extent :return: None if not found""" if "end_time" in self.D: return self.D["end_time"] return None
# Set the properties id = property(fget=get_id, fset=set_id) start_time = property(fget=get_start_time, fset=set_start_time) end_time = property(fget=get_end_time, fset=set_end_time)
[docs] def print_info(self) -> None: """Print information about this class in human readable style""" # 0123456789012345678901234567890 print(" | Start time:................. " + str(self.get_start_time())) print(" | End time:................... " + str(self.get_end_time()))
[docs] def print_shell_info(self) -> None: """Print information about this class in shell style""" print("start_time='{}'".format(str(self.get_start_time()))) print("end_time='{}'".format(str(self.get_end_time())))
###############################################################################
[docs]class AbsoluteTemporalExtent(TemporalExtent): """This is the absolute time class for all maps and spacetime datasets start_time and end_time must be of type datetime """ def __init__(self, table=None, ident=None, start_time=None, end_time=None) -> None: TemporalExtent.__init__(self, table, ident, start_time, end_time)
[docs] def print_info(self) -> None: """Print information about this class in human readable style""" # 0123456789012345678901234567890 print( " +-------------------- Absolute time -----------------------------------------+" # noqa: E501 ) TemporalExtent.print_info(self)
[docs] def print_shell_info(self) -> None: """Print information about this class in shell style""" TemporalExtent.print_shell_info(self)
###############################################################################
[docs]class RasterAbsoluteTime(AbsoluteTemporalExtent): def __init__(self, ident=None, start_time=None, end_time=None) -> None: AbsoluteTemporalExtent.__init__( self, "raster_absolute_time", ident, start_time, end_time )
[docs]class Raster3DAbsoluteTime(AbsoluteTemporalExtent): def __init__(self, ident=None, start_time=None, end_time=None) -> None: AbsoluteTemporalExtent.__init__( self, "raster3d_absolute_time", ident, start_time, end_time )
[docs]class VectorAbsoluteTime(AbsoluteTemporalExtent): def __init__(self, ident=None, start_time=None, end_time=None) -> None: AbsoluteTemporalExtent.__init__( self, "vector_absolute_time", ident, start_time, end_time )
###############################################################################
[docs]class STDSAbsoluteTime(AbsoluteTemporalExtent): """This class implements the absolute time extent for space time dataset In addition to the existing functionality the granularity and the map_time are added. Usage: .. code-block:: python >>> init() >>> A = STDSAbsoluteTime( ... table="strds_absolute_time", ... ident="strds@PERMANENT", ... start_time=datetime(2001, 1, 1), ... end_time=datetime(2005, 1, 1), ... granularity="1 days", ... map_time="interval", ... ) >>> A.id 'strds@PERMANENT' >>> A.start_time datetime.datetime(2001, 1, 1, 0, 0) >>> A.end_time datetime.datetime(2005, 1, 1, 0, 0) >>> A.granularity '1 days' >>> A.map_time 'interval' >>> A.print_info() +-------------------- Absolute time -----------------------------------------+ | Start time:................. 2001-01-01 00:00:00 | End time:................... 2005-01-01 00:00:00 | Granularity:................ 1 days | Temporal type of maps:...... interval >>> A.print_shell_info() start_time='2001-01-01 00:00:00' end_time='2005-01-01 00:00:00' granularity='1 days' map_time=interval """ def __init__( self, table=None, ident=None, start_time=None, end_time=None, granularity=None, map_time=None, ) -> None: AbsoluteTemporalExtent.__init__(self, table, ident, start_time, end_time) self.set_granularity(granularity) self.set_map_time(map_time)
[docs] def set_granularity(self, granularity) -> None: """Set the granularity of the space time dataset""" self.D["granularity"] = granularity
[docs] def set_map_time(self, map_time) -> None: """Set the type of the map time Registered maps may have different types of time: - Single point of time "point" - Time intervals "interval" - Single point and interval time "mixed" This variable will be set automatically when maps are registered. """ self.D["map_time"] = map_time
[docs] def get_granularity(self): """Get the granularity of the space time dataset :return: None if not found""" if "granularity" in self.D: return self.D["granularity"] return None
[docs] def get_map_time(self): """Get the type of the map time Registered maps may have different types of time: - Single point of time "point" - Time intervals "interval" - Single point and interval time "mixed" This variable will be set automatically when maps are registered. """ if "map_time" in self.D: return self.D["map_time"] return None
# Properties granularity = property(fget=get_granularity, fset=set_granularity) map_time = property(fget=get_map_time, fset=set_map_time)
[docs] def print_info(self) -> None: """Print information about this class in human readable style""" AbsoluteTemporalExtent.print_info(self) # 0123456789012345678901234567890 print(" | Granularity:................ " + str(self.get_granularity())) print(" | Temporal type of maps:...... " + str(self.get_map_time()))
[docs] def print_shell_info(self) -> None: """Print information about this class in shell style""" AbsoluteTemporalExtent.print_shell_info(self) print("granularity='{}'".format(str(self.get_granularity()))) print("map_time=" + str(self.get_map_time()))
###############################################################################
[docs]class STRDSAbsoluteTime(STDSAbsoluteTime): def __init__( self, ident=None, start_time=None, end_time=None, granularity=None ) -> None: STDSAbsoluteTime.__init__( self, "strds_absolute_time", ident, start_time, end_time, granularity )
[docs]class STR3DSAbsoluteTime(STDSAbsoluteTime): def __init__( self, ident=None, start_time=None, end_time=None, granularity=None ) -> None: STDSAbsoluteTime.__init__( self, "str3ds_absolute_time", ident, start_time, end_time, granularity )
[docs]class STVDSAbsoluteTime(STDSAbsoluteTime): def __init__( self, ident=None, start_time=None, end_time=None, granularity=None ) -> None: STDSAbsoluteTime.__init__( self, "stvds_absolute_time", ident, start_time, end_time, granularity )
###############################################################################
[docs]class RelativeTemporalExtent(TemporalExtent): """This is the relative time class for all maps and space time datasets start_time and end_time must be of type integer Usage: .. code-block:: python >>> init() >>> A = RelativeTemporalExtent( ... table="raster_relative_time", ... ident="soil@PERMANENT", ... start_time=0, ... end_time=1, ... unit="years", ... ) >>> A.id 'soil@PERMANENT' >>> A.start_time 0 >>> A.end_time 1 >>> A.unit 'years' >>> A.print_info() +-------------------- Relative time -----------------------------------------+ | Start time:................. 0 | End time:................... 1 | Relative time unit:......... years >>> A.print_shell_info() start_time='0' end_time='1' unit=years """ def __init__( self, table=None, ident=None, start_time=None, end_time=None, unit=None ) -> None: TemporalExtent.__init__(self, table, ident, start_time, end_time) self.set_unit(unit)
[docs] def set_unit(self, unit) -> None: """Set the unit of the relative time. Valid units are: - years - months - days - hours - minutes - seconds """ self.D["unit"] = unit
[docs] def get_unit(self): """Get the unit of the relative time :return: None if not found""" if "unit" in self.D: return self.D["unit"] return None
[docs] def temporal_relation(self, map): """Returns the temporal relation between temporal objects Temporal relationships are implemented after [Allen and Ferguson 1994 Actions and Events in Interval Temporal Logic] """ # Check units for relative time if "unit" not in self.D: return None if "unit" not in map.D: return None # Units must be equal if self.D["unit"] != map.D["unit"]: return None return TemporalExtent.temporal_relation(self, map)
# Properties unit = property(fget=get_unit, fset=set_unit)
[docs] def print_info(self) -> None: """Print information about this class in human readable style""" # 0123456789012345678901234567890 print( " +-------------------- Relative time -----------------------------------------+" # noqa: E501 ) TemporalExtent.print_info(self) print(" | Relative time unit:......... " + str(self.get_unit()))
[docs] def print_shell_info(self) -> None: """Print information about this class in shell style""" TemporalExtent.print_shell_info(self) print("unit=" + str(self.get_unit()))
###############################################################################
[docs]class RasterRelativeTime(RelativeTemporalExtent): def __init__(self, ident=None, start_time=None, end_time=None, unit=None) -> None: RelativeTemporalExtent.__init__( self, "raster_relative_time", ident, start_time, end_time, unit )
[docs]class Raster3DRelativeTime(RelativeTemporalExtent): def __init__(self, ident=None, start_time=None, end_time=None, unit=None) -> None: RelativeTemporalExtent.__init__( self, "raster3d_relative_time", ident, start_time, end_time, unit )
[docs]class VectorRelativeTime(RelativeTemporalExtent): def __init__(self, ident=None, start_time=None, end_time=None, unit=None) -> None: RelativeTemporalExtent.__init__( self, "vector_relative_time", ident, start_time, end_time, unit )
###############################################################################
[docs]class STDSRelativeTime(RelativeTemporalExtent): """This is the relative time class for all maps and space time datasets start_time and end_time must be of type integer Usage: .. code-block:: python >>> init() >>> A = STDSRelativeTime( ... table="strds_relative_time", ... ident="strds@PERMANENT", ... start_time=0, ... end_time=1, ... unit="years", ... granularity=5, ... map_time="interval", ... ) >>> A.id 'strds@PERMANENT' >>> A.start_time 0 >>> A.end_time 1 >>> A.unit 'years' >>> A.granularity 5 >>> A.map_time 'interval' >>> A.print_info() +-------------------- Relative time -----------------------------------------+ | Start time:................. 0 | End time:................... 1 | Relative time unit:......... years | Granularity:................ 5 | Temporal type of maps:...... interval >>> A.print_shell_info() start_time='0' end_time='1' unit=years granularity=5 map_time=interval """ def __init__( self, table=None, ident=None, start_time=None, end_time=None, unit=None, granularity=None, map_time=None, ) -> None: RelativeTemporalExtent.__init__(self, table, ident, start_time, end_time, unit) self.set_granularity(granularity) self.set_map_time(map_time)
[docs] def set_granularity(self, granularity) -> None: """Set the granularity of the space time dataset""" self.D["granularity"] = granularity
[docs] def set_map_time(self, map_time) -> None: """Set the type of the map time Registered maps may have different types of time: - Single point of time "point" - Time intervals "interval" - Single point and interval time "mixed" This variable will be set automatically when maps are registered. """ self.D["map_time"] = map_time
[docs] def get_granularity(self): """Get the granularity of the space time dataset :return: None if not found""" if "granularity" in self.D: return self.D["granularity"] return None
[docs] def get_map_time(self): """Get the type of the map time Registered maps may have different types of time: - Single point of time "point" - Time intervals "interval" - Single point and interval time "mixed" This variable will be set automatically when maps are registered. """ if "map_time" in self.D: return self.D["map_time"] return None
# Properties granularity = property(fget=get_granularity, fset=set_granularity) map_time = property(fget=get_map_time, fset=set_map_time)
[docs] def print_info(self) -> None: """Print information about this class in human readable style""" RelativeTemporalExtent.print_info(self) # 0123456789012345678901234567890 print(" | Granularity:................ " + str(self.get_granularity())) print(" | Temporal type of maps:...... " + str(self.get_map_time()))
[docs] def print_shell_info(self) -> None: """Print information about this class in shell style""" RelativeTemporalExtent.print_shell_info(self) print("granularity=" + str(self.get_granularity())) print("map_time=" + str(self.get_map_time()))
###############################################################################
[docs]class STRDSRelativeTime(STDSRelativeTime): def __init__( self, ident=None, start_time=None, end_time=None, unit=None, granularity=None, map_time=None, ) -> None: STDSRelativeTime.__init__( self, "strds_relative_time", ident, start_time, end_time, unit, granularity, map_time, )
[docs]class STR3DSRelativeTime(STDSRelativeTime): def __init__( self, ident=None, start_time=None, end_time=None, unit=None, granularity=None, map_time=None, ) -> None: STDSRelativeTime.__init__( self, "str3ds_relative_time", ident, start_time, end_time, unit, granularity, map_time, )
[docs]class STVDSRelativeTime(STDSRelativeTime): def __init__( self, ident=None, start_time=None, end_time=None, unit=None, granularity=None, map_time=None, ) -> None: STDSRelativeTime.__init__( self, "stvds_relative_time", ident, start_time, end_time, unit, granularity, map_time, )
############################################################################### if __name__ == "__main__": import doctest doctest.testmod()