2016-01-18 20:02:25 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
"""
|
|
|
|
Migrate a Compose file from the V1 format in Compose 1.5 to the V2 format
|
|
|
|
supported by Compose 1.6+
|
|
|
|
"""
|
|
|
|
from __future__ import absolute_import
|
|
|
|
from __future__ import unicode_literals
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import logging
|
|
|
|
import sys
|
|
|
|
|
|
|
|
import ruamel.yaml
|
|
|
|
|
2016-01-26 17:01:43 +01:00
|
|
|
from compose.config.types import VolumeSpec
|
|
|
|
|
2016-01-18 20:02:25 +01:00
|
|
|
|
|
|
|
log = logging.getLogger('migrate')
|
|
|
|
|
|
|
|
|
|
|
|
def migrate(content):
|
|
|
|
data = ruamel.yaml.load(content, ruamel.yaml.RoundTripLoader)
|
|
|
|
|
|
|
|
service_names = data.keys()
|
2016-01-26 16:53:02 +01:00
|
|
|
|
2016-01-18 20:02:25 +01:00
|
|
|
for name, service in data.items():
|
2016-01-26 17:08:34 +01:00
|
|
|
warn_for_links(name, service)
|
|
|
|
warn_for_external_links(name, service)
|
|
|
|
rewrite_net(service, service_names)
|
|
|
|
rewrite_build(service)
|
|
|
|
rewrite_logging(service)
|
|
|
|
rewrite_volumes_from(service, service_names)
|
2016-01-18 20:02:25 +01:00
|
|
|
|
2016-01-26 16:54:18 +01:00
|
|
|
services = {name: data.pop(name) for name in data.keys()}
|
|
|
|
|
2016-02-17 09:56:49 +01:00
|
|
|
data['version'] = "2"
|
2016-01-26 16:54:18 +01:00
|
|
|
data['services'] = services
|
2016-01-26 17:01:43 +01:00
|
|
|
create_volumes_section(data)
|
2016-01-26 16:54:18 +01:00
|
|
|
|
2016-01-18 20:02:25 +01:00
|
|
|
return data
|
|
|
|
|
|
|
|
|
2016-01-26 17:08:34 +01:00
|
|
|
def warn_for_links(name, service):
|
|
|
|
links = service.get('links')
|
|
|
|
if links:
|
|
|
|
example_service = links[0].partition(':')[0]
|
|
|
|
log.warn(
|
|
|
|
"Service {name} has links, which no longer create environment "
|
|
|
|
"variables such as {example_service_upper}_PORT. "
|
|
|
|
"If you are using those in your application code, you should "
|
|
|
|
"instead connect directly to the hostname, e.g. "
|
|
|
|
"'{example_service}'."
|
|
|
|
.format(name=name, example_service=example_service,
|
|
|
|
example_service_upper=example_service.upper()))
|
|
|
|
|
|
|
|
|
|
|
|
def warn_for_external_links(name, service):
|
|
|
|
external_links = service.get('external_links')
|
|
|
|
if external_links:
|
|
|
|
log.warn(
|
|
|
|
"Service {name} has external_links: {ext}, which now work "
|
|
|
|
"slightly differently. In particular, two containers must be "
|
|
|
|
"connected to at least one network in common in order to "
|
|
|
|
"communicate, even if explicitly linked together.\n\n"
|
|
|
|
"Either connect the external container to your app's default "
|
|
|
|
"network, or connect both the external container and your "
|
|
|
|
"service's containers to a pre-existing network. See "
|
|
|
|
"https://docs.docker.com/compose/networking/ "
|
|
|
|
"for more on how to do this."
|
|
|
|
.format(name=name, ext=external_links))
|
|
|
|
|
|
|
|
|
|
|
|
def rewrite_net(service, service_names):
|
|
|
|
if 'net' in service:
|
|
|
|
network_mode = service.pop('net')
|
|
|
|
|
|
|
|
# "container:<service name>" is now "service:<service name>"
|
|
|
|
if network_mode.startswith('container:'):
|
|
|
|
name = network_mode.partition(':')[2]
|
|
|
|
if name in service_names:
|
|
|
|
network_mode = 'service:{}'.format(name)
|
|
|
|
|
|
|
|
service['network_mode'] = network_mode
|
|
|
|
|
|
|
|
|
|
|
|
def rewrite_build(service):
|
|
|
|
if 'dockerfile' in service:
|
|
|
|
service['build'] = {
|
|
|
|
'context': service.pop('build'),
|
|
|
|
'dockerfile': service.pop('dockerfile'),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def rewrite_logging(service):
|
|
|
|
if 'log_driver' in service:
|
|
|
|
service['logging'] = {'driver': service.pop('log_driver')}
|
|
|
|
if 'log_opt' in service:
|
|
|
|
service['logging']['options'] = service.pop('log_opt')
|
|
|
|
|
|
|
|
|
|
|
|
def rewrite_volumes_from(service, service_names):
|
|
|
|
for idx, volume_from in enumerate(service.get('volumes_from', [])):
|
|
|
|
if volume_from.split(':', 1)[0] not in service_names:
|
|
|
|
service['volumes_from'][idx] = 'container:%s' % volume_from
|
|
|
|
|
|
|
|
|
2016-01-26 17:01:43 +01:00
|
|
|
def create_volumes_section(data):
|
|
|
|
named_volumes = get_named_volumes(data['services'])
|
|
|
|
if named_volumes:
|
|
|
|
log.warn(
|
|
|
|
"Named volumes ({names}) must be explicitly declared. Creating a "
|
|
|
|
"'volumes' section with declarations.\n\n"
|
|
|
|
"For backwards-compatibility, they've been declared as external. "
|
|
|
|
"If you don't mind the volume names being prefixed with the "
|
|
|
|
"project name, you can remove the 'external' option from each one."
|
|
|
|
.format(names=', '.join(list(named_volumes))))
|
|
|
|
|
|
|
|
data['volumes'] = named_volumes
|
|
|
|
|
|
|
|
|
|
|
|
def get_named_volumes(services):
|
|
|
|
volume_specs = [
|
|
|
|
VolumeSpec.parse(volume)
|
|
|
|
for service in services.values()
|
|
|
|
for volume in service.get('volumes', [])
|
|
|
|
]
|
|
|
|
names = {
|
|
|
|
spec.external
|
|
|
|
for spec in volume_specs
|
|
|
|
if spec.is_named_volume
|
|
|
|
}
|
|
|
|
return {name: {'external': True} for name in names}
|
|
|
|
|
|
|
|
|
2016-01-18 20:02:25 +01:00
|
|
|
def write(stream, new_format, indent, width):
|
|
|
|
ruamel.yaml.dump(
|
|
|
|
new_format,
|
|
|
|
stream,
|
|
|
|
Dumper=ruamel.yaml.RoundTripDumper,
|
|
|
|
indent=indent,
|
|
|
|
width=width)
|
|
|
|
|
|
|
|
|
|
|
|
def parse_opts(args):
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument("filename", help="Compose file filename.")
|
|
|
|
parser.add_argument("-i", "--in-place", action='store_true')
|
|
|
|
parser.add_argument(
|
|
|
|
"--indent", type=int, default=2,
|
|
|
|
help="Number of spaces used to indent the output yaml.")
|
|
|
|
parser.add_argument(
|
|
|
|
"--width", type=int, default=80,
|
|
|
|
help="Number of spaces used as the output width.")
|
|
|
|
return parser.parse_args()
|
|
|
|
|
|
|
|
|
|
|
|
def main(args):
|
2016-02-16 14:46:47 +01:00
|
|
|
logging.basicConfig(format='\033[33m%(levelname)s:\033[37m %(message)s\033[0m\n')
|
2016-01-18 20:02:25 +01:00
|
|
|
|
|
|
|
opts = parse_opts(args)
|
|
|
|
|
|
|
|
with open(opts.filename, 'r') as fh:
|
|
|
|
new_format = migrate(fh.read())
|
|
|
|
|
|
|
|
if opts.in_place:
|
|
|
|
output = open(opts.filename, 'w')
|
|
|
|
else:
|
|
|
|
output = sys.stdout
|
|
|
|
write(output, new_format, opts.indent, opts.width)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main(sys.argv)
|