~abentley/juju-ci-tools/old-from-new

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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
#!/usr/bin/env python
"""Assess juju blocks prevent users from making changes and models"""

from __future__ import print_function

import argparse
import logging
import sys

import yaml

from assess_min_version import (
    JujuAssertionError
)
from deploy_stack import (
    BootstrapManager,
    deploy_dummy_stack,
)
from utility import (
    add_basic_testing_arguments,
    configure_logging,
    wait_for_removed_services,
)


__metaclass__ = type


log = logging.getLogger("assess_block")


def get_block_list(client):
    """Return a list of blocks and their status."""
    return yaml.safe_load(client.get_juju_output(
        'block list', '--format', 'yaml'))


def make_block_list(client, des_env, rm_obj, all_changes):
    """Return a manually made list of blocks and their status"""
    block_list = [
        {'block': client.destroy_model_command, 'enabled': des_env},
        {'block': 'remove-object', 'enabled': rm_obj},
        {'block': 'all-changes', 'enabled': all_changes}]
    for block in block_list:
        if block['enabled']:
            block['message'] = ''
    return block_list


def test_blocked(client, command, args, include_e=True):
    """Test if a command is blocked as expected"""
    try:
        if command == 'deploy':
            client.deploy(args)
        elif command == 'remove-service':
            client.remove_service(args)
        else:
            client.juju(command, args, include_e=include_e)
        raise JujuAssertionError()
    except Exception:
        pass


def assess_block_destroy_model(client, charm_series):
    """Test block destroy-model

    When "block destroy-model" is set,
    the model cannot be destroyed, but objects
    can be added, related, and removed.
    """
    client.juju('block ' + client.destroy_model_command, ())
    block_list = get_block_list(client)
    if block_list != make_block_list(client, True, False, False):
        raise JujuAssertionError(block_list)
    test_blocked(client, client.destroy_model_command,
                 ('-y', client.env.environment), include_e=False)
    # Adding, relating, and removing are not blocked.
    deploy_dummy_stack(client, charm_series)


def assess_block_remove_object(client, charm_series):
    """Test block remove-object

    When "block remove-object" is set,
    objects can be added and related, but they
    cannot be removed or the model/environment deleted.
    """
    client.juju('block remove-object', ())
    block_list = get_block_list(client)
    if block_list != make_block_list(client, False, True, False):
        raise JujuAssertionError(block_list)
    test_blocked(client, client.destroy_model_command,
                 ('-y', client.env.environment), include_e=False)
    # Adding and relating are not blocked.
    deploy_dummy_stack(client, charm_series)
    test_blocked(client, 'remove-service', 'dummy-source')
    test_blocked(client, 'remove-unit', ('dummy-source/1',))
    test_blocked(client, 'remove-relation', ('dummy-source', 'dummy-sink'))


def assess_block_all_changes(client, charm_series):
    """Test Block Functionality: block all-changes"""
    client.juju('remove-relation', ('dummy-source', 'dummy-sink'))
    client.juju('block all-changes', ())
    block_list = get_block_list(client)
    if block_list != make_block_list(client, False, False, True):
        raise JujuAssertionError(block_list)
    test_blocked(client, 'add-relation', ('dummy-source', 'dummy-sink'))
    test_blocked(client, 'unexpose', ('dummy-sink',))
    test_blocked(client, 'remove-service', 'dummy-sink')
    client.juju('unblock all-changes', ())
    client.juju('unexpose', ('dummy-sink',))
    client.juju('block all-changes', ())
    test_blocked(client, 'expose', ('dummy-sink',))
    client.juju('unblock all-changes', ())
    client.remove_service('dummy-sink')
    wait_for_removed_services(client, 'dummy-sink')
    client.juju('block all-changes', ())
    test_blocked(client, 'deploy', ('dummy-sink',))
    test_blocked(client, client.destroy_model_command,
                 ('-y', client.env.environment), include_e=False)


def assess_unblock(client, type):
    """Test Block Functionality
    unblock destroy-model/remove-object/all-changes."""
    client.juju('unblock ' + type, ())
    block_list = get_block_list(client)
    if block_list != make_block_list(client, False, False, False):
        raise JujuAssertionError(block_list)
    if type == client.destroy_model_command:
        client.remove_service('dummy-source')
        wait_for_removed_services(client, 'dummy-source')
        client.remove_service('dummy-sink')
        wait_for_removed_services(client, 'dummy-sink')


def assess_block(client, charm_series):
    """Test Block Functionality:
    block/unblock destroy-model/remove-object/all-changes.
    """
    block_list = get_block_list(client)
    client.wait_for_started()
    expected_none_blocked = make_block_list(client, False, False, False)
    if block_list != expected_none_blocked:
        raise JujuAssertionError(block_list)
    assess_block_destroy_model(client, charm_series)
    assess_unblock(client, client.destroy_model_command)
    assess_block_remove_object(client, charm_series)
    assess_unblock(client, 'remove-object')
    assess_block_all_changes(client, charm_series)
    assess_unblock(client, 'all-changes')


def parse_args(argv):
    """Parse all arguments."""
    parser = argparse.ArgumentParser(description="Test Block Functionality")
    add_basic_testing_arguments(parser)
    parser.set_defaults(series='trusty')
    return parser.parse_args(argv)


def main(argv=None):
    args = parse_args(argv)
    configure_logging(args.verbose)
    bs_manager = BootstrapManager.from_args(args)
    with bs_manager.booted_context(args.upload_tools):
        assess_block(bs_manager.client, bs_manager.series)
    return 0


if __name__ == '__main__':
    sys.exit(main())