quizzy 0.4.0__py3-none-any.whl → 0.6.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- quizzy/app.py +44 -10
- quizzy/models.py +56 -12
- {quizzy-0.4.0.dist-info → quizzy-0.6.0.dist-info}/METADATA +2 -2
- quizzy-0.6.0.dist-info/RECORD +10 -0
- quizzy-0.4.0.dist-info/RECORD +0 -10
- {quizzy-0.4.0.dist-info → quizzy-0.6.0.dist-info}/WHEEL +0 -0
- {quizzy-0.4.0.dist-info → quizzy-0.6.0.dist-info}/entry_points.txt +0 -0
- {quizzy-0.4.0.dist-info → quizzy-0.6.0.dist-info}/licenses/LICENSE +0 -0
quizzy/app.py
CHANGED
@@ -189,7 +189,7 @@ class QuestionButton(widgets.Button):
|
|
189
189
|
def wait_for_result(team: QuestionScreenResult) -> None:
|
190
190
|
if team is None:
|
191
191
|
return
|
192
|
-
|
192
|
+
|
193
193
|
self.disabled = True
|
194
194
|
self.question.answered = True
|
195
195
|
if isinstance(team, NoCorrectAnswerType):
|
@@ -244,8 +244,17 @@ class QuizzyApp(app.App[None]):
|
|
244
244
|
|
245
245
|
|
246
246
|
def get_arg_parser() -> argparse.ArgumentParser:
|
247
|
-
parser = argparse.ArgumentParser(
|
247
|
+
parser = argparse.ArgumentParser(
|
248
|
+
prog=__name__.split(".")[0],
|
249
|
+
description="A quiz app to run in the terminal or the browser",
|
250
|
+
epilog=(
|
251
|
+
"When an unfinished quiz is closed, the state is saved to a timestamped quiz file in the current working "
|
252
|
+
"directory."
|
253
|
+
),
|
254
|
+
)
|
255
|
+
|
248
256
|
parser.add_argument("--version", action="version", version=f"%(prog)s {__version__}")
|
257
|
+
parser.add_argument("--dump-state-on-exit", action="store_true", help="Dump game state on exit")
|
249
258
|
|
250
259
|
serve_group = parser.add_argument_group("Serve options")
|
251
260
|
serve_group.add_argument("--serve", action="store_true", help="Serve the app through the browser")
|
@@ -256,18 +265,43 @@ def get_arg_parser() -> argparse.ArgumentParser:
|
|
256
265
|
return parser
|
257
266
|
|
258
267
|
|
268
|
+
def serve_app(host: str, port: int) -> None:
|
269
|
+
"""Serve the app through textual-serve.
|
270
|
+
|
271
|
+
:param host: Host to serve the app on
|
272
|
+
:param port: Port to serve the app on
|
273
|
+
"""
|
274
|
+
# The --serve flag is set, drop it from the args and serve the app instead through textual-serve
|
275
|
+
args = list(sys.argv)
|
276
|
+
args.remove("--serve")
|
277
|
+
|
278
|
+
server.Server(shlex.join(args), host=host, port=port).serve()
|
279
|
+
|
280
|
+
|
281
|
+
def run_app(quiz_file: pathlib.Path, dump_state_on_exit: bool = False) -> None:
|
282
|
+
"""Run the app in the terminal.
|
283
|
+
|
284
|
+
:param quizfile: File to use for this quiz
|
285
|
+
"""
|
286
|
+
try:
|
287
|
+
config = models.load_config(quiz_file)
|
288
|
+
except Exception as e:
|
289
|
+
print(f"Error loading quiz file: {e}", file=sys.stderr)
|
290
|
+
sys.exit(1)
|
291
|
+
app = QuizzyApp(config)
|
292
|
+
try:
|
293
|
+
app.run()
|
294
|
+
finally:
|
295
|
+
if dump_state_on_exit:
|
296
|
+
models.dump_config_if_not_finished(config)
|
297
|
+
|
298
|
+
|
259
299
|
def main() -> NoReturn:
|
260
300
|
parser = get_arg_parser()
|
261
301
|
namespace = parser.parse_args()
|
262
302
|
|
263
303
|
if namespace.serve:
|
264
|
-
|
265
|
-
args = list(sys.argv)
|
266
|
-
args.remove("--serve")
|
267
|
-
|
268
|
-
server.Server(shlex.join(args), host=namespace.host, port=namespace.port).serve()
|
304
|
+
serve_app(namespace.host, namespace.port)
|
269
305
|
else:
|
270
|
-
|
271
|
-
app = QuizzyApp(config)
|
272
|
-
app.run()
|
306
|
+
run_app(namespace.quizfile, namespace.dump_state_on_exit)
|
273
307
|
sys.exit(0)
|
quizzy/models.py
CHANGED
@@ -1,8 +1,11 @@
|
|
1
1
|
from __future__ import annotations
|
2
2
|
|
3
|
+
import datetime
|
4
|
+
import json
|
3
5
|
import pathlib
|
4
|
-
from typing import Self
|
6
|
+
from typing import Annotated, Any, Self
|
5
7
|
|
8
|
+
import pydantic
|
6
9
|
import yaml
|
7
10
|
from pydantic import Field, model_validator
|
8
11
|
from pydantic.dataclasses import dataclass
|
@@ -21,12 +24,20 @@ class Team:
|
|
21
24
|
return self.name
|
22
25
|
|
23
26
|
|
27
|
+
def is_multiple_of_100(v: int) -> int:
|
28
|
+
if v % 100 != 0:
|
29
|
+
raise ValueError("Value must be a multiple of 100")
|
30
|
+
return v
|
31
|
+
|
32
|
+
|
24
33
|
@dataclass
|
25
34
|
class Question:
|
26
|
-
question: str
|
27
|
-
answer: str
|
28
|
-
value: int = Field(
|
29
|
-
|
35
|
+
question: str = Field(description="The question to ask. Markdown is supported.")
|
36
|
+
answer: str = Field(description="The answer to the question. Markdown is supported.")
|
37
|
+
value: Annotated[int, pydantic.AfterValidator(is_multiple_of_100)] = Field(
|
38
|
+
gt=0, description="The value of the question in points. Must be a multiple of 100."
|
39
|
+
)
|
40
|
+
answered: bool = Field(default=False, description="Whether the question has been answered already.")
|
30
41
|
|
31
42
|
def __str__(self) -> str:
|
32
43
|
return self.question
|
@@ -34,8 +45,8 @@ class Question:
|
|
34
45
|
|
35
46
|
@dataclass
|
36
47
|
class Category:
|
37
|
-
name: str
|
38
|
-
questions: list[Question] = Field(max_length=5)
|
48
|
+
name: str = Field(description="The name of the category.")
|
49
|
+
questions: list[Question] = Field(max_length=5, description="The questions in this category.")
|
39
50
|
|
40
51
|
@model_validator(mode="after")
|
41
52
|
def sort_questions(self) -> Self:
|
@@ -47,13 +58,46 @@ class Category:
|
|
47
58
|
|
48
59
|
@dataclass
|
49
60
|
class Config:
|
50
|
-
categories: list[Category] = Field(max_length=5)
|
51
|
-
teams: list[Team] = Field(default=[Team("Team 1"), Team("Team 2")])
|
52
|
-
# TODO: Support random questions
|
61
|
+
categories: list[Category] = Field(max_length=5, description="The categories in the quiz.")
|
62
|
+
teams: list[Team] = Field(default=[Team("Team 1"), Team("Team 2")], description="The teams in the quiz.")
|
53
63
|
|
64
|
+
def is_finished(self) -> bool:
|
65
|
+
for category in self.categories:
|
66
|
+
for question in category.questions:
|
67
|
+
if not question.answered:
|
68
|
+
return False
|
69
|
+
return True
|
54
70
|
|
55
|
-
|
71
|
+
|
72
|
+
def _get_dict(path: pathlib.Path) -> dict[str, Any]:
|
73
|
+
"""Get the dictionary from a file.
|
74
|
+
|
75
|
+
:param path: Path of the file to load.
|
76
|
+
:raises ValueError: Raised if the file format is not supported.
|
77
|
+
:return: Dictionary with the contents of the file.
|
78
|
+
"""
|
56
79
|
with path.open() as f:
|
57
|
-
|
80
|
+
if path.suffix == ".json":
|
81
|
+
return json.load(f)
|
82
|
+
if path.suffix in {".yml", ".yaml"}:
|
83
|
+
return yaml.safe_load(f)
|
84
|
+
raise ValueError(f"Unsupported file format: {path.suffix}")
|
58
85
|
|
86
|
+
|
87
|
+
def load_config(path: pathlib.Path) -> Config:
|
88
|
+
raw = _get_dict(path)
|
59
89
|
return Config(**raw)
|
90
|
+
|
91
|
+
|
92
|
+
def dump_config_if_not_finished(config: Config) -> None:
|
93
|
+
"""Dump the config if any of the questions is not answered yet.
|
94
|
+
|
95
|
+
:param config: _description_
|
96
|
+
"""
|
97
|
+
if config.is_finished():
|
98
|
+
return
|
99
|
+
p = pathlib.Path() / f"quizzy-run-{datetime.datetime.now().strftime('%Y%m%d-%H%M%S')}.json"
|
100
|
+
print(f"This quiz is not finished yet. Saving current state to '{p}'.")
|
101
|
+
print("This file can be re-used later.")
|
102
|
+
with p.open("wb") as f:
|
103
|
+
f.write(pydantic.TypeAdapter(Config).dump_json(config))
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.4
|
2
2
|
Name: quizzy
|
3
|
-
Version: 0.
|
3
|
+
Version: 0.6.0
|
4
4
|
Summary: A Python TUI quiz app
|
5
5
|
Author-email: Jonas Ehrlich <jonas.ehrlich@gmail.com>
|
6
6
|
License-Expression: MIT
|
@@ -16,7 +16,7 @@ Description-Content-Type: text/markdown
|
|
16
16
|
|
17
17
|
A quiz app using [textual](https://textual.textualize.io/).
|
18
18
|
|
19
|
-

|
20
20
|
|
21
21
|
## Configuration and Questions
|
22
22
|
|
@@ -0,0 +1,10 @@
|
|
1
|
+
quizzy/__init__.py,sha256=IILIKzO2OO60Rhsh3SqhdKHDf7XI_RLRwzztBGZZmHY,78
|
2
|
+
quizzy/__main__.py,sha256=GiZpRpqbaOVG0pDtJswF_VsLByVFP2VMhHgm0Lm0LWw,36
|
3
|
+
quizzy/app.py,sha256=gO5aGchsu9PLjepdrgcaglHbo2Wc_1B-AFOszpjU--A,11086
|
4
|
+
quizzy/models.py,sha256=CxWOvAi5TbZZK76-aKRIPZhrDbHNM4U3tcrZaQ7xCJs,3189
|
5
|
+
quizzy/quizzy.tcss,sha256=VcPEGQ1J7oGatPf3kB-Hzo9AHQdrI7cOV8hDT_l9-3A,1810
|
6
|
+
quizzy-0.6.0.dist-info/METADATA,sha256=ffwK69fcadxDA5KDfIcdO_OMfKedgPT9y6s6jZ5ASVU,1598
|
7
|
+
quizzy-0.6.0.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
|
8
|
+
quizzy-0.6.0.dist-info/entry_points.txt,sha256=2RiVMgcS4h7TM59u9dyBQFm53cG6Eyekmb8fqZ5rXHM,48
|
9
|
+
quizzy-0.6.0.dist-info/licenses/LICENSE,sha256=JWN3MACgsucm6y_rlL_2MUzst0-wNh-Wab3XkxtfVzM,1070
|
10
|
+
quizzy-0.6.0.dist-info/RECORD,,
|
quizzy-0.4.0.dist-info/RECORD
DELETED
@@ -1,10 +0,0 @@
|
|
1
|
-
quizzy/__init__.py,sha256=IILIKzO2OO60Rhsh3SqhdKHDf7XI_RLRwzztBGZZmHY,78
|
2
|
-
quizzy/__main__.py,sha256=GiZpRpqbaOVG0pDtJswF_VsLByVFP2VMhHgm0Lm0LWw,36
|
3
|
-
quizzy/app.py,sha256=K1f0_wzcoAsTHbu4VPxKqIdb-j5rrLawWkR1Ez9PD-s,10173
|
4
|
-
quizzy/models.py,sha256=JrAc2IvtXByIKNg_lX7uQMK19zwi2WsYcNbttXccbOs,1265
|
5
|
-
quizzy/quizzy.tcss,sha256=VcPEGQ1J7oGatPf3kB-Hzo9AHQdrI7cOV8hDT_l9-3A,1810
|
6
|
-
quizzy-0.4.0.dist-info/METADATA,sha256=lk2-Z9mhWOUrVla4dbrC5VmFshu4Z8KXIMmbjhYww44,1608
|
7
|
-
quizzy-0.4.0.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
|
8
|
-
quizzy-0.4.0.dist-info/entry_points.txt,sha256=2RiVMgcS4h7TM59u9dyBQFm53cG6Eyekmb8fqZ5rXHM,48
|
9
|
-
quizzy-0.4.0.dist-info/licenses/LICENSE,sha256=JWN3MACgsucm6y_rlL_2MUzst0-wNh-Wab3XkxtfVzM,1070
|
10
|
-
quizzy-0.4.0.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|