# -*- coding: utf-8 -*-
#
# Copyright 2018-2020- Swiss Data Science Center (SDSC)
# A partnership between École Polytechnique Fédérale de Lausanne (EPFL) and
# Eidgenössische Technische Hochschule Zürich (ETHZ).
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Represent elaborated information about relations."""
import weakref
from urllib.parse import quote
import attr
from marshmallow import EXCLUDE
from renku.core.models.calamus import JsonLDSchema, Nested, fields, prov
from renku.core.models.datasets import DatasetFileSchema, DatasetSchema
from renku.core.models.entities import CollectionSchema, EntitySchema
from renku.core.models.provenance.agents import PersonSchema, SoftwareAgentSchema
from renku.core.models.workflow.run import RunSchema
@attr.s
class Association:
"""Assign responsibility to an agent for an activity."""
plan = attr.ib()
agent = attr.ib(default=None,)
_id = attr.ib(kw_only=True)
@classmethod
def from_activity(cls, activity, commit=None):
"""Create an instance from the activity."""
from .agents import SoftwareAgent
agent = SoftwareAgent.from_commit(activity.commit)
return cls(
plan=activity.__association_cls__(
commit=commit or activity.commit, client=activity.client, path=activity.path, activity=activity,
),
agent=agent,
id=activity._id + "/association", # add plan and agent
)
@classmethod
def from_jsonld(cls, data):
"""Create an instance from JSON-LD data."""
if isinstance(data, cls):
return data
if not isinstance(data, dict):
raise ValueError(data)
return AssociationSchema().load(data)
def as_jsonld(self):
"""Create JSON-LD."""
return AssociationSchema().dump(self)
class EntityProxyMixin:
"""Implement proxy to entity attribute."""
def __getattribute__(self, name):
"""Proxy entity attributes."""
cls = object.__getattribute__(self, "__class__")
names = {field.name for field in attr.fields(cls)}
names |= set(dir(cls))
if name in names:
return object.__getattribute__(self, name)
entity = object.__getattribute__(self, "entity")
return getattr(entity, name)
[docs]@attr.s(cmp=False,)
class Usage(EntityProxyMixin):
"""Represent a dependent path."""
entity = attr.ib(kw_only=True)
role = attr.ib(default=None, kw_only=True)
_id = attr.ib(default=None, kw_only=True)
[docs] @classmethod
def from_revision(cls, client, path, revision="HEAD", **kwargs):
"""Return dependency from given path and revision."""
from renku.core.models.entities import Entity
return cls(entity=Entity.from_revision(client, path, revision), **kwargs)
[docs] @classmethod
def from_jsonld(cls, data):
"""Create an instance from JSON-LD data."""
if isinstance(data, cls):
return data
if not isinstance(data, dict):
raise ValueError(data)
return UsageSchema().load(data)
[docs] def as_jsonld(self):
"""Create JSON-LD."""
return UsageSchema().dump(self)
[docs]@attr.s(cmp=False,)
class Generation(EntityProxyMixin):
"""Represent an act of generating a file."""
entity = attr.ib()
role = attr.ib(default=None)
_activity = attr.ib(
default=None, kw_only=True, converter=lambda value: weakref.ref(value) if value is not None else None,
)
_id = attr.ib(kw_only=True)
@property
def activity(self):
"""Return the activity object."""
return self._activity() if self._activity is not None else None
[docs] @_id.default
def default_id(self):
"""Configure calculated ID."""
if self.role:
return f"{self.activity._id}/{self.role}"
return f"{self.activity._id}/tree/{quote(str(self.entity.path))}"
[docs] @classmethod
def from_jsonld(cls, data):
"""Create an instance from JSON-LD data."""
if isinstance(data, cls):
return data
if not isinstance(data, dict):
raise ValueError(data)
return GenerationSchema().load(data)
[docs] def as_jsonld(self):
"""Create JSON-LD."""
return GenerationSchema().dump(self)
class AssociationSchema(JsonLDSchema):
"""Association schema."""
class Meta:
"""Meta class."""
rdf_type = prov.Association
model = Association
unknown = EXCLUDE
_id = fields.Id(init_name="id")
plan = Nested(prov.hadPlan, RunSchema)
agent = Nested(prov.agent, [SoftwareAgentSchema, PersonSchema])
class UsageSchema(JsonLDSchema):
"""Usage schema."""
class Meta:
"""Meta class."""
rdf_type = prov.Usage
model = Usage
unknown = EXCLUDE
_id = fields.Id(init_name="id")
entity = Nested(prov.entity, [EntitySchema, CollectionSchema, DatasetSchema, DatasetFileSchema])
role = fields.String(prov.hadRole, missing=None)
class GenerationSchema(JsonLDSchema):
"""Generation schema."""
class Meta:
"""Meta class."""
rdf_type = prov.Generation
model = Generation
unknown = EXCLUDE
_id = fields.Id(init_name="id")
entity = Nested(
prov.qualifiedGeneration, [EntitySchema, CollectionSchema, DatasetSchema, DatasetFileSchema], reverse=True
)
role = fields.String(prov.hadRole, missing=None)