123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899 |
- #!/usr/bin/env python3
-
- from collections import defaultdict
- import argparse
- import docker
- import etcd
- import sys
-
-
- def etcd_put(client, prefix, obj):
- for key, value in obj.items():
- new_prefix = "%s/%s" % (prefix, key)
-
- if isinstance(value, dict):
- etcd_put(client, new_prefix, value)
- else:
- client.write(new_prefix, str(value))
-
-
- def get_addresses(container):
- return {k: v['IPAddress'] for k, v in container['NetworkSettings']['Networks'].items()}
-
-
- def get_ports(container):
- ports = defaultdict(dict)
- for port in container['Ports']:
- ports[port['Type']][port['PrivatePort']] = port['PublicPort'] if 'PublicPort' in port else 0
- return ports
-
-
- def add_containers(infos):
- global containers, host_index, label_index, network_index
- for info in infos:
- container = {
- 'host': host,
- 'image': info['Image'],
- 'labels': info['Labels'],
- 'net': {
- 'addr': get_addresses(info),
- 'ports': get_ports(info)
- }
- }
-
- name = info['Names'][0][1:]
- containers[name] = container
-
- for k, v in container['labels'].items():
- label_index[k][name] = v
- for k, v in container['net']['addr'].items():
- network_index[k][name] = v
-
- host_index[host][name] = name
-
-
- def write_all():
- etcd_put(etcd_client, prefix + '/containers', containers)
- etcd_put(etcd_client, prefix + '/labels', label_index)
- etcd_put(etcd_client, prefix + '/networks', network_index)
- etcd_put(etcd_client, prefix + '/hosts', host_index)
-
-
- parser = argparse.ArgumentParser()
- parser.add_argument('--name', help='Name of this docker host', default='unknown')
- parser.add_argument('--etcd-port', type=int, help='Port to connect to etcd on', default=2379)
- parser.add_argument('--etcd-host', help='Host to connect to etcd on', default='etcd')
- parser.add_argument('--etcd-prefix', help='Prefix to use when adding keys to etcd', default='/docker')
- args = parser.parse_args()
-
- docker_client = docker.Client(base_url='unix://var/run/docker.sock')
- etcd_client = etcd.Client(host=args.etcd_host, port=args.etcd_port)
- prefix = args.etcd_prefix
- host = args.name
-
- event_gen = docker_client.events(decode=True, filters={'type': 'container', 'event': ['die', 'start']})
-
- containers = {}
- label_index = defaultdict(dict)
- network_index = defaultdict(dict)
- host_index = defaultdict(dict)
-
- add_containers(docker_client.containers())
-
- try:
- etcd_client.delete(prefix, recursive=True)
- except etcd.EtcdKeyNotFound:
- pass
-
- write_all()
-
- for event in event_gen:
- if event['Action'] == 'start':
- print('New container %s' % event['id'])
- add_containers(docker_client.containers(filters={'id': event['id']}))
- write_all()
- elif event['Action'] == 'die':
- print('Dead container %s' % event['id'])
- else:
- print('Unexpected event %s' % event['Action'])
|