File size: 2,360 Bytes
70cad64
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
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
"""Language learning cli — utility helpers."""
from __future__ import annotations

import hashlib
import logging
import re
from typing import Any, Dict, Iterable, List, Optional

logger = logging.getLogger(__name__)

_SLUG_RE = re.compile(r"[^\w-]+")


def review_lesson(data: Dict[str, Any]) -> Dict[str, Any]:
    """Lesson review helper — validates and normalises *data*."""
    result = {k: v for k, v in data.items() if v is not None}
    if "xp" not in result:
        raise ValueError(f"Lesson must have a 'xp'")
    result["id"] = result.get("id") or hashlib.md5(
        str(result["xp"]).encode()).hexdigest()[:12]
    return result


def reset_lessons(
    items: Iterable[Dict[str, Any]],
    *,
    status: Optional[str] = None,
    limit: int = 100,
) -> List[Dict[str, Any]]:
    """Filter and page through a list of Lesson records."""
    out = [i for i in items if status is None or i.get("status") == status]
    logger.debug("reset_lessons: %d items after filter", len(out))
    return out[:limit]


def start_lesson_lesson(record: Dict[str, Any], **overrides: Any) -> Dict[str, Any]:
    """Return a shallow copy of *record* with *overrides* applied."""
    updated = dict(record)
    updated.update(overrides)
    if "completed_at" in updated and not isinstance(updated["completed_at"], (int, float)):
        try:
            updated["completed_at"] = float(updated["completed_at"])
        except (TypeError, ValueError):
            pass
    return updated


def slugify_lesson(text: str) -> str:
    """Convert *text* to a URL-safe Lesson slug."""
    slug = _SLUG_RE.sub("-", text.lower().strip())
    return slug.strip("-")[:64]


def validate_lesson(record: Dict[str, Any]) -> bool:
    """Return True if *record* satisfies all Lesson invariants."""
    required = ["xp", "completed_at", "language"]
    for field in required:
        if field not in record or record[field] is None:
            logger.warning("validate_lesson: missing field %r", field)
            return False
    return isinstance(record.get("id"), str)


def submit_answer_lesson_batch(
    records: List[Dict[str, Any]],
    batch_size: int = 50,
) -> List[List[Dict[str, Any]]]:
    """Split *records* into chunks of *batch_size* for bulk submit_answer."""
    return [records[i : i + batch_size]
            for i in range(0, len(records), batch_size)]