| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 | from __future__ import print_functionfrom __future__ import unicode_literalsimport loggingimport sysimport reimport signalfrom inspect import getdocfrom .. import __version__from ..project import NoSuchService, ConfigurationErrorfrom ..service import CannotBeScaledErrorfrom .command import Commandfrom .formatter import Formatterfrom .log_printer import LogPrinterfrom .utils import yesnofrom ..packages.docker.client import APIErrorfrom .errors import UserErrorfrom .docopt_command import NoSuchCommandfrom .socketclient import SocketClientlog = logging.getLogger(__name__)def main():    console_handler = logging.StreamHandler(stream=sys.stderr)    console_handler.setFormatter(logging.Formatter())    console_handler.setLevel(logging.INFO)    root_logger = logging.getLogger()    root_logger.addHandler(console_handler)    root_logger.setLevel(logging.DEBUG)    # Disable requests logging    logging.getLogger("requests").propagate = False    try:        command = TopLevelCommand()        command.sys_dispatch()    except KeyboardInterrupt:        log.error("\nAborting.")        sys.exit(1)    except (UserError, NoSuchService, ConfigurationError) as e:        log.error(e.msg)        sys.exit(1)    except NoSuchCommand as e:        log.error("No such command: %s", e.command)        log.error("")        log.error("\n".join(parse_doc_section("commands:", getdoc(e.supercommand))))        sys.exit(1)    except APIError as e:        log.error(e.explanation)        sys.exit(1)# stolen from docopt masterdef parse_doc_section(name, source):    pattern = re.compile('^([^\n]*' + name + '[^\n]*\n?(?:[ \t].*?(?:\n|$))*)',                         re.IGNORECASE | re.MULTILINE)    return [s.strip() for s in pattern.findall(source)]class TopLevelCommand(Command):    """Punctual, lightweight development environments using Docker.    Usage:      fig [options] [COMMAND] [ARGS...]      fig -h|--help    Options:      --verbose            Show more output      --version            Print version and exit      -f, --file FILE      Specify an alternate fig file (default: fig.yml)    Commands:      build     Build or rebuild services      help      Get help on a command      kill      Kill containers      logs      View output from containers      ps        List containers      rm        Remove stopped containers      run       Run a one-off command      scale     Set number of containers for a service      start     Start services      stop      Stop services      up        Create and start containers    """    def docopt_options(self):        options = super(TopLevelCommand, self).docopt_options()        options['version'] = "fig %s" % __version__        return options    def build(self, options):        """        Build or rebuild services.        Services are built once and then tagged as `project_service`,        e.g. `figtest_db`. If you change a service's `Dockerfile` or the        contents of its build directory, you can run `fig build` to rebuild it.        Usage: build [SERVICE...]        """        self.project.build(service_names=options['SERVICE'])    def help(self, options):        """        Get help on a command.        Usage: help COMMAND        """        command = options['COMMAND']        if not hasattr(self, command):            raise NoSuchCommand(command, self)        raise SystemExit(getdoc(getattr(self, command)))    def kill(self, options):        """        Force stop service containers.        Usage: kill [SERVICE...]        """        self.project.kill(service_names=options['SERVICE'])    def logs(self, options):        """        View output from containers.        Usage: logs [SERVICE...]        """        containers = self.project.containers(service_names=options['SERVICE'], stopped=True)        print("Attaching to", list_containers(containers))        LogPrinter(containers, attach_params={'logs': True}).run()    def ps(self, options):        """        List containers.        Usage: ps [options] [SERVICE...]        Options:            -q    Only display IDs        """        containers = self.project.containers(service_names=options['SERVICE'], stopped=True) + self.project.containers(service_names=options['SERVICE'], one_off=True)        if options['-q']:            for container in containers:                print(container.id)        else:            headers = [                'Name',                'Command',                'State',                'Ports',            ]            rows = []            for container in containers:                command = container.human_readable_command                if len(command) > 30:                    command = '%s ...' % command[:26]                rows.append([                    container.name,                    command,                    container.human_readable_state,                    container.human_readable_ports,                ])            print(Formatter().table(headers, rows))    def rm(self, options):        """        Remove stopped service containers.        Usage: rm [options] [SERVICE...]        Options:            --force   Don't ask to confirm removal            -v        Remove volumes associated with containers        """        all_containers = self.project.containers(service_names=options['SERVICE'], stopped=True)        stopped_containers = [c for c in all_containers if not c.is_running]        if len(stopped_containers) > 0:            print("Going to remove", list_containers(stopped_containers))            if options.get('--force') \                    or yesno("Are you sure? [yN] ", default=False):                self.project.remove_stopped(                    service_names=options['SERVICE'],                    v=options.get('-v', False)                )        else:            print("No stopped containers")    def run(self, options):        """        Run a one-off command on a service.        For example:            $ fig run web python manage.py shell        Note that this will not start any services that the command's service        links to. So if, for example, your one-off command talks to your        database, you will need to run `fig up -d db` first.        Usage: run [options] SERVICE COMMAND [ARGS...]        Options:            -d    Detached mode: Run container in the background, print new                  container name            -T    Disable pseudo-tty allocation. By default `fig run`                  allocates a TTY.            --rm  Remove container after run. Ignored in detached mode.        """        service = self.project.get_service(options['SERVICE'])        tty = True        if options['-d'] or options['-T'] or not sys.stdin.isatty():            tty = False        container_options = {            'command': [options['COMMAND']] + options['ARGS'],            'tty': tty,            'stdin_open': not options['-d'],        }        container = service.create_container(one_off=True, **container_options)        if options['-d']:            service.start_container(container, ports=None)            print(container.name)        else:            with self._attach_to_container(container.id, raw=tty) as c:                service.start_container(container, ports=None)                c.run()            if options['--rm']:                container.wait()                log.info("Removing %s..." % container.name)                self.client.remove_container(container.id)    def scale(self, options):        """        Set number of containers to run for a service.        Numbers are specified in the form `service=num` as arguments.        For example:            $ fig scale web=2 worker=3        Usage: scale [SERVICE=NUM...]        """        for s in options['SERVICE=NUM']:            if '=' not in s:                raise UserError('Arguments to scale should be in the form service=num')            service_name, num = s.split('=', 1)            try:                num = int(num)            except ValueError:                raise UserError('Number of containers for service "%s" is not a number' % service)            try:                self.project.get_service(service_name).scale(num)            except CannotBeScaledError:                raise UserError('Service "%s" cannot be scaled because it specifies a port on the host. If multiple containers for this service were created, the port would clash.\n\nRemove the ":" from the port definition in fig.yml so Docker can choose a random port for each container.' % service_name)    def start(self, options):        """        Start existing containers.        Usage: start [SERVICE...]        """        self.project.start(service_names=options['SERVICE'])    def stop(self, options):        """        Stop running containers without removing them.        They can be started again with `fig start`.        Usage: stop [SERVICE...]        """        self.project.stop(service_names=options['SERVICE'])    def up(self, options):        """        Build, (re)create, start and attach to containers for a service.        By default, `fig up` will aggregate the output of each container, and        when it exits, all containers will be stopped. If you run `fig up -d`,        it'll start the containers in the background and leave them running.        If there are existing containers for a service, `fig up` will stop        and recreate them (preserving mounted volumes with volumes-from),        so that changes in `fig.yml` are picked up.        Usage: up [options] [SERVICE...]        Options:            -d    Detached mode: Run containers in the background, print new                  container names        """        detached = options['-d']        new = self.project.up(service_names=options['SERVICE'])        if not detached:            to_attach = [c for (s, c) in new]            print("Attaching to", list_containers(to_attach))            log_printer = LogPrinter(to_attach, attach_params={"logs": True})            try:                log_printer.run()            finally:                def handler(signal, frame):                    self.project.kill(service_names=options['SERVICE'])                    sys.exit(0)                signal.signal(signal.SIGINT, handler)                print("Gracefully stopping... (press Ctrl+C again to force)")                self.project.stop(service_names=options['SERVICE'])    def _attach_to_container(self, container_id, raw=False):        socket_in = self.client.attach_socket(container_id, params={'stdin': 1, 'stream': 1})        socket_out = self.client.attach_socket(container_id, params={'stdout': 1, 'logs': 1, 'stream': 1})        socket_err = self.client.attach_socket(container_id, params={'stderr': 1, 'logs': 1, 'stream': 1})        return SocketClient(            socket_in=socket_in,            socket_out=socket_out,            socket_err=socket_err,            raw=raw,        )def list_containers(containers):    return ", ".join(c.name for c in containers)
 |