kryten-webqueue 0.2.4__tar.gz → 0.2.6__tar.gz

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.
Files changed (58) hide show
  1. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/PKG-INFO +1 -1
  2. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/catalog/sync.py +43 -13
  3. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/pyproject.toml +1 -1
  4. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/.github/workflows/python-publish.yml +0 -0
  5. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/.github/workflows/release.yml +0 -0
  6. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/.gitignore +0 -0
  7. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/README.md +0 -0
  8. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/config.example.json +0 -0
  9. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/deploy/kryten-webqueue.service +0 -0
  10. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/deploy/nginx-queue.conf +0 -0
  11. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/docs/IMPLEMENTATION_SPEC.md +0 -0
  12. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/docs/IMPL_API_GATE.md +0 -0
  13. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/docs/IMPL_ECONOMY.md +0 -0
  14. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/docs/IMPL_KRYTEN_PY.md +0 -0
  15. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/docs/IMPL_ROBOT.md +0 -0
  16. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/docs/PRE_PLAN_GAPS.md +0 -0
  17. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/docs/PRODUCT_PLAN.md +0 -0
  18. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/__init__.py +0 -0
  19. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/__main__.py +0 -0
  20. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/api_gate/__init__.py +0 -0
  21. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/api_gate/client.py +0 -0
  22. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/app.py +0 -0
  23. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/auth/__init__.py +0 -0
  24. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/auth/otp.py +0 -0
  25. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/auth/rate_limit.py +0 -0
  26. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/auth/session.py +0 -0
  27. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/catalog/__init__.py +0 -0
  28. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/catalog/db.py +0 -0
  29. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/catalog/images.py +0 -0
  30. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/config.py +0 -0
  31. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/playlists/__init__.py +0 -0
  32. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/playlists/fire.py +0 -0
  33. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/playlists/importer.py +0 -0
  34. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/playlists/scheduler.py +0 -0
  35. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/queue/__init__.py +0 -0
  36. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/queue/ordering.py +0 -0
  37. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/queue/poller.py +0 -0
  38. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/queue/shadow.py +0 -0
  39. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/__init__.py +0 -0
  40. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/admin_playlists.py +0 -0
  41. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/admin_queue.py +0 -0
  42. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/admin_schedules.py +0 -0
  43. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/auth.py +0 -0
  44. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/catalog.py +0 -0
  45. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/pages.py +0 -0
  46. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/queue.py +0 -0
  47. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/routes/user.py +0 -0
  48. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/static/css/main.css +0 -0
  49. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/static/js/main.js +0 -0
  50. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/templates/admin/index.html +0 -0
  51. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/templates/auth/login.html +0 -0
  52. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/templates/base.html +0 -0
  53. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/templates/catalog/browse.html +0 -0
  54. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/templates/queue/index.html +0 -0
  55. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/templates/user/dashboard.html +0 -0
  56. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/ws/__init__.py +0 -0
  57. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/ws/handler.py +0 -0
  58. {kryten_webqueue-0.2.4 → kryten_webqueue-0.2.6}/kryten_webqueue/ws/manager.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: kryten-webqueue
3
- Version: 0.2.4
3
+ Version: 0.2.6
4
4
  Summary: Netflix/Tubi-style catalog browser and pay-to-play queue management for CyTube
5
5
  Author: grobertson
6
6
  License-Expression: MIT
@@ -1,10 +1,30 @@
1
1
  import httpx
2
2
  import logging
3
3
  from datetime import datetime, UTC
4
+ from urllib.parse import urlparse
4
5
 
5
6
  logger = logging.getLogger(__name__)
6
7
 
7
8
 
9
+ def _describe_httpx_error(exc: Exception, url: str) -> str:
10
+ """Return a human-readable description of an httpx network error."""
11
+ host = urlparse(url).netloc or url
12
+ if isinstance(exc, httpx.ConnectTimeout):
13
+ return f"Connection to {host} timed out — check host/port and firewall"
14
+ if isinstance(exc, httpx.ConnectError):
15
+ cause = str(exc.__cause__ or exc).lower()
16
+ if "name or service not known" in cause or "nodename nor servname" in cause or "getaddrinfo" in cause:
17
+ return f"DNS lookup failed for {host} — check mediacms_url in config"
18
+ if "connection refused" in cause:
19
+ return f"Connection refused by {host} — service may be down"
20
+ return f"Could not connect to {host}: {exc.__cause__ or exc}"
21
+ if isinstance(exc, httpx.ReadTimeout):
22
+ return f"Read timed out waiting for response from {host}"
23
+ if isinstance(exc, httpx.TimeoutException):
24
+ return f"Request to {host} timed out"
25
+ return f"{type(exc).__name__} connecting to {host}: {exc}"
26
+
27
+
8
28
  class CatalogSync:
9
29
  """Synchronizes catalog data from MediaCMS."""
10
30
 
@@ -32,12 +52,19 @@ class CatalogSync:
32
52
  stats = {"seen": 0, "new": 0, "updated": 0, "errors": 0}
33
53
 
34
54
  try:
35
- page = 1
36
- while True:
37
- resp = await self._client.get(
38
- f"{self._url}/api/v1/media",
39
- params={"page": page, "page_size": 50},
40
- )
55
+ next_url: str | None = f"{self._url}/api/v1/media"
56
+ params: dict = {"page_size": 50}
57
+ page = 0
58
+
59
+ while next_url:
60
+ page += 1
61
+ try:
62
+ resp = await self._client.get(next_url, params=params if page == 1 else None)
63
+ except httpx.TransportError as exc:
64
+ logger.error(_describe_httpx_error(exc, next_url))
65
+ stats["errors"] += 1
66
+ break
67
+
41
68
  if resp.status_code != 200:
42
69
  logger.error(
43
70
  f"MediaCMS API returned {resp.status_code} for URL {resp.url} — "
@@ -60,15 +87,18 @@ class CatalogSync:
60
87
  logger.warning(f"Error processing {media.get('friendly_token')}: {e}")
61
88
  stats["errors"] += 1
62
89
 
63
- # Check for next page
64
- if isinstance(data, dict) and not data.get("next"):
65
- break
66
- page += 1
90
+ # Follow the next URL from the response — don't construct it ourselves
91
+ next_url = data.get("next") if isinstance(data, dict) else None
92
+ logger.debug(f"Catalog sync page {page}: seen={stats['seen']} next={next_url!r}")
67
93
 
68
94
  await self._db.finish_sync_log(log_id, stats, "completed")
69
- logger.info(f"Catalog sync: {stats}")
70
- except Exception as e:
71
- logger.exception(f"Catalog sync failed: {type(e).__name__}: {e}")
95
+ logger.info(f"Catalog sync complete: {stats} ({page} pages)")
96
+ except httpx.TransportError as exc:
97
+ logger.error(_describe_httpx_error(exc, f"{self._url}/api/v1/media"))
98
+ stats["errors"] += 1
99
+ await self._db.finish_sync_log(log_id, stats, "error")
100
+ except Exception as exc:
101
+ logger.exception(f"Catalog sync failed: {type(exc).__name__}: {exc}")
72
102
  stats["errors"] += 1
73
103
  await self._db.finish_sync_log(log_id, stats, "error")
74
104
 
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "kryten-webqueue"
3
- version = "0.2.4"
3
+ version = "0.2.6"
4
4
  description = "Netflix/Tubi-style catalog browser and pay-to-play queue management for CyTube"
5
5
  readme = "README.md"
6
6
  license = "MIT"