import time
from typing import List
import logging
from .. import entities, repositories, exceptions, miscellaneous
from ..services.api_client import ApiClient
logger = logging.getLogger(name='dtlpy')
MIN_INTERVAL = 1
BACKOFF_FACTOR = 1.2
MAX_INTERVAL = 12
[docs]class Models:
"""
Models Repository
"""
def __init__(self,
client_api: ApiClient,
package: entities.Package = None,
project: entities.Project = None,
project_id: str = None):
self._client_api = client_api
self._project = project
self._package = package
self._project_id = project_id
if self._project is not None:
self._project_id = self._project.id
############
# entities #
############
@property
def project(self) -> entities.Project:
if self._project is None:
if self._project_id is not None:
projects = repositories.Projects(client_api=self._client_api)
self._project = projects.get(project_id=self._project_id)
if self._project is None:
if self._package is not None:
if self._package._project is not None:
self._project = self._package._project
if self._project is None:
raise exceptions.PlatformException(
error='2001',
message='Missing "project". need to set a Project entity or use project.models repository')
assert isinstance(self._project, entities.Project)
return self._project
@project.setter
def project(self, project: entities.Project):
if not isinstance(project, entities.Project):
raise ValueError('Must input a valid Project entity')
self._project = project
@property
def package(self) -> entities.Package:
if self._package is None:
raise exceptions.PlatformException(
error='2001',
message='Cannot perform action WITHOUT Package entity in {} repository.'.format(
self.__class__.__name__) +
' Please use package.models or set a model')
assert isinstance(self._package, entities.Package)
return self._package
###########
# methods #
###########
[docs] def get(self, model_name=None, model_id=None) -> entities.Model:
"""
Get model object
:param model_name:
:param model_id:
:return: dl.Model object
"""
if model_id is not None:
success, response = self._client_api.gen_request(req_type="get",
path="/ml/models/{}".format(model_id))
if not success:
raise exceptions.PlatformException(response)
model = entities.Model.from_json(client_api=self._client_api,
_json=response.json(),
project=self._project,
package=self._package)
# verify input model name is same as the given id
if model_name is not None and model.name != model_name:
logger.warning(
"Mismatch found in models.get: model_name is different then model.name:"
" {!r} != {!r}".format(
model_name,
model.name))
elif model_name is not None:
filters = entities.Filters(
resource=entities.FiltersResource.MODEL,
field='name',
values=model_name
)
project_id = None
if self._project is not None:
project_id = self._project.id
elif self._project_id is not None:
project_id = self._project_id
if project_id is not None:
filters.add(field='projectId', values=project_id)
if self._package is not None:
filters.add(field='packageId', values=self._package.id)
models = self.list(filters=filters)
if models.items_count == 0:
raise exceptions.PlatformException(
error='404',
message='Model not found. Name: {}'.format(model_name))
elif models.items_count > 1:
raise exceptions.PlatformException(
error='400',
message='More than one Model found by the name of: {}. Try "get" by id or "list()".'.format(
model_name))
model = models.items[0]
else:
raise exceptions.PlatformException(
error='400',
message='No checked-out Model was found, must checkout or provide an identifier in inputs')
return model
def _build_entities_from_response(self, response_items) -> miscellaneous.List[entities.Model]:
jobs = [None for _ in range(len(response_items))]
pool = self._client_api.thread_pools(pool_name='entity.create')
# return triggers list
for i_service, service in enumerate(response_items):
jobs[i_service] = pool.submit(entities.Model._protected_from_json,
**{'client_api': self._client_api,
'_json': service,
'package': self._package,
'project': self._project})
# get all results
results = [j.result() for j in jobs]
# log errors
_ = [logger.warning(r[1]) for r in results if r[0] is False]
# return good jobs
return miscellaneous.List([r[1] for r in results if r[0] is True])
def _list(self, filters: entities.Filters):
# request
success, response = self._client_api.gen_request(req_type='POST',
path='/ml/models/query',
json_req=filters.prepare())
if not success:
raise exceptions.PlatformException(response)
return response.json()
[docs] def list(self, filters: entities.Filters = None) -> entities.PagedEntities:
"""
List project model
:param dtlpy.entities.filters.Filters filters: Filters entity or a dictionary containing filters parameters
:return: Paged entity
:rtype: dtlpy.entities.paged_entities.PagedEntities
"""
# default filters
if filters is None:
filters = entities.Filters(resource=entities.FiltersResource.MODEL)
if self._project is not None:
filters.add(field='projectId', values=self._project.id)
if self._package is not None:
filters.add(field='packageId', values=self._package.id)
# assert type filters
if not isinstance(filters, entities.Filters):
raise exceptions.PlatformException(error='400',
message='Unknown filters type: {!r}'.format(type(filters)))
if filters.resource != entities.FiltersResource.MODEL:
raise exceptions.PlatformException(
error='400',
message='Filters resource must to be FiltersResource.MODEL. Got: {!r}'.format(filters.resource))
paged = entities.PagedEntities(items_repository=self,
filters=filters,
page_offset=filters.page,
page_size=filters.page_size,
client_api=self._client_api)
paged.get_page()
return paged
def _set_model_filter(self,
metadata: dict,
train_filter: entities.Filters = None,
validation_filter: entities.Filters = None):
if metadata is None:
metadata = {}
if 'system' not in metadata:
metadata['system'] = {}
if 'subsets' not in metadata['system']:
metadata['system']['subsets'] = {}
if train_filter is not None:
metadata['system']['subsets']['train'] = train_filter.prepare() if isinstance(train_filter,
entities.Filters) else train_filter
if validation_filter is not None:
metadata['system']['subsets']['validation'] = validation_filter.prepare() if isinstance(validation_filter,
entities.Filters) else validation_filter
return metadata
[docs] @staticmethod
def add_subset(model: entities.Model, subset_name: str, subset_filter: entities.Filters):
"""
Adds a subset for a model, specifying a subset of the model's dataset that could be used for training or
validation.
:param dtlpy.entities.Model model: the model to which the subset should be added
:param str subset_name: the name of the subset
:param dtlpy.entities.Filters subset_filter: the filtering operation that this subset performs in the dataset.
**Example**
.. code-block:: python
project.models.add_subset(model=model_entity, subset_name='train', subset_filter=dtlpy.Filters(field='dir', values='/train'))
model_entity.metadata['system']['subsets']
{'train': <dtlpy.entities.filters.Filters object at 0x1501dfe20>}
"""
if 'system' not in model.metadata:
model.metadata['system'] = dict()
if 'subsets' not in model.metadata['system']:
model.metadata['system']['subsets'] = dict()
model.metadata['system']['subsets'][subset_name] = subset_filter.prepare()
model.update(system_metadata=True)
[docs] @staticmethod
def delete_subset(model: entities.Model, subset_name: str):
"""
Removes a subset from a model's metadata.
:param dtlpy.entities.Model model: the model to which the subset should be added
:param str subset_name: the name of the subset
**Example**
.. code-block:: python
project.models.add_subset(model=model_entity, subset_name='train', subset_filter=dtlpy.Filters(field='dir', values='/train'))
model_entity.metadata['system']['subsets']
{'train': <dtlpy.entities.filters.Filters object at 0x1501dfe20>}
project.models.delete_subset(model=model_entity, subset_name='train')
model_entity.metadata['system']['subsets']
{}
"""
if model.metadata.get("system", dict()).get("subsets", dict()).get(subset_name) is None:
logger.error(f"Model system metadata incomplete, could not delete subset {subset_name}.")
else:
_ = model.metadata['system']['subsets'].pop(subset_name)
model.update(system_metadata=True)
[docs] def create(
self,
model_name: str,
dataset_id: str = None,
labels: list = None,
ontology_id: str = None,
description: str = None,
model_artifacts: List[entities.Artifact] = None,
project_id=None,
tags: List[str] = None,
package: entities.Package = None,
configuration: dict = None,
status: str = None,
scope: entities.EntityScopeLevel = entities.EntityScopeLevel.PROJECT,
version: str = '1.0.0',
input_type=None,
output_type=None,
train_filter: entities.Filters = None,
validation_filter: entities.Filters = None,
app: entities.App = None
) -> entities.Model:
"""
Create a Model entity
:param str model_name: name of the model
:param str dataset_id: dataset id
:param list labels: list of labels from ontology (must mach ontology id) can be a subset
:param str ontology_id: ontology to connect to the model
:param str description: description
:param model_artifacts: optional list of dl.Artifact. Can be ItemArtifact, LocaArtifact or LinkArtifact
:param str project_id: project that owns the model
:param list tags: list of string tags
:param package: optional - Package object
:param dict configuration: optional - model configuration - dict
:param str status: `str` of the optional values of
:param str scope: the scope level of the model dl.EntityScopeLevel
:param str version: version of the model
:param str input_type: the file type the model expect as input (image, video, txt, etc)
:param str output_type: dl.AnnotationType - the type of annotations the model produces (class, box segment, text, etc)
:param dtlpy.entities.filters.Filters train_filter: Filters entity or a dictionary to define the items' scope in the specified dataset_id for the model train
:param dtlpy.entities.filters.Filters validation_filter: Filters entity or a dictionary to define the items' scope in the specified dataset_id for the model validation
:param dtlpy.entities.App app: App entity to connect the model to
:return: Model Entity
**Example**:
.. code-block:: python
project.models.create(model_name='model_name', dataset_id='dataset_id', labels=['label1', 'label2'], train_filter={filter: {$and: [{dir: "/10K short videos"}]},page: 0,pageSize: 1000,resource: "items"}})
"""
if ontology_id is not None:
# take labels from ontology
ontologies = repositories.Ontologies(client_api=self._client_api)
labels = [label.tag for label in ontologies.get(ontology_id=ontology_id).labels]
if labels is None:
# dont have to have labels. can use an empty list
labels = list()
if input_type is None:
input_type = 'image'
if output_type is None:
output_type = entities.AnnotationType.CLASSIFICATION
if package is None and self._package is None:
raise exceptions.PlatformException('Must provide a Package or create from package.models')
elif package is None:
package = self._package
# TODO need to remove the entire project id user interface - need to take it from dataset id (in BE)
if project_id is None:
if self._project is None:
raise exceptions.PlatformException('Please provide project_id')
project_id = self._project.id
else:
if project_id != self._project_id:
if (isinstance(package, entities.Package) and not package.is_global) or \
(isinstance(package, entities.Dpk) and not package.scope != 'public'):
logger.warning(
"Note! you are specified project_id {!r} which is different from repository context: {!r}".format(
project_id, self._project_id))
if model_artifacts is None:
model_artifacts = []
if not isinstance(model_artifacts, list):
raise ValueError('`model_artifacts` must be a list of dl.Artifact entities')
# create payload for request
payload = {
'packageId': package.id,
'name': model_name,
'projectId': project_id,
'datasetId': dataset_id,
'labels': labels,
'artifacts': [artifact.to_json(as_artifact=True) for artifact in model_artifacts],
'scope': scope,
'version': version,
'inputType': input_type,
'outputType': output_type,
}
if app is not None:
if not isinstance(package, entities.Dpk):
raise ValueError('package must be a Dpk entity')
if app.dpk_name != package.name or app.dpk_version != package.version:
raise ValueError('App and package must be the same')
component_name = None
compute_config = None
for model in package.components.models:
if model['name'] == model_name:
component_name = model['name']
compute_config = model.get('computeConfigs', None)
break
if component_name is None:
raise ValueError('Model name not found in package')
payload['app'] = {
"id": app.id,
"componentName": component_name,
"dpkName": package.name,
"dpkVersion": package.version
}
if compute_config is not None:
payload['app']['computeConfig'] = compute_config
if configuration is not None:
payload['configuration'] = configuration
if tags is not None:
payload['tags'] = tags
if description is not None:
payload['description'] = description
if status is not None:
payload['status'] = status
if train_filter or validation_filter:
metadata = self._set_model_filter(metadata={},
train_filter=train_filter,
validation_filter=validation_filter)
payload['metadata'] = metadata
# request
success, response = self._client_api.gen_request(req_type='post',
path='/ml/models',
json_req=payload)
# exception handling
if not success:
raise exceptions.PlatformException(response)
model = entities.Model.from_json(_json=response.json(),
client_api=self._client_api,
project=self._project,
package=package)
return model
[docs] def clone(self,
from_model: entities.Model,
model_name: str,
dataset: entities.Dataset = None,
configuration: dict = None,
status=None,
scope=None,
project_id: str = None,
labels: list = None,
description: str = None,
tags: list = None,
train_filter: entities.Filters = None,
validation_filter: entities.Filters = None,
wait=True,
) -> entities.Model:
"""
Clones and creates a new model out of existing one
:param from_model: existing model to clone from
:param str model_name: `str` new model name
:param str dataset: dataset object for the cloned model
:param dict configuration: `dict` (optional) if passed replaces the current configuration
:param str status: `str` (optional) set the new status
:param str scope: `str` (optional) set the new scope. default is "project"
:param str project_id: `str` specify the project id to create the new model on (if other than the source model)
:param list labels: `list` of `str` - label of the model
:param str description: `str` description of the new model
:param list tags: `list` of `str` - label of the model
:param dtlpy.entities.filters.Filters train_filter: Filters entity or a dictionary to define the items' scope in the specified dataset_id for the model train
:param dtlpy.entities.filters.Filters validation_filter: Filters entity or a dictionary to define the items' scope in the specified dataset_id for the model validation
:param bool wait: `bool` wait for model to be ready
:return: dl.Model which is a clone version of the existing model
"""
from_json = {"name": model_name,
"packageId": from_model.package_id,
"configuration": from_model.configuration,
"outputType": from_model.output_type,
"inputType": from_model.input_type}
if project_id is None:
if dataset is not None:
# take dataset project
project_id = dataset.project.id
else:
# take model's project
project_id = self.project.id
from_json['projectId'] = project_id
if dataset is not None:
if labels is None:
labels = list(dataset.labels_flat_dict.keys())
from_json['datasetId'] = dataset.id
if labels is not None:
from_json['labels'] = labels
# if there are new labels - pop the mapping from the original
_ = from_json['configuration'].pop('id_to_label_map', None)
_ = from_json['configuration'].pop('label_to_id_map', None)
if configuration is not None:
from_json['configuration'].update(configuration)
if description is not None:
from_json['description'] = description
if tags is not None:
from_json['tags'] = tags
if scope is not None:
from_json['scope'] = scope
if status is not None:
from_json['status'] = status
metadata = self._set_model_filter(metadata={},
train_filter=train_filter if train_filter is not None else from_model.metadata.get(
'system', {}).get('subsets', {}).get('train', None),
validation_filter=validation_filter if validation_filter is not None else from_model.metadata.get(
'system', {}).get('subsets', {}).get('validation', None))
if metadata:
from_json['metadata'] = metadata
success, response = self._client_api.gen_request(req_type='post',
path='/ml/models/{}/clone'.format(from_model.id),
json_req=from_json)
if not success:
raise exceptions.PlatformException(response)
new_model = entities.Model.from_json(_json=response.json(),
client_api=self._client_api,
project=self._project,
package=from_model._package)
if wait:
new_model = self.wait_for_model_ready(model=new_model)
return new_model
[docs] def wait_for_model_ready(self, model: entities.Model):
"""
Wait for model to be ready
:param model: Model entity
"""
sleep_time = MIN_INTERVAL
while model.status == entities.ModelStatus.CLONING:
model = self.get(model_id=model.id)
time.sleep(sleep_time)
sleep_time = min(sleep_time * BACKOFF_FACTOR, MAX_INTERVAL)
time.sleep(sleep_time)
return model
@property
def platform_url(self):
return self._client_api._get_resource_url("projects/{}/models".format(self.project.id))
[docs] def open_in_web(self, model=None, model_id=None):
"""
Open the model in web platform
:param model: model entity
:param str model_id: model id
"""
if model is not None:
model.open_in_web()
elif model_id is not None:
self._client_api._open_in_web(url=self.platform_url + '/' + str(model_id) + '/main')
else:
self._client_api._open_in_web(url=self.platform_url)
[docs] def delete(self, model: entities.Model = None, model_name=None, model_id=None):
"""
Delete Model object
:param model: Model entity to delete
:param str model_name: delete by model name
:param str model_id: delete by model id
:return: True
:rtype: bool
"""
# get id and name
if model_id is None:
if model is not None:
model_id = model.id
elif model_name is not None:
model = self.get(model_name=model_name)
model_id = model.id
else:
raise exceptions.PlatformException(error='400',
message='Must input at least one parameter to models.delete')
# request
success, response = self._client_api.gen_request(
req_type="delete",
path="/ml/models/{}".format(model_id)
)
# exception handling
if not success:
raise exceptions.PlatformException(response)
# return results
return True
[docs] def update(self,
model: entities.Model,
system_metadata: bool = False) -> entities.Model:
"""
Update Model changes to platform
:param model: Model entity
:param bool system_metadata: True, if you want to change metadata system
:return: Model entity
"""
# payload
payload = model.to_json()
# url
url_path = '/ml/models/{}'.format(model.id)
if system_metadata:
url_path += '?system=true'
# request
success, response = self._client_api.gen_request(req_type='patch',
path=url_path,
json_req=payload)
# exception handling
if not success:
raise exceptions.PlatformException(response)
# return entity
return entities.Model.from_json(_json=response.json(),
client_api=self._client_api,
project=self._project,
package=model._package)
[docs] def train(self, model_id: str, service_config=None):
"""
Train the model in the cloud. This will create a service and will run the adapter's train function as an execution
:param model_id: id of the model to train
:param dict service_config : Service object as dict. Contains the spec of the default service to create.
:return:
"""
payload = dict()
if service_config is not None:
payload['serviceConfig'] = service_config
success, response = self._client_api.gen_request(req_type="post",
path=f"/ml/models/{model_id}/train",
json_req=payload)
if not success:
raise exceptions.PlatformException(response)
return entities.Execution.from_json(_json=response.json(),
client_api=self._client_api,
project=self._project)
[docs] def evaluate(self, model_id: str, dataset_id: str, filters: entities.Filters = None, service_config=None):
"""
Evaluate Model, provide data to evaluate the model on You can also provide specific config for the deployed service
:param str model_id: Model id to predict
:param dict service_config : Service object as dict. Contains the spec of the default service to create.
:param str dataset_id: ID of the dataset to evaluate
:param entities.Filters filters: dl.Filter entity to run the predictions on
:return:
"""
payload = {'input': {'datasetId': dataset_id}}
if service_config is not None:
payload['config'] = {'serviceConfig': service_config}
if filters is None:
filters = entities.Filters()
if filters is not None:
payload['input']['datasetQuery'] = filters.prepare()
success, response = self._client_api.gen_request(req_type="post",
path=f"/ml/models/{model_id}/evaluate",
json_req=payload)
if not success:
raise exceptions.PlatformException(response)
return entities.Execution.from_json(_json=response.json(),
client_api=self._client_api,
project=self._project)
[docs] def predict(self, model, item_ids):
"""
Run model prediction with items
:param model: dl.Model entity to run the prediction.
:param item_ids: a list of item id to run the prediction.
:return:
"""
if len(model.metadata['system'].get('deploy', {}).get('services', [])) == 0:
# no services for model
raise ValueError("Model doesnt have any associated services. Need to deploy before predicting")
payload = {'input': {'itemIds': item_ids},
'config': {'serviceId': model.metadata['system']['deploy']['services'][0]}}
success, response = self._client_api.gen_request(req_type="post",
path=f"/ml/models/{model.id}/predict",
json_req=payload)
if not success:
raise exceptions.PlatformException(response)
return entities.Execution.from_json(_json=response.json(),
client_api=self._client_api,
project=self._project)
[docs] def deploy(self, model_id: str, service_config=None) -> entities.Service:
"""
Deploy a trained model. This will create a service that will execute predictions
:param model_id: id of the model to deploy
:param dict service_config : Service object as dict. Contains the spec of the default service to create.
:return: dl.Service: the deployed service
"""
payload = dict()
if service_config is not None:
payload['serviceConfig'] = service_config
success, response = self._client_api.gen_request(req_type="post",
path=f"/ml/models/{model_id}/deploy",
json_req=payload)
if not success:
raise exceptions.PlatformException(response)
return entities.Service.from_json(_json=response.json(),
client_api=self._client_api,
project=self._project,
package=self._package)
class Metrics:
def __init__(self, client_api, model=None, model_id=None):
self._client_api = client_api
self._model_id = model_id
self._model = model
@property
def model(self):
return self._model
def create(self, samples, dataset_id) -> bool:
"""
Add Samples for model analytics and metrics
:param samples: list of dl.PlotSample - must contain: model_id, figure, legend, x, y
:param model_id: model id to save samples on
:param dataset_id:
:return: bool: True if success
"""
if not isinstance(samples, list):
samples = [samples]
payload = list()
for sample in samples:
_json = sample.to_json()
_json['modelId'] = self.model.id
_json['datasetId'] = dataset_id
payload.append(_json)
# request
success, response = self._client_api.gen_request(req_type='post',
path='/ml/metrics/publish',
json_req=payload)
# exception handling
if not success:
raise exceptions.PlatformException(response)
# return entity
return True
def _list(self, filters: entities.Filters):
# request
success, response = self._client_api.gen_request(req_type='POST',
path='/ml/metrics/query',
json_req=filters.prepare())
if not success:
raise exceptions.PlatformException(response)
return response.json()
def _build_entities_from_response(self, response_items) -> miscellaneous.List[entities.Model]:
jobs = [None for _ in range(len(response_items))]
pool = self._client_api.thread_pools(pool_name='entity.create')
# return triggers list
for i_service, sample in enumerate(response_items):
jobs[i_service] = pool.submit(entities.PlotSample,
**{'x': sample.get('data', dict()).get('x', None),
'y': sample.get('data', dict()).get('y', None),
'legend': sample.get('legend', ''),
'figure': sample.get('figure', '')})
# get all results
results = [j.result() for j in jobs]
# return good jobs
return miscellaneous.List(results)
def list(self, filters=None) -> entities.PagedEntities:
"""
List Samples for model analytics and metrics
:param filters: dl.Filter query entity
"""
if filters is None:
filters = entities.Filters(resource=entities.FiltersResource.METRICS)
if not isinstance(filters, entities.Filters):
raise exceptions.PlatformException(error='400',
message='Unknown filters type: {!r}'.format(type(filters)))
if filters.resource != entities.FiltersResource.METRICS:
raise exceptions.PlatformException(
error='400',
message='Filters resource must to be FiltersResource.METRICS. Got: {!r}'.format(filters.resource))
if self._model is not None:
filters.add(field='modelId', values=self._model.id)
paged = entities.PagedEntities(items_repository=self,
filters=filters,
page_offset=filters.page,
page_size=filters.page_size,
client_api=self._client_api)
paged.get_page()
return paged