mirror of https://github.com/docker/docker-py.git
352 lines
13 KiB
Python
352 lines
13 KiB
Python
import warnings
|
|
from .. import auth, errors, utils
|
|
from ..types import ServiceMode
|
|
|
|
|
|
def _check_api_features(version, task_template, update_config):
|
|
if update_config is not None:
|
|
if utils.version_lt(version, '1.25'):
|
|
if 'MaxFailureRatio' in update_config:
|
|
raise errors.InvalidVersion(
|
|
'UpdateConfig.max_failure_ratio is not supported in'
|
|
' API version < 1.25'
|
|
)
|
|
if 'Monitor' in update_config:
|
|
raise errors.InvalidVersion(
|
|
'UpdateConfig.monitor is not supported in'
|
|
' API version < 1.25'
|
|
)
|
|
|
|
if task_template is not None:
|
|
if 'ForceUpdate' in task_template and utils.version_lt(
|
|
version, '1.25'):
|
|
raise errors.InvalidVersion(
|
|
'force_update is not supported in API version < 1.25'
|
|
)
|
|
|
|
if task_template.get('Placement'):
|
|
if utils.version_lt(version, '1.30'):
|
|
if task_template['Placement'].get('Platforms'):
|
|
raise errors.InvalidVersion(
|
|
'Placement.platforms is not supported in'
|
|
' API version < 1.30'
|
|
)
|
|
|
|
if utils.version_lt(version, '1.27'):
|
|
if task_template['Placement'].get('Preferences'):
|
|
raise errors.InvalidVersion(
|
|
'Placement.preferences is not supported in'
|
|
' API version < 1.27'
|
|
)
|
|
if task_template.container_spec.get('TTY'):
|
|
if utils.version_lt(version, '1.25'):
|
|
raise errors.InvalidVersion(
|
|
'ContainerSpec.TTY is not supported in API version < 1.25'
|
|
)
|
|
|
|
|
|
class ServiceApiMixin(object):
|
|
@utils.minimum_version('1.24')
|
|
def create_service(
|
|
self, task_template, name=None, labels=None, mode=None,
|
|
update_config=None, networks=None, endpoint_config=None,
|
|
endpoint_spec=None
|
|
):
|
|
"""
|
|
Create a service.
|
|
|
|
Args:
|
|
task_template (TaskTemplate): Specification of the task to start as
|
|
part of the new service.
|
|
name (string): User-defined name for the service. Optional.
|
|
labels (dict): A map of labels to associate with the service.
|
|
Optional.
|
|
mode (ServiceMode): Scheduling mode for the service (replicated
|
|
or global). Defaults to replicated.
|
|
update_config (UpdateConfig): Specification for the update strategy
|
|
of the service. Default: ``None``
|
|
networks (:py:class:`list`): List of network names or IDs to attach
|
|
the service to. Default: ``None``.
|
|
endpoint_spec (EndpointSpec): Properties that can be configured to
|
|
access and load balance a service. Default: ``None``.
|
|
|
|
Returns:
|
|
A dictionary containing an ``ID`` key for the newly created
|
|
service.
|
|
|
|
Raises:
|
|
:py:class:`docker.errors.APIError`
|
|
If the server returns an error.
|
|
"""
|
|
if endpoint_config is not None:
|
|
warnings.warn(
|
|
'endpoint_config has been renamed to endpoint_spec.',
|
|
DeprecationWarning
|
|
)
|
|
endpoint_spec = endpoint_config
|
|
|
|
_check_api_features(self._version, task_template, update_config)
|
|
|
|
url = self._url('/services/create')
|
|
headers = {}
|
|
image = task_template.get('ContainerSpec', {}).get('Image', None)
|
|
if image is None:
|
|
raise errors.DockerException(
|
|
'Missing mandatory Image key in ContainerSpec'
|
|
)
|
|
if mode and not isinstance(mode, dict):
|
|
mode = ServiceMode(mode)
|
|
|
|
registry, repo_name = auth.resolve_repository_name(image)
|
|
auth_header = auth.get_config_header(self, registry)
|
|
if auth_header:
|
|
headers['X-Registry-Auth'] = auth_header
|
|
data = {
|
|
'Name': name,
|
|
'Labels': labels,
|
|
'TaskTemplate': task_template,
|
|
'Mode': mode,
|
|
'Networks': utils.convert_service_networks(networks),
|
|
'EndpointSpec': endpoint_spec
|
|
}
|
|
|
|
if update_config is not None:
|
|
data['UpdateConfig'] = update_config
|
|
|
|
return self._result(
|
|
self._post_json(url, data=data, headers=headers), True
|
|
)
|
|
|
|
@utils.minimum_version('1.24')
|
|
@utils.check_resource('service')
|
|
def inspect_service(self, service):
|
|
"""
|
|
Return information about a service.
|
|
|
|
Args:
|
|
service (str): Service name or ID
|
|
|
|
Returns:
|
|
``True`` if successful.
|
|
|
|
Raises:
|
|
:py:class:`docker.errors.APIError`
|
|
If the server returns an error.
|
|
"""
|
|
url = self._url('/services/{0}', service)
|
|
return self._result(self._get(url), True)
|
|
|
|
@utils.minimum_version('1.24')
|
|
@utils.check_resource('task')
|
|
def inspect_task(self, task):
|
|
"""
|
|
Retrieve information about a task.
|
|
|
|
Args:
|
|
task (str): Task ID
|
|
|
|
Returns:
|
|
(dict): Information about the task.
|
|
|
|
Raises:
|
|
:py:class:`docker.errors.APIError`
|
|
If the server returns an error.
|
|
"""
|
|
url = self._url('/tasks/{0}', task)
|
|
return self._result(self._get(url), True)
|
|
|
|
@utils.minimum_version('1.24')
|
|
@utils.check_resource('service')
|
|
def remove_service(self, service):
|
|
"""
|
|
Stop and remove a service.
|
|
|
|
Args:
|
|
service (str): Service name or ID
|
|
|
|
Returns:
|
|
``True`` if successful.
|
|
|
|
Raises:
|
|
:py:class:`docker.errors.APIError`
|
|
If the server returns an error.
|
|
"""
|
|
|
|
url = self._url('/services/{0}', service)
|
|
resp = self._delete(url)
|
|
self._raise_for_status(resp)
|
|
return True
|
|
|
|
@utils.minimum_version('1.24')
|
|
def services(self, filters=None):
|
|
"""
|
|
List services.
|
|
|
|
Args:
|
|
filters (dict): Filters to process on the nodes list. Valid
|
|
filters: ``id`` and ``name``. Default: ``None``.
|
|
|
|
Returns:
|
|
A list of dictionaries containing data about each service.
|
|
|
|
Raises:
|
|
:py:class:`docker.errors.APIError`
|
|
If the server returns an error.
|
|
"""
|
|
params = {
|
|
'filters': utils.convert_filters(filters) if filters else None
|
|
}
|
|
url = self._url('/services')
|
|
return self._result(self._get(url, params=params), True)
|
|
|
|
@utils.minimum_version('1.25')
|
|
@utils.check_resource('service')
|
|
def service_logs(self, service, details=False, follow=False, stdout=False,
|
|
stderr=False, since=0, timestamps=False, tail='all',
|
|
is_tty=None):
|
|
"""
|
|
Get log stream for a service.
|
|
Note: This endpoint works only for services with the ``json-file``
|
|
or ``journald`` logging drivers.
|
|
|
|
Args:
|
|
service (str): ID or name of the service
|
|
details (bool): Show extra details provided to logs.
|
|
Default: ``False``
|
|
follow (bool): Keep connection open to read logs as they are
|
|
sent by the Engine. Default: ``False``
|
|
stdout (bool): Return logs from ``stdout``. Default: ``False``
|
|
stderr (bool): Return logs from ``stderr``. Default: ``False``
|
|
since (int): UNIX timestamp for the logs staring point.
|
|
Default: 0
|
|
timestamps (bool): Add timestamps to every log line.
|
|
tail (string or int): Number of log lines to be returned,
|
|
counting from the current end of the logs. Specify an
|
|
integer or ``'all'`` to output all log lines.
|
|
Default: ``all``
|
|
is_tty (bool): Whether the service's :py:class:`ContainerSpec`
|
|
enables the TTY option. If omitted, the method will query
|
|
the Engine for the information, causing an additional
|
|
roundtrip.
|
|
|
|
Returns (generator): Logs for the service.
|
|
"""
|
|
params = {
|
|
'details': details,
|
|
'follow': follow,
|
|
'stdout': stdout,
|
|
'stderr': stderr,
|
|
'since': since,
|
|
'timestamps': timestamps,
|
|
'tail': tail
|
|
}
|
|
|
|
url = self._url('/services/{0}/logs', service)
|
|
res = self._get(url, params=params, stream=True)
|
|
if is_tty is None:
|
|
is_tty = self.inspect_service(
|
|
service
|
|
)['Spec']['TaskTemplate']['ContainerSpec'].get('TTY', False)
|
|
return self._get_result_tty(True, res, is_tty)
|
|
|
|
@utils.minimum_version('1.24')
|
|
def tasks(self, filters=None):
|
|
"""
|
|
Retrieve a list of tasks.
|
|
|
|
Args:
|
|
filters (dict): A map of filters to process on the tasks list.
|
|
Valid filters: ``id``, ``name``, ``service``, ``node``,
|
|
``label`` and ``desired-state``.
|
|
|
|
Returns:
|
|
(:py:class:`list`): List of task dictionaries.
|
|
|
|
Raises:
|
|
:py:class:`docker.errors.APIError`
|
|
If the server returns an error.
|
|
"""
|
|
|
|
params = {
|
|
'filters': utils.convert_filters(filters) if filters else None
|
|
}
|
|
url = self._url('/tasks')
|
|
return self._result(self._get(url, params=params), True)
|
|
|
|
@utils.minimum_version('1.24')
|
|
@utils.check_resource('service')
|
|
def update_service(self, service, version, task_template=None, name=None,
|
|
labels=None, mode=None, update_config=None,
|
|
networks=None, endpoint_config=None,
|
|
endpoint_spec=None):
|
|
"""
|
|
Update a service.
|
|
|
|
Args:
|
|
service (string): A service identifier (either its name or service
|
|
ID).
|
|
version (int): The version number of the service object being
|
|
updated. This is required to avoid conflicting writes.
|
|
task_template (TaskTemplate): Specification of the updated task to
|
|
start as part of the service.
|
|
name (string): New name for the service. Optional.
|
|
labels (dict): A map of labels to associate with the service.
|
|
Optional.
|
|
mode (ServiceMode): Scheduling mode for the service (replicated
|
|
or global). Defaults to replicated.
|
|
update_config (UpdateConfig): Specification for the update strategy
|
|
of the service. Default: ``None``.
|
|
networks (:py:class:`list`): List of network names or IDs to attach
|
|
the service to. Default: ``None``.
|
|
endpoint_spec (EndpointSpec): Properties that can be configured to
|
|
access and load balance a service. Default: ``None``.
|
|
|
|
Returns:
|
|
``True`` if successful.
|
|
|
|
Raises:
|
|
:py:class:`docker.errors.APIError`
|
|
If the server returns an error.
|
|
"""
|
|
if endpoint_config is not None:
|
|
warnings.warn(
|
|
'endpoint_config has been renamed to endpoint_spec.',
|
|
DeprecationWarning
|
|
)
|
|
endpoint_spec = endpoint_config
|
|
|
|
_check_api_features(self._version, task_template, update_config)
|
|
|
|
url = self._url('/services/{0}/update', service)
|
|
data = {}
|
|
headers = {}
|
|
if name is not None:
|
|
data['Name'] = name
|
|
if labels is not None:
|
|
data['Labels'] = labels
|
|
if mode is not None:
|
|
if not isinstance(mode, dict):
|
|
mode = ServiceMode(mode)
|
|
data['Mode'] = mode
|
|
if task_template is not None:
|
|
image = task_template.get('ContainerSpec', {}).get('Image', None)
|
|
if image is not None:
|
|
registry, repo_name = auth.resolve_repository_name(image)
|
|
auth_header = auth.get_config_header(self, registry)
|
|
if auth_header:
|
|
headers['X-Registry-Auth'] = auth_header
|
|
data['TaskTemplate'] = task_template
|
|
if update_config is not None:
|
|
data['UpdateConfig'] = update_config
|
|
|
|
if networks is not None:
|
|
data['Networks'] = utils.convert_service_networks(networks)
|
|
if endpoint_spec is not None:
|
|
data['EndpointSpec'] = endpoint_spec
|
|
|
|
resp = self._post_json(
|
|
url, data=data, params={'version': version}, headers=headers
|
|
)
|
|
self._raise_for_status(resp)
|
|
return True
|