| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182 | import loggingimport refrom .const import LABEL_VERSIONfrom .container import Containerfrom .container import get_container_namelog = logging.getLogger(__name__)# TODO: remove this section when migrate_project_to_labels is removedNAME_RE = re.compile(r'^([^_]+)_([^_]+)_(run_)?(\d+)$')ERROR_MESSAGE_FORMAT = """Compose found the following containers without labels:{names_list}As of Compose 1.3.0, containers are identified with labels instead of namingconvention. If you want to continue using these containers, run:    $ docker-compose migrate-to-labelsAlternatively, remove them:    $ docker rm -f {rm_args}"""ONE_OFF_ADDENDUM_FORMAT = """You should also remove your one-off containers:    $ docker rm -f {rm_args}"""ONE_OFF_ERROR_MESSAGE_FORMAT = """Compose found the following containers without labels:{names_list}As of Compose 1.3.0, containers are identified with labels instead of naming convention.Remove them before continuing:    $ docker rm -f {rm_args}"""def check_for_legacy_containers(        client,        project,        services,        allow_one_off=True):    """Check if there are containers named using the old naming convention    and warn the user that those containers may need to be migrated to    using labels, so that compose can find them.    """    containers = get_legacy_containers(client, project, services, one_off=False)    if containers:        one_off_containers = get_legacy_containers(client, project, services, one_off=True)        raise LegacyContainersError(            [c.name for c in containers],            [c.name for c in one_off_containers],        )    if not allow_one_off:        one_off_containers = get_legacy_containers(client, project, services, one_off=True)        if one_off_containers:            raise LegacyOneOffContainersError(                [c.name for c in one_off_containers],            )class LegacyError(Exception):    def __unicode__(self):        return self.msg    __str__ = __unicode__class LegacyContainersError(LegacyError):    def __init__(self, names, one_off_names):        self.names = names        self.one_off_names = one_off_names        self.msg = ERROR_MESSAGE_FORMAT.format(            names_list="\n".join("    {}".format(name) for name in names),            rm_args=" ".join(names),        )        if one_off_names:            self.msg += ONE_OFF_ADDENDUM_FORMAT.format(rm_args=" ".join(one_off_names))class LegacyOneOffContainersError(LegacyError):    def __init__(self, one_off_names):        self.one_off_names = one_off_names        self.msg = ONE_OFF_ERROR_MESSAGE_FORMAT.format(            names_list="\n".join("    {}".format(name) for name in one_off_names),            rm_args=" ".join(one_off_names),        )def add_labels(project, container):    project_name, service_name, one_off, number = NAME_RE.match(container.name).groups()    if project_name != project.name or service_name not in project.service_names:        return    service = project.get_service(service_name)    service.recreate_container(container)def migrate_project_to_labels(project):    log.info("Running migration to labels for project %s", project.name)    containers = get_legacy_containers(        project.client,        project.name,        project.service_names,        one_off=False,    )    for container in containers:        add_labels(project, container)def get_legacy_containers(        client,        project,        services,        one_off=False):    return list(_get_legacy_containers_iter(        client,        project,        services,        one_off=one_off,    ))def _get_legacy_containers_iter(        client,        project,        services,        one_off=False):    containers = client.containers(all=True)    for service in services:        for container in containers:            if LABEL_VERSION in (container.get('Labels') or {}):                continue            name = get_container_name(container)            if has_container(project, service, name, one_off=one_off):                yield Container.from_ps(client, container)def has_container(project, service, name, one_off=False):    if not name or not is_valid_name(name, one_off):        return False    container_project, container_service, _container_number = parse_name(name)    return container_project == project and container_service == servicedef is_valid_name(name, one_off=False):    match = NAME_RE.match(name)    if match is None:        return False    if one_off:        return match.group(3) == 'run_'    else:        return match.group(3) is Nonedef parse_name(name):    match = NAME_RE.match(name)    (project, service_name, _, suffix) = match.groups()    return (project, service_name, int(suffix))
 |