Source code for pystac.extensions.label

"""STAC Model classes for Label extension.
"""
from enum import Enum

from pystac import STACError
from pystac.extensions import Extensions
from pystac.extensions.base import (ItemExtension, ExtensionDefinition, ExtendedObject)
from pystac.item import (Item, Asset)
from pystac.link import Link


[docs]class LabelType(str, Enum): """Enumerates valid label types (RASTER or VECTOR).""" def __str__(self): return str(self.value) VECTOR = 'vector' RASTER = 'raster' ALL = [VECTOR, RASTER] """Convenience attribute for checking if values are valid label types"""
[docs]class LabelItemExt(ItemExtension): """A LabelItemExt is the extension of the Item in the label extension which represents a polygon, set of polygons, or raster data defining labels and label metadata and should be part of a Collection. Args: item (Item): The item to be extended. Attributes: item (Item): The Item that is being extended. See: `Item fields in the label extension spec <https://github.com/radiantearth/stac-spec/tree/v0.8.1/extensions/label#item-fields>`_ Note: Using LabelItemExt to directly wrap an item will add the 'label' extension ID to the item's stac_extensions. """ # noqa E501 def __init__(self, item): if item.stac_extensions is None: item.stac_extensions = [Extensions.LABEL] elif Extensions.LABEL not in item.stac_extensions: item.stac_extensions.append(Extensions.LABEL) self.item = item
[docs] def apply(self, label_description, label_type, label_properties=None, label_classes=None, label_tasks=None, label_methods=None, label_overviews=None): """Applies label extension properties to the extended Item. Args: label_description (str): A description of the label, how it was created, and what it is recommended for label_type (str): An ENUM of either vector label type or raster label type. Use one of :class:`~pystac.LabelType`. label_properties (list or None): These are the names of the property field(s) in each Feature of the label asset's FeatureCollection that contains the classes (keywords from label:classes if the property defines classes). If labels are rasters, this should be None. label_classes (List[LabelClass]): Optional, but reqiured if ussing categorical data. A list of LabelClasses defining the list of possible class names for each label:properties. (e.g., tree, building, car, hippo) label_tasks (List[str]): Recommended to be a subset of 'regression', 'classification', 'detection', or 'segmentation', but may be an arbitrary value. label_methods: Recommended to be a subset of 'automated' or 'manual', but may be an arbitrary value. label_overviews (List[LabelOverview]): Optional list of LabelOverview classes that store counts (for classification-type data) or summary statistics (for continuous numerical/regression data). """ # noqa E501 self.label_description = label_description self.label_type = label_type self.label_properties = label_properties self.label_classes = label_classes self.label_tasks = label_tasks self.label_methods = label_methods self.label_overviews = label_overviews
@property def label_description(self): """Get or sets a description of the label, how it was created, and what it is recommended for. Returns: str """ return self.item.properties.get('label:description') @label_description.setter def label_description(self, v): self.item.properties['label:description'] = v @property def label_type(self): """Gets or sets an ENUM of either vector label type or raster label type (one of :class:`~pystac.LabelType`). Returns: str """ return self.item.properties.get('label:type') @label_type.setter def label_type(self, v): if v not in LabelType.ALL: raise STACError("label_type must be one of " "{}. Invalid input: {}".format(LabelType.ALL, v)) self.item.properties['label:type'] = v @property def label_properties(self): """Label Properties Gets or sets the names of the property field(s) in each Feature of the label asset's FeatureCollection that contains the classes (keywords from label:classes if the property defines classes). If labels are rasters, this should be None. Returns: List[str] or None """ return self.item.properties.get('label:properties') @label_properties.setter def label_properties(self, v): if v is not None: if not type(v) is list: raise STACError("label_properties must be a list! Invalid input: {}".format(v)) self.item.properties['label:properties'] = v @property def label_classes(self): """Get or set a list of LabelClasses defining the list of possible class names for each label:properties. (e.g., tree, building, car, hippo). Optional, but reqiured if using categorical data. Returns: List[LabelClasses] or None """ label_classes = self.item.properties.get('label:classes') if label_classes is not None: return [LabelClasses(classes) for classes in label_classes] else: return None @label_classes.setter def label_classes(self, v): if v is None: self.item.properties.pop('label:classes', None) else: if not type(v) is list: raise STACError("label_classes must be a list! Invalid input: {}".format(v)) classes = [x.to_dict() for x in v] self.item.properties['label:classes'] = classes @property def label_tasks(self): """Get or set a list of tasks these labels apply to. Usually a subset of 'regression', 'classification', 'detection', or 'segmentation', but may be arbitrary values. Returns: List[str] or None """ return self.item.properties.get('label:tasks') @label_tasks.setter def label_tasks(self, v): if v is None: self.item.properties.pop('label:tasks', None) else: if not type(v) is list: raise STACError("label_tasks must be a list! Invalid input: {}".format(v)) self.item.properties['label:tasks'] = v @property def label_methods(self): """Get or set a list of methods used for labeling. Usually a subset of 'automated' or 'manual', but may be arbitrary values. Returns: List[str] or None """ return self.item.properties.get('label:methods') @label_methods.setter def label_methods(self, v): if v is None: self.item.properties.pop('label:methods', None) else: if not type(v) is list: raise STACError("label_methods must be a list! Invalid input: {}".format(v)) self.item.properties['label:methods'] = v @property def label_overviews(self): """Get or set a list of LabelOverview classes that store counts (for classification-type data) or summary statistics (for continuous numerical/regression data). Returns: List[LabelOverview] or None """ overviews = self.item.properties.get('label:overviews') if overviews is not None: return [LabelOverview(overview) for overview in overviews] else: return None @label_overviews.setter def label_overviews(self, v): if v is None: self.item.properties.pop('label:overviews', None) else: if not type(v) is list: raise STACError("label_overviews must be a list! Invalid input: {}".format(v)) overviews = [x.to_dict() for x in v] self.item.properties['label:overviews'] = overviews def __repr__(self): return '<LabelItemExt Item id={}>'.format(self.item.id)
[docs] def add_source(self, source_item, title=None, assets=None): """Adds a link to a source item. Args: source_item (Item): Source imagery that the LabelItem applies to. title (str): Optional title for the link. assets (List[str]): Optional list of assets that determine what assets in the source item this label item data applies to. """ properties = None if assets is not None: properties = {'label:assets': assets} link = Link('source', source_item, title=title, media_type='application/json', properties=properties) self.item.add_link(link)
[docs] def get_sources(self): """Gets any source items that describe the source imagery used to generate this LabelItem. Returns: Generator[Items]: A possibly empty list of source imagery items. Determined by links of this LabelItem that have ``rel=='source'``. """ return self.item.get_stac_objects('source')
[docs] def add_labels(self, href, title=None, media_type=None, properties=None): """Adds a label asset to this LabelItem. Args: href (str): Link to the asset object. Relative and absolute links are both allowed. title (str): Optional displayed title for clients and users. media_type (str): Optional description of the media type. Registered Media Types are preferred. See :class:`~pystac.MediaType` for common media types. properties (dict): Optional, additional properties for this asset. This is used by extensions as a way to serialize and deserialize properties on asset object JSON. """ self.item.add_asset( "labels", Asset(href=href, title=title, media_type=media_type, properties=properties))
[docs] def add_geojson_labels(self, href, title=None, properties=None): """Adds a GeoJSON label asset to this LabelItem. Args: href (str): Link to the asset object. Relative and absolute links are both allowed. title (str): Optional displayed title for clients and users. properties (dict): Optional, additional properties for this asset. This is used by extensions as a way to serialize and deserialize properties on asset object JSON. """ self.add_labels(href, title=title, properties=properties, media_type='application/geo+json')
@classmethod def _object_links(cls): return ['source']
[docs] @classmethod def from_item(cls, item): return cls(item)
[docs]class LabelClasses: """Defines the list of possible class names (e.g., tree, building, car, hippo) Use LabelClasses.create to create a new instance of LabelClasses from property values. """ def __init__(self, properties): self.properties = properties
[docs] def apply(self, classes, name=None): """Sets the properties for this LabelClasses. Args: classes (List[str] or List[int] or List[float]): The different possible class values. name (str): The property key within the asset's each Feature corresponding to class labels. If labels are raster-formatted, do not supply; required otherwise. """ self.classes = classes self.name = name
[docs] @classmethod def create(cls, classes, name=None): """Creates a new LabelClasses. Args: classes (List[str] or List[int] or List[float]): The different possible class values. name (str): The property key within the asset's each Feature corresponding to class labels. If labels are raster-formatted, do not supply; required otherwise. Returns: LabelClasses """ c = cls({}) c.apply(classes, name) return c
@property def classes(self): """Get or sets the class values. Returns: List[str] or List[int] or List[float] """ return self.properties.get('classes') @classes.setter def classes(self, v): if not type(v) is list: raise STACError("classes must be a list! Invalid input: {}".format(v)) self.properties['classes'] = v @property def name(self): """Get or sets the property key within the asset's each Feature corresponding to class labels. If labels are raster-formatted, do not supply; required otherwise. Returns: str """ return self.properties.get('name') @name.setter def name(self, v): if v is not None: self.properties['name'] = v else: self.properties.pop('name', None) def __repr__(self): return '<LabelClasses classes={}>'.format(','.join(self.classes))
[docs] def to_dict(self): """Returns the dictionary representing the JSON of this LabelClasses. Returns: dict: The wrapped dict of the LabelClasses that can be written out as JSON. """ return self.properties
[docs]class LabelOverview: """Stores counts (for classification-type data) or summary statistics (for continuous numerical/regression data). Use LabelOverview.create to create a new LabelOverview. """ def __init__(self, properties): self.properties = properties
[docs] def apply(self, property_key, counts=None, statistics=None): """Sets the properties for this LabelOverview. Either ``counts`` or ``statistics``, or both, can be placed in an overview; at least one is required. Args: property_key (str): The property key within the asset corresponding to class labels. counts (List[LabelCounts]): Optional list of LabelCounts containing counts for categorical data. statistics (List[Statistics]): Optional list of Statistics containing statistics for regression/continuous numeric value data. """ self.property_key = property_key self.counts = counts self.statistics = statistics
[docs] @classmethod def create(cls, property_key, counts=None, statistics=None): """Creates a new LabelOverview. Either ``counts`` or ``statistics``, or both, can be placed in an overview; at least one is required. Args: property_key (str): The property key within the asset corresponding to class labels. counts (List[LabelCounts]): Optional list of LabelCounts containing counts for categorical data. statistics (List[Statistics]): Optional list of Statistics containing statistics for regression/continuous numeric value data. """ x = LabelOverview({}) x.apply(property_key, counts, statistics) return x
@property def property_key(self): """Get or sets the property key within the asset corresponding to class labels. Returns: str """ return self.properties.get('property_key') @property_key.setter def property_key(self, v): self.properties['property_key'] = v @property def counts(self): """Get or sets the list of LabelCounts containing counts for categorical data. Returns: List[LabelCount] """ counts = self.properties.get('counts') if counts is not None: counts = [LabelCount(c) for c in counts] return counts @counts.setter def counts(self, v): if v is None: self.properties.pop('counts', None) else: if not type(v) is list: raise STACError("counts must be a list! Invalid input: {}".format(v)) self.properties['counts'] = [c.to_dict() for c in v] @property def statistics(self): """Get or sets the list of Statistics containing statistics for regression/continuous numeric value data. Returns: List[Statistics] """ statistics = self.properties.get('statistics') if statistics is not None: statistics = [LabelStatistics(s) for s in statistics] return statistics @statistics.setter def statistics(self, v): if v is None: self.properties.pop('statistics', None) else: if not type(v) is list: raise STACError("statistics must be a list! Invalid input: {}".format(v)) self.properties['statistics'] = [s.to_dict() for s in v]
[docs] def merge_counts(self, other): """Merges the counts associated with this overview with another overview. Creates a new LabelOverview. Args: other (LabelOverview): The other LabelOverview to merge. Returns: LabelOverview: A new LabelOverview with the counts merged. This will drop any statistics associated with either of the LabelOverviews. """ assert (self.property_key == other.property_key) new_counts = None if self.counts is None: new_counts = other.counts else: if other.counts is None: new_counts = self.counts else: count_by_prop = {} def add_counts(counts): for c in counts: if c.name not in count_by_prop: count_by_prop[c.name] = c.count else: count_by_prop[c.name] += c.count add_counts(self.counts) add_counts(other.counts) new_counts = [LabelCount(k, v) for k, v in count_by_prop.items()] return LabelOverview.create(self.property_key, counts=new_counts)
[docs] def to_dict(self): """Returns the dictionary representing the JSON of this LabelOverview. Returns: dict: The wrapped dict of the LabelOverview that can be written out as JSON. """ return self.properties
[docs]class LabelCount: """Contains counts for categorical data. Use LabelCount.create to create a new LabelCount """ def __init__(self, properties): self.properties = properties
[docs] def apply(self, name, count): """Sets the properties for this LabelCount. Args: name (str): One of the different possible classes within the property. count (int): The number of occurrences of the class. """ self.name = name self.count = count
[docs] @classmethod def create(cls, name, count): """Creates a LabelCount. Args: name (str): One of the different possible classes within the property. count (int): The number of occurrences of the class. """ x = cls({}) x.apply(name, count) return x
@property def name(self): """Get or sets the class that this count represents. Returns: str """ return self.properties.get('name') @name.setter def name(self, v): self.properties['name'] = v @property def count(self): """Get or sets the number of occurrences of the class. Returns: int """ return self.properties.get('count') @count.setter def count(self, v): self.properties['count'] = v
[docs] def to_dict(self): """Returns the dictionary representing the JSON of this LabelCount. Returns: dict: The wrapped dict of the LabelCount that can be written out as JSON. """ return {'name': self.name, 'count': self.count}
[docs]class LabelStatistics: """Contains statistics for regression/continuous numeric value data. Use LabelStatistics.create to create a new instance. """ def __init__(self, properties): self.properties = properties
[docs] def apply(self, name, value): """Sets the property values for this instance. Args: name (str): The name of the statistic being reported. value (float): The value of the statistic """ self.name = name self.value = value
[docs] @classmethod def create(cls, name, value): """Sets the property values for this instance. Args: name (str): The name of the statistic being reported. value (float): The value of the statistic """ x = cls({}) x.apply(name, value) return x
@property def name(self): """Get or sets the name of the statistic being reported. Returns: str """ return self.properties.get('name') @name.setter def name(self, v): self.properties['name'] = v @property def value(self): """Get or sets the value of the statistic Returns: int or float """ return self.properties.get('value') @value.setter def value(self, v): self.properties['value'] = v
[docs] def to_dict(self): """Returns the dictionary representing the JSON of this LabelStatistics. Returns: dict: The wrapped dict of the LabelStatistics that can be written out as JSON. """ return {'name': self.name, 'value': self.value}
LABEL_EXTENSION_DEFINITION = ExtensionDefinition(Extensions.LABEL, [ExtendedObject(Item, LabelItemExt)])