kryten-webqueue 0.2.3__tar.gz → 0.2.5__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.3 → kryten_webqueue-0.2.5}/PKG-INFO +1 -1
  2. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/catalog/sync.py +48 -8
  3. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/pyproject.toml +1 -1
  4. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/.github/workflows/python-publish.yml +0 -0
  5. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/.github/workflows/release.yml +0 -0
  6. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/.gitignore +0 -0
  7. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/README.md +0 -0
  8. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/config.example.json +0 -0
  9. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/deploy/kryten-webqueue.service +0 -0
  10. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/deploy/nginx-queue.conf +0 -0
  11. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/docs/IMPLEMENTATION_SPEC.md +0 -0
  12. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/docs/IMPL_API_GATE.md +0 -0
  13. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/docs/IMPL_ECONOMY.md +0 -0
  14. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/docs/IMPL_KRYTEN_PY.md +0 -0
  15. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/docs/IMPL_ROBOT.md +0 -0
  16. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/docs/PRE_PLAN_GAPS.md +0 -0
  17. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/docs/PRODUCT_PLAN.md +0 -0
  18. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/__init__.py +0 -0
  19. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/__main__.py +0 -0
  20. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/api_gate/__init__.py +0 -0
  21. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/api_gate/client.py +0 -0
  22. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/app.py +0 -0
  23. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/auth/__init__.py +0 -0
  24. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/auth/otp.py +0 -0
  25. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/auth/rate_limit.py +0 -0
  26. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/auth/session.py +0 -0
  27. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/catalog/__init__.py +0 -0
  28. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/catalog/db.py +0 -0
  29. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/catalog/images.py +0 -0
  30. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/config.py +0 -0
  31. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/playlists/__init__.py +0 -0
  32. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/playlists/fire.py +0 -0
  33. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/playlists/importer.py +0 -0
  34. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/playlists/scheduler.py +0 -0
  35. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/queue/__init__.py +0 -0
  36. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/queue/ordering.py +0 -0
  37. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/queue/poller.py +0 -0
  38. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/queue/shadow.py +0 -0
  39. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/__init__.py +0 -0
  40. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/admin_playlists.py +0 -0
  41. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/admin_queue.py +0 -0
  42. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/admin_schedules.py +0 -0
  43. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/auth.py +0 -0
  44. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/catalog.py +0 -0
  45. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/pages.py +0 -0
  46. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/queue.py +0 -0
  47. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/routes/user.py +0 -0
  48. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/static/css/main.css +0 -0
  49. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/static/js/main.js +0 -0
  50. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/templates/admin/index.html +0 -0
  51. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/templates/auth/login.html +0 -0
  52. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/templates/base.html +0 -0
  53. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/templates/catalog/browse.html +0 -0
  54. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/templates/queue/index.html +0 -0
  55. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/templates/user/dashboard.html +0 -0
  56. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/ws/__init__.py +0 -0
  57. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/kryten_webqueue/ws/handler.py +0 -0
  58. {kryten_webqueue-0.2.3 → kryten_webqueue-0.2.5}/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.3
3
+ Version: 0.2.5
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,15 +1,41 @@
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
 
11
31
  def __init__(self, *, mediacms_url: str, mediacms_token: str, db):
12
- self._url = mediacms_url.rstrip("/")
32
+ # Strip any accidental /api/v1 suffix — the sync code appends it itself
33
+ url = mediacms_url.rstrip("/")
34
+ for suffix in ("/api/v1", "/api"):
35
+ if url.endswith(suffix):
36
+ url = url[: -len(suffix)]
37
+ break
38
+ self._url = url
13
39
  self._token = mediacms_token
14
40
  self._db = db
15
41
  self._client = httpx.AsyncClient(
@@ -28,12 +54,22 @@ class CatalogSync:
28
54
  try:
29
55
  page = 1
30
56
  while True:
31
- resp = await self._client.get(
32
- f"{self._url}/api/v1/media",
33
- params={"page": page, "page_size": 50},
34
- )
57
+ api_url = f"{self._url}/api/v1/media"
58
+ try:
59
+ resp = await self._client.get(
60
+ api_url,
61
+ params={"page": page, "page_size": 50},
62
+ )
63
+ except httpx.TransportError as exc:
64
+ logger.error(_describe_httpx_error(exc, api_url))
65
+ stats["errors"] += 1
66
+ break
67
+
35
68
  if resp.status_code != 200:
36
- logger.error(f"MediaCMS API returned {resp.status_code}")
69
+ logger.error(
70
+ f"MediaCMS API returned {resp.status_code} for URL {resp.url} — "
71
+ f"body: {resp.text[:200]}"
72
+ )
37
73
  stats["errors"] += 1
38
74
  break
39
75
 
@@ -58,8 +94,12 @@ class CatalogSync:
58
94
 
59
95
  await self._db.finish_sync_log(log_id, stats, "completed")
60
96
  logger.info(f"Catalog sync: {stats}")
61
- except Exception as e:
62
- logger.exception(f"Catalog sync failed: {type(e).__name__}: {e}")
97
+ except httpx.TransportError as exc:
98
+ logger.error(_describe_httpx_error(exc, f"{self._url}/api/v1/media"))
99
+ stats["errors"] += 1
100
+ await self._db.finish_sync_log(log_id, stats, "error")
101
+ except Exception as exc:
102
+ logger.exception(f"Catalog sync failed: {type(exc).__name__}: {exc}")
63
103
  stats["errors"] += 1
64
104
  await self._db.finish_sync_log(log_id, stats, "error")
65
105
 
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "kryten-webqueue"
3
- version = "0.2.3"
3
+ version = "0.2.5"
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"