"""
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()