~maas-committers/maas/trunk

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
# Copyright 2012-2015 Canonical Ltd.  This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).

"""Django command: start a database shell.

Overrides the default implementation.
"""

__all__ = ['Command']

from optparse import make_option
import subprocess

from django.core.management.base import (
    BaseCommand,
    CommandError,
)
from django.core.management.commands import dbshell


class Command(dbshell.Command):
    """Customized "dbshell" command."""

    help = "Interactive psql shell for the MAAS database."

    option_list = BaseCommand.option_list + (
        make_option(
            '--database', default=None, help="Database to connect to."),
        make_option(
            '--dev', action='store_true', default=False,
            help=(
                "Connect to a development database database. "
                "Default is to start, and connect to, a system-installed "
                "database.")),
        make_option(
            '--installed', '-i', action='store_true', default=False,
            help=(
                "Connect to global, system-installed database. "
                "This is the default, unless a development environment is "
                "detected.")),
        )

    def get_development_database(self):
        database = None
        try:
            # Installed systems won't have this fixture.
            from maasserver.testing import database
        except ImportError:
            pass

        return database

    def handle(self, **options):
        database_fixture = self.get_development_database()
        if options.get('dev'):
            if database_fixture is None:
                raise CommandError("No development database found.")
        elif options.get('installed'):
            # If we have a development database but the user passed in
            # --installed, we need to use the system database instead.
            # So clear it out if we found one.
            database_fixture = None

        if database_fixture is None:
            # Access the global system-installed MAAS database.
            database_name = options.get('database')
            if database_name is None:
                database_name = 'maasdb'
            try:
                subprocess.check_call(
                    ['sudo', '-u', 'postgres', 'psql', database_name])
            except subprocess.CalledProcessError:
                # If psql fails to run, it will print a message to stderr.
                # Capturing that can get a little involved; psql might think
                # it was running noninteractively.  So just produce a standard
                # error message, and rely on the stderr output coming from
                # psql itself.
                raise CommandError("psql failed.")
        else:
            print("Using development database.")

            # Don't call up to Django's dbshell, because that ends up exec'ing
            # the shell, preventing this from clearing down the fixture.
            cluster = database_fixture.MAASClusterFixture(
                options.get('database'))
            with cluster:
                cluster.shell(cluster.dbname)