Files @ 407076b32362
Branch filter:

Location: gimmecert/tests/test_cli.py - annotation

branko
GC-15: Prevent server command from overwriting artifacts and clean-up incorrect CLI tests:

- Added functional test for scenario where a server certificate has
already been issued.
- Updated server certificate issuance command not to overwrite the
artifacts.
- Fixed a couple of both server and init tests related to CLI parsing
to not create artifacts unless necessary, and if necessary to create
artifacts in temporary (test) directory. Otherwise intermittent
failures will happen.
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
0590c952df64
d8ec591edb40
e17474c5ef5d
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
d8ec591edb40
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
cf5813628ae1
e82ee8e024f4
e82ee8e024f4
e82ee8e024f4
e82ee8e024f4
e82ee8e024f4
e82ee8e024f4
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
0590c952df64
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
95e112372df9
9f09715ce550
9f09715ce550
9f09715ce550
130684592129
f2df385b1493
11712c273c16
11712c273c16
9f09715ce550
9f09715ce550
9f09715ce550
9f09715ce550
11712c273c16
f2df385b1493
f2df385b1493
f2df385b1493
f2df385b1493
f2df385b1493
11712c273c16
11712c273c16
f2df385b1493
f2df385b1493
f2df385b1493
f2df385b1493
11712c273c16
f2df385b1493
f2df385b1493
f2df385b1493
407076b32362
407076b32362
f2df385b1493
f2df385b1493
f2df385b1493
f2df385b1493
f2df385b1493
407076b32362
407076b32362
f2df385b1493
f2df385b1493
11712c273c16
11712c273c16
11712c273c16
407076b32362
407076b32362
11712c273c16
11712c273c16
11712c273c16
11712c273c16
11712c273c16
407076b32362
407076b32362
11712c273c16
11712c273c16
e17474c5ef5d
e17474c5ef5d
407076b32362
407076b32362
407076b32362
407076b32362
407076b32362
407076b32362
407076b32362
407076b32362
407076b32362
407076b32362
407076b32362
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
407076b32362
407076b32362
407076b32362
407076b32362
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
407076b32362
407076b32362
407076b32362
407076b32362
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
e17474c5ef5d
407076b32362
407076b32362
407076b32362
407076b32362
e17474c5ef5d
e17474c5ef5d
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
1d67951da5af
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
988ac40d5cec
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
1d67951da5af
7a2919409da2
7a2919409da2
7a2919409da2
7a2919409da2
7a2919409da2
7a2919409da2
7a2919409da2
7a2919409da2
7a2919409da2
7a2919409da2
# -*- coding: utf-8 -*-
#
# Copyright (C) 2018 Branko Majic
#
# This file is part of Gimmecert.
#
# Gimmecert is free software: you can redistribute it and/or modify it
# under the terms of the GNU General Public License as published by the Free
# Software Foundation, either version 3 of the License, or (at your option) any
# later version.
#
# Gimmecert is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along with
# Gimmecert.  If not, see <http://www.gnu.org/licenses/>.
#


import argparse

import gimmecert.cli
import gimmecert.decorators

import pytest
from unittest import mock


def test_get_parser_returns_parser():
    parser = gimmecert.cli.get_parser()

    assert isinstance(parser, argparse.ArgumentParser)


@mock.patch('gimmecert.cli.get_parser')
def test_main_invokes_get_parser(mock_get_parser):

    gimmecert.cli.main()

    mock_get_parser.assert_called_once_with()


@mock.patch('gimmecert.cli.get_parser')
def test_main_invokes_argument_parsing(mock_get_parser):
    mock_parser = mock.Mock()
    mock_get_parser.return_value = mock_parser

    gimmecert.cli.main()

    mock_parser.parse_args.assert_called_once_with()


def test_cli_parser_has_description():
    parser = gimmecert.cli.get_parser()

    assert parser.description


def test_parser_sets_up_default_callback_function():
    parser = gimmecert.cli.get_parser()

    assert callable(parser.get_default('func'))


@mock.patch('gimmecert.cli.argparse.ArgumentParser.print_usage')
def test_parser_default_callback_function_calls_print_usage(mock_print_usage):
    parser = gimmecert.cli.get_parser()
    func = parser.get_default('func')
    func(mock.Mock())

    assert mock_print_usage.called


@mock.patch('gimmecert.cli.get_parser')
def test_main_invokes_parser_function(mock_get_parser):
    mock_parser = mock.Mock()
    mock_args = mock.Mock()

    mock_parser.parse_args.return_value = mock_args
    mock_get_parser.return_value = mock_parser

    gimmecert.cli.main()

    mock_args.func.assert_called_once_with(mock_args)


def test_parser_help_contains_examples():
    parser = gimmecert.cli.get_parser()

    assert 'Examples' in parser.description


def test_setup_help_subcommand_parser_registered():
    registered_functions = gimmecert.decorators.get_subcommand_parser_setup_functions()

    assert gimmecert.cli.setup_help_subcommand_parser in registered_functions


@mock.patch('gimmecert.cli.get_subcommand_parser_setup_functions')
def test_get_parser_calls_setup_subcommand_parser_functions(mock_get_subcommand_parser_setup_functions):
    mock_setup1 = mock.Mock()
    mock_setup2 = mock.Mock()
    mock_get_subcommand_parser_setup_functions.return_value = [mock_setup1, mock_setup2]

    gimmecert.cli.get_parser()

    assert mock_setup1.called
    assert mock_setup2.called


def test_setup_help_subcommand_parser_adds_parser():
    mock_parser = mock.Mock()
    mock_subparsers = mock.Mock()

    gimmecert.cli.setup_help_subcommand_parser(mock_parser, mock_subparsers)

    assert mock_subparsers.add_parser.called


def test_help_subcommand_returns_parser():
    parser = argparse.ArgumentParser()
    subparsers = parser.add_subparsers()

    subparser = gimmecert.cli.setup_help_subcommand_parser(parser, subparsers)

    assert isinstance(subparser, argparse.ArgumentParser)


def test_help_subcommand_sets_function_callback():
    parser = argparse.ArgumentParser()
    subparsers = parser.add_subparsers()

    subparser = gimmecert.cli.setup_help_subcommand_parser(parser, subparsers)

    assert callable(subparser.get_default('func'))


def test_setup_init_subcommand_parser_registered():
    registered_functions = gimmecert.decorators.get_subcommand_parser_setup_functions()

    assert gimmecert.cli.setup_init_subcommand_parser in registered_functions


def test_setup_init_subcommand_returns_parser():
    parser = argparse.ArgumentParser()
    subparsers = parser.add_subparsers()

    subparser = gimmecert.cli.setup_init_subcommand_parser(parser, subparsers)

    assert isinstance(subparser, argparse.ArgumentParser)


def test_setup_init_subcommand_sets_function_callback():
    parser = argparse.ArgumentParser()
    subparsers = parser.add_subparsers()

    subparser = gimmecert.cli.setup_init_subcommand_parser(parser, subparsers)

    assert callable(subparser.get_default('func'))


@mock.patch('sys.argv', ['gimmecert', 'init'])
@mock.patch('gimmecert.cli.init')
def test_init_command_invoked_with_correct_parameters_no_options(mock_init, tmpdir):
    default_depth = 1

    tmpdir.chdir()

    gimmecert.cli.main()

    mock_init.assert_called_once_with(tmpdir.strpath, tmpdir.basename, default_depth)


@mock.patch('sys.argv', ['gimmecert', 'init', '-b', 'My Project'])
@mock.patch('gimmecert.cli.init')
def test_init_command_invoked_with_correct_parameters_with_options(mock_init, tmpdir):
    default_depth = 1

    tmpdir.chdir()

    gimmecert.cli.main()

    mock_init.assert_called_once_with(tmpdir.strpath, 'My Project', default_depth)


@mock.patch('sys.argv', ['gimmecert', 'init', '--ca-base-name', 'My Project'])
@mock.patch('gimmecert.cli.init')
def test_init_command_accepts_ca_base_name_option_long_form(mock_init):

    gimmecert.cli.main()  # Should not raise


@mock.patch('sys.argv', ['gimmecert', 'init', '-b', 'My Project'])
@mock.patch('gimmecert.cli.init')
def test_init_command_accepts_ca_base_name_option_short_form(mock_init):

    gimmecert.cli.main()  # Should not raise


@mock.patch('sys.argv', ['gimmecert', 'init', '--ca-hierarchy-depth', '3'])
@mock.patch('gimmecert.cli.init')
def test_init_command_accepts_ca_hierarchy_depth_option_long_form(mock_init):

    gimmecert.cli.main()  # Should not raise


@mock.patch('sys.argv', ['gimmecert', 'init', '-d', '3'])
@mock.patch('gimmecert.cli.init')
def test_init_command_accepts_ca_hierarchy_depth_option_short_form(mock_init):

    gimmecert.cli.main()  # Should not raise


@mock.patch('sys.argv', ['gimmecert', 'init'])
def test_init_command_exists_with_error_if_hierarchy_already_initialised(tmpdir):
    tmpdir.chdir()
    gimmecert.commands.init(tmpdir.strpath, tmpdir.basename, 1)

    with pytest.raises(SystemExit) as e_info:
        gimmecert.cli.main()

    assert e_info.value.code != 0


@mock.patch('sys.argv', ['gimmecert', 'server', '-h'])
def test_server_command_exists_and_accepts_help_flag():
    with pytest.raises(SystemExit) as e_info:
        gimmecert.cli.main()

    assert e_info.value.code == 0


def test_setup_server_subcommand_parser_registered():
    registered_functions = gimmecert.decorators.get_subcommand_parser_setup_functions()

    assert gimmecert.cli.setup_server_subcommand_parser in registered_functions


def test_setup_server_subcommand_parser_returns_parser():
    parser = argparse.ArgumentParser()
    subparsers = parser.add_subparsers()

    subparser = gimmecert.cli.setup_server_subcommand_parser(parser, subparsers)

    assert isinstance(subparser, argparse.ArgumentParser)


@mock.patch('sys.argv', ['gimmecert', 'server'])
def test_setup_server_subcommand_fails_without_arguments():
    with pytest.raises(SystemExit) as e_info:
        gimmecert.cli.main()

    assert e_info.value.code != 0


@mock.patch('sys.argv', ['gimmecert', 'server', 'myserver'])
@mock.patch('gimmecert.cli.server')
def test_setup_server_subcommand_succeeds_with_just_entity_name_argument(mock_server):
    # We are just testing the parsing here.
    mock_server.return_value = True, "Fake message"

    gimmecert.cli.main()  # Should not raise.


@mock.patch('sys.argv', ['gimmecert', 'server', 'myserver', 'myserver.example.com'])
@mock.patch('gimmecert.cli.server')
def test_setup_server_subcommand_succeeds_with_entity_name_argument_and_one_dns_name(mock_server):
    # We are just testing the parsing here.
    mock_server.return_value = True, "Fake message"

    gimmecert.cli.main()  # Should not raise.


@mock.patch('sys.argv', ['gimmecert', 'server', 'myserver', 'myserver1.example.com', 'myserver2.example.com', 'myserver3.example.com', 'myserver4.example.com'])
@mock.patch('gimmecert.cli.server')
def test_setup_server_subcommand_succeeds_with_entity_name_argument_and_four_dns_names(mock_server):
    # We are just testing the parsing here.
    mock_server.return_value = True, "Fake message"

    gimmecert.cli.main()  # Should not raise.


def test_setup_server_subcommand_sets_function_callback():
    parser = argparse.ArgumentParser()
    subparsers = parser.add_subparsers()

    subparser = gimmecert.cli.setup_server_subcommand_parser(parser, subparsers)

    assert callable(subparser.get_default('func'))


@mock.patch('sys.argv', ['gimmecert', 'server', 'myserver'])
@mock.patch('gimmecert.cli.server')
def test_server_command_invoked_with_correct_parameters_without_extra_dns_names(mock_server, tmpdir):
    mock_server.return_value = True, "Bogus"

    tmpdir.chdir()

    gimmecert.cli.main()

    mock_server.assert_called_once_with(tmpdir.strpath, 'myserver', [])


@mock.patch('sys.argv', ['gimmecert', 'server', 'myserver', 'service.local', 'service.example.com'])
@mock.patch('gimmecert.cli.server')
def test_server_command_invoked_with_correct_parameters_with_extra_dns_names(mock_server, tmpdir):
    mock_server.return_value = True, "Bogus"

    tmpdir.chdir()

    gimmecert.cli.main()

    mock_server.assert_called_once_with(tmpdir.strpath, 'myserver', ['service.local', 'service.example.com'])


@mock.patch('sys.argv', ['gimmecert', 'server', 'myserver'])
def test_server_command_exists_with_error_if_hierarchy_not_initialised(tmpdir):
    tmpdir.chdir()

    with pytest.raises(SystemExit) as e_info:
        gimmecert.cli.main()

    assert e_info.value.code != 0