-
Notifications
You must be signed in to change notification settings - Fork 12
/
test_scripts_commons.py
189 lines (143 loc) · 7.13 KB
/
test_scripts_commons.py
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
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
import getpass
import logging
from unittest.mock import call
import pytest
from pythonanywhere.scripts_commons import (
ScriptSchema,
get_logger,
get_task_from_id,
tabulate_formats,
)
@pytest.mark.tasks
class TestScriptSchema:
def test_validates_boolean(self):
schema = ScriptSchema({"--toggle": ScriptSchema.boolean})
for val in (True, False, None):
result = schema.validate_user_input({"--toggle": val})
assert result == {"toggle": val}
def test_exits_because_boolean_not_satisfied(self, mocker):
mock_exit = mocker.patch("pythonanywhere.scripts_commons.sys.exit")
mock_snake = mocker.patch("pythonanywhere.scripts_commons.snakesay")
mock_warning = mocker.patch("pythonanywhere.scripts_commons.logger.warning")
schema = ScriptSchema({"--toggle": ScriptSchema.boolean})
schema.validate_user_input({"--toggle": "not valid value"})
assert mock_exit.call_args == call(1)
assert mock_warning.call_count == 1
assert mock_snake.call_args == call(
"Key '--toggle' error:\nOr(None, <class 'bool'>) did not validate 'not valid value'\n"
"'not valid value' should be instance of 'bool'"
)
def test_validates_bour(self):
schema = ScriptSchema({"--hour": ScriptSchema.hour})
for val in (None, 0, 12, 23):
result = schema.validate_user_input({"--hour": val})
assert result == {"hour": val}
def test_exits_because_hour_not_satisfied(self, mocker):
mock_exit = mocker.patch("pythonanywhere.scripts_commons.sys.exit")
mock_snake = mocker.patch("pythonanywhere.scripts_commons.snakesay")
mock_warning = mocker.patch("pythonanywhere.scripts_commons.logger.warning")
schema = ScriptSchema({"--hour": ScriptSchema.hour})
schema.validate_user_input({"--hour": 30})
assert mock_exit.call_args == call(1)
assert mock_warning.call_count == 1
assert mock_snake.call_args == call("--hour has to be in 0..23")
def test_validates_minute(self):
schema = ScriptSchema({"--minute": ScriptSchema.minute})
for val in (None, 1, 30, 59):
result = schema.validate_user_input({"--minute": val})
assert result == {"minute": val}
def test_exits_because_minute_not_satisfied(self, mocker):
mock_exit = mocker.patch("pythonanywhere.scripts_commons.sys.exit")
mock_snake = mocker.patch("pythonanywhere.scripts_commons.snakesay")
mock_warning = mocker.patch("pythonanywhere.scripts_commons.logger.warning")
schema = ScriptSchema({"--minute": ScriptSchema.minute})
schema.validate_user_input({"--minute": 60})
assert mock_exit.call_args == call(1)
assert mock_warning.call_count == 1
assert mock_snake.call_args == call("--minute has to be in 0..59")
def test_validates_using_conversions(self):
schema = ScriptSchema({"<id>": ScriptSchema.id_required})
result = schema.validate_user_input({"<id>": 42}, conversions={"id": "task_id"})
assert result == {"task_id": 42}
def test_exits_because_id_not_satisfied(self, mocker):
mock_exit = mocker.patch("pythonanywhere.scripts_commons.sys.exit")
mock_snake = mocker.patch("pythonanywhere.scripts_commons.snakesay")
mock_warning = mocker.patch("pythonanywhere.scripts_commons.logger.warning")
schema = ScriptSchema({"<id>": ScriptSchema.id_required})
schema.validate_user_input({"<id>": None})
assert mock_exit.call_args == call(1)
assert mock_warning.call_count == 1
assert mock_snake.call_args == call("<id> has to be an integer")
def test_validates_tabulate_format(self):
schema = ScriptSchema({"--format": ScriptSchema.tabulate_format})
for val in tabulate_formats:
result = schema.validate_user_input({"--format": val})
assert result == {"format": val}
def test_exits_because_tabulate_format_not_satisfied(self, mocker):
mock_exit = mocker.patch("pythonanywhere.scripts_commons.sys.exit")
mock_snake = mocker.patch("pythonanywhere.scripts_commons.snakesay")
mock_warning = mocker.patch("pythonanywhere.scripts_commons.logger.warning")
schema = ScriptSchema({"--format": ScriptSchema.tabulate_format})
schema.validate_user_input({"--format": "non_existing_format"})
assert mock_exit.call_args == call(1)
assert mock_warning.call_count == 1
assert mock_snake.call_args == call(
"--format should match one of: plain, simple, github, grid, fancy_grid, pipe, orgtbl, "
"jira, presto, psql, rst, mediawiki, moinmoin, youtrack, html, latex, latex_raw, "
"latex_booktabs, textile"
)
@pytest.mark.tasks
class TestScriptSchemaConvert:
def test_replaces_default_strings(self):
was = ("--option", "<arg>")
should_be = ("option", "arg")
for string, expected in zip(was, should_be):
assert ScriptSchema({}).convert(string) == expected
def test_returns_unchanged_string(self):
assert ScriptSchema({}).convert("will_not_be_changed") == "will_not_be_changed"
@pytest.mark.tasks
class TestGetLogger:
def test_returns_pa_logger(self, caplog):
# get_logger should change logger to WARNING, i.e. level 30
caplog.set_level(logging.INFO, logger="pythonanywhere")
logger = get_logger()
assert logger.name == "pythonanywhere"
assert logger.level == 30
def test_returns_pa_logger_info(self, caplog):
# get_logger should change logger to INFO, i.e. level 20
caplog.set_level(logging.WARNING, logger="pythonanywhere")
logger = get_logger(set_info=True)
assert logger.name == "pythonanywhere"
assert logger.level == 20
@pytest.mark.tasks
class TestGetTaskFromId:
def test_returns_task(self, mocker):
user = getpass.getuser()
specs = {
"can_enable": False,
"command": "echo foo",
"enabled": True,
"hour": 10,
"interval": "daily",
"logfile": f"/user/{user}/files/foo",
"minute": 23,
"printable_time": "10:23",
"task_id": 42,
"username": user,
}
mock_task = mocker.patch("pythonanywhere.scripts_commons.Task.from_id")
for spec, value in specs.items():
setattr(mock_task.return_value, spec, value)
task = get_task_from_id(42)
for spec, value in specs.items():
assert getattr(task, spec) == value
def test_catches_exception(self, mocker):
mock_exit = mocker.patch("pythonanywhere.scripts_commons.sys.exit")
mock_snakesay = mocker.patch("pythonanywhere.scripts_commons.snakesay")
mock_warning = mocker.patch("pythonanywhere.scripts_commons.logger.warning")
mock_task_from_id = mocker.patch("pythonanywhere.task.Task.from_id")
mock_task_from_id.side_effect = Exception("exception")
task = get_task_from_id(1)
assert mock_snakesay.call_args == call("exception")
assert mock_warning.call_count == 1
assert mock_exit.call_args == call(1)