project.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. from __future__ import unicode_literals
  2. from __future__ import absolute_import
  3. import logging
  4. from .service import Service
  5. log = logging.getLogger(__name__)
  6. def sort_service_dicts(services):
  7. # Topological sort (Cormen/Tarjan algorithm).
  8. unmarked = services[:]
  9. temporary_marked = set()
  10. sorted_services = []
  11. def visit(n):
  12. if n['name'] in temporary_marked:
  13. if n['name'] in n.get('links', []):
  14. raise DependencyError('A service can not link to itself: %s' % n['name'])
  15. else:
  16. raise DependencyError('Circular import between %s' % ' and '.join(temporary_marked))
  17. if n in unmarked:
  18. temporary_marked.add(n['name'])
  19. dependents = [m for m in services if n['name'] in m.get('links', [])]
  20. for m in dependents:
  21. visit(m)
  22. temporary_marked.remove(n['name'])
  23. unmarked.remove(n)
  24. sorted_services.insert(0, n)
  25. while unmarked:
  26. visit(unmarked[-1])
  27. return sorted_services
  28. class Project(object):
  29. """
  30. A collection of services.
  31. """
  32. def __init__(self, name, services, client):
  33. self.name = name
  34. self.services = services
  35. self.client = client
  36. @classmethod
  37. def from_dicts(cls, name, service_dicts, client):
  38. """
  39. Construct a ServiceCollection from a list of dicts representing services.
  40. """
  41. project = cls(name, [], client)
  42. for service_dict in sort_service_dicts(service_dicts):
  43. # Reference links by object
  44. links = []
  45. if 'links' in service_dict:
  46. for service_name in service_dict.get('links', []):
  47. links.append(project.get_service(service_name))
  48. del service_dict['links']
  49. project.services.append(Service(client=client, project=name, links=links, **service_dict))
  50. return project
  51. @classmethod
  52. def from_config(cls, name, config, client):
  53. dicts = []
  54. for service_name, service in list(config.items()):
  55. service['name'] = service_name
  56. dicts.append(service)
  57. return cls.from_dicts(name, dicts, client)
  58. def get_service(self, name):
  59. """
  60. Retrieve a service by name. Raises NoSuchService
  61. if the named service does not exist.
  62. """
  63. for service in self.services:
  64. if service.name == name:
  65. return service
  66. raise NoSuchService(name)
  67. def get_services(self, service_names=None):
  68. """
  69. Returns a list of this project's services filtered
  70. by the provided list of names, or all services if
  71. service_names is None or [].
  72. Preserves the original order of self.services.
  73. Raises NoSuchService if any of the named services
  74. do not exist.
  75. """
  76. if service_names is None or len(service_names) == 0:
  77. return self.services
  78. else:
  79. unsorted = [self.get_service(name) for name in service_names]
  80. return [s for s in self.services if s in unsorted]
  81. def recreate_containers(self, service_names=None):
  82. """
  83. For each service, create or recreate their containers.
  84. Returns a tuple with two lists. The first is a list of
  85. (service, old_container) tuples; the second is a list
  86. of (service, new_container) tuples.
  87. """
  88. old = []
  89. new = []
  90. for service in self.get_services(service_names):
  91. (s_old, s_new) = service.recreate_containers()
  92. old += [(service, container) for container in s_old]
  93. new += [(service, container) for container in s_new]
  94. return (old, new)
  95. def start(self, service_names=None, **options):
  96. for service in self.get_services(service_names):
  97. service.start(**options)
  98. def stop(self, service_names=None, **options):
  99. for service in reversed(self.get_services(service_names)):
  100. service.stop(**options)
  101. def kill(self, service_names=None, **options):
  102. for service in reversed(self.get_services(service_names)):
  103. service.kill(**options)
  104. def build(self, service_names=None, **options):
  105. for service in self.get_services(service_names):
  106. if service.can_be_built():
  107. service.build(**options)
  108. else:
  109. log.info('%s uses an image, skipping' % service.name)
  110. def remove_stopped(self, service_names=None, **options):
  111. for service in self.get_services(service_names):
  112. service.remove_stopped(**options)
  113. def containers(self, service_names=None, *args, **kwargs):
  114. l = []
  115. for service in self.get_services(service_names):
  116. for container in service.containers(*args, **kwargs):
  117. l.append(container)
  118. return l
  119. class NoSuchService(Exception):
  120. def __init__(self, name):
  121. self.name = name
  122. self.msg = "No such service: %s" % self.name
  123. def __str__(self):
  124. return self.msg
  125. class DependencyError(Exception):
  126. def __init__(self, msg):
  127. self.msg = msg
  128. def __str__(self):
  129. return self.msg