compose/fig/project.py

167 lines
5.6 KiB
Python
Raw Normal View History

from __future__ import unicode_literals
from __future__ import absolute_import
2014-01-03 00:27:47 +01:00
import logging
2013-12-09 15:10:23 +01:00
from .service import Service
2014-01-03 00:27:47 +01:00
log = logging.getLogger(__name__)
2013-12-09 15:10:23 +01:00
def sort_service_dicts(services):
# Topological sort (Cormen/Tarjan algorithm).
unmarked = services[:]
temporary_marked = set()
sorted_services = []
2014-03-01 17:17:19 +01:00
get_service_names = lambda links: [link.split(':')[0] for link in links]
def visit(n):
if n['name'] in temporary_marked:
2014-03-01 17:17:19 +01:00
if n['name'] in get_service_names(n.get('links', [])):
raise DependencyError('A service can not link to itself: %s' % n['name'])
else:
raise DependencyError('Circular import between %s' % ' and '.join(temporary_marked))
if n in unmarked:
temporary_marked.add(n['name'])
2014-03-01 17:17:19 +01:00
dependents = [m for m in services if n['name'] in get_service_names(m.get('links', []))]
for m in dependents:
visit(m)
temporary_marked.remove(n['name'])
unmarked.remove(n)
sorted_services.insert(0, n)
while unmarked:
visit(unmarked[-1])
return sorted_services
2013-12-09 15:10:23 +01:00
class Project(object):
"""
A collection of services.
"""
def __init__(self, name, services, client):
self.name = name
self.services = services
self.client = client
2013-12-09 15:10:23 +01:00
@classmethod
def from_dicts(cls, name, service_dicts, client):
2013-12-09 15:10:23 +01:00
"""
Construct a ServiceCollection from a list of dicts representing services.
"""
project = cls(name, [], client)
2013-12-09 15:10:23 +01:00
for service_dict in sort_service_dicts(service_dicts):
# Reference links by object
links = []
if 'links' in service_dict:
2014-03-01 17:17:19 +01:00
for link in service_dict.get('links', []):
if ':' in link:
service_name, link_name = link.split(':', 1)
else:
service_name, link_name = link, None
try:
links.append((project.get_service(service_name), link_name))
except NoSuchService:
raise ConfigurationError('Service "%s" has a link to service "%s" which does not exist.' % (service_dict['name'], service_name))
2013-12-09 15:10:23 +01:00
del service_dict['links']
project.services.append(Service(client=client, project=name, links=links, **service_dict))
return project
2013-12-09 15:10:23 +01:00
2013-12-11 15:25:32 +01:00
@classmethod
def from_config(cls, name, config, client):
2013-12-11 15:25:32 +01:00
dicts = []
for service_name, service in list(config.items()):
if not isinstance(service, dict):
raise ConfigurationError('Service "%s" doesn\'t have any configuration options. All top level keys in your fig.yml must map to a dictionary of configuration options.')
service['name'] = service_name
2013-12-11 15:25:32 +01:00
dicts.append(service)
return cls.from_dicts(name, dicts, client)
2013-12-11 15:25:32 +01:00
def get_service(self, name):
"""
Retrieve a service by name. Raises NoSuchService
if the named service does not exist.
"""
for service in self.services:
2013-12-09 15:10:23 +01:00
if service.name == name:
return service
raise NoSuchService(name)
def get_services(self, service_names=None):
"""
Returns a list of this project's services filtered
by the provided list of names, or all services if
service_names is None or [].
Preserves the original order of self.services.
Raises NoSuchService if any of the named services
do not exist.
"""
if service_names is None or len(service_names) == 0:
return self.services
else:
unsorted = [self.get_service(name) for name in service_names]
return [s for s in self.services if s in unsorted]
def start(self, service_names=None, **options):
for service in self.get_services(service_names):
service.start(**options)
def stop(self, service_names=None, **options):
for service in reversed(self.get_services(service_names)):
service.stop(**options)
def kill(self, service_names=None, **options):
for service in reversed(self.get_services(service_names)):
2013-12-20 17:53:07 +01:00
service.kill(**options)
2014-01-02 16:28:33 +01:00
def build(self, service_names=None, **options):
for service in self.get_services(service_names):
if service.can_be_built():
service.build(**options)
else:
2014-01-03 00:28:18 +01:00
log.info('%s uses an image, skipping' % service.name)
2014-01-02 16:28:33 +01:00
def up(self, service_names=None):
new_containers = []
for service in self.get_services(service_names):
for (_, new) in service.recreate_containers():
new_containers.append(new)
return new_containers
def remove_stopped(self, service_names=None, **options):
for service in self.get_services(service_names):
2013-12-20 17:53:07 +01:00
service.remove_stopped(**options)
def containers(self, service_names=None, *args, **kwargs):
2013-12-19 14:02:04 +01:00
l = []
for service in self.get_services(service_names):
2013-12-19 14:02:04 +01:00
for container in service.containers(*args, **kwargs):
l.append(container)
return l
2013-12-09 15:10:23 +01:00
class NoSuchService(Exception):
def __init__(self, name):
self.name = name
self.msg = "No such service: %s" % self.name
def __str__(self):
return self.msg
class ConfigurationError(Exception):
def __init__(self, msg):
self.msg = msg
def __str__(self):
return self.msg
class DependencyError(ConfigurationError):
pass