project.py 4.5 KB

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