|
@@ -14,6 +14,10 @@ class BuildError(Exception):
|
|
|
pass
|
|
|
|
|
|
|
|
|
+class CannotBeScaledError(Exception):
|
|
|
+ pass
|
|
|
+
|
|
|
+
|
|
|
class Service(object):
|
|
|
def __init__(self, name, client=None, project='default', links=[], **options):
|
|
|
if not re.match('^[a-zA-Z0-9]+$', name):
|
|
@@ -56,6 +60,40 @@ class Service(object):
|
|
|
log.info("Killing %s..." % c.name)
|
|
|
c.kill(**options)
|
|
|
|
|
|
+ def scale(self, desired_num):
|
|
|
+ if not self.can_be_scaled():
|
|
|
+ raise CannotBeScaledError()
|
|
|
+
|
|
|
+ # Create enough containers
|
|
|
+ containers = self.containers(stopped=True)
|
|
|
+ while len(containers) < desired_num:
|
|
|
+ containers.append(self.create_container())
|
|
|
+
|
|
|
+ running_containers = []
|
|
|
+ stopped_containers = []
|
|
|
+ for c in containers:
|
|
|
+ if c.is_running:
|
|
|
+ running_containers.append(c)
|
|
|
+ else:
|
|
|
+ stopped_containers.append(c)
|
|
|
+ running_containers.sort(key=lambda c: c.number)
|
|
|
+ stopped_containers.sort(key=lambda c: c.number)
|
|
|
+
|
|
|
+ # Stop containers
|
|
|
+ while len(running_containers) > desired_num:
|
|
|
+ c = running_containers.pop()
|
|
|
+ log.info("Stopping %s..." % c.name)
|
|
|
+ c.stop(timeout=1)
|
|
|
+ stopped_containers.append(c)
|
|
|
+
|
|
|
+ # Start containers
|
|
|
+ while len(running_containers) < desired_num:
|
|
|
+ c = stopped_containers.pop(0)
|
|
|
+ log.info("Starting %s..." % c.name)
|
|
|
+ c.start()
|
|
|
+ running_containers.append(c)
|
|
|
+
|
|
|
+
|
|
|
def remove_stopped(self, **options):
|
|
|
for c in self.containers(stopped=True):
|
|
|
if not c.is_running:
|
|
@@ -231,6 +269,12 @@ class Service(object):
|
|
|
"""
|
|
|
return '%s_%s' % (self.project, self.name)
|
|
|
|
|
|
+ def can_be_scaled(self):
|
|
|
+ for port in self.options.get('ports', []):
|
|
|
+ if ':' in str(port):
|
|
|
+ return False
|
|
|
+ return True
|
|
|
+
|
|
|
|
|
|
NAME_RE = re.compile(r'^([^_]+)_([^_]+)_(run_)?(\d+)$')
|
|
|
|