iptv/M3U8/scrapers/livetvsx.py

214 lines
5.7 KiB
Python
Raw Normal View History

2026-01-26 22:39:47 -05:00
import asyncio
from functools import partial
2026-03-02 20:15:30 -05:00
from playwright.async_api import Browser, Page, TimeoutError
2026-04-21 15:55:40 -04:00
from selectolax.parser import HTMLParser
2026-01-26 22:39:47 -05:00
from .utils import Cache, Time, get_logger, leagues, network
log = get_logger(__name__)
urls: dict[str, dict[str, str | float]] = {}
2026-04-21 15:55:40 -04:00
TAG = "LTVSX"
2026-01-26 22:39:47 -05:00
CACHE_FILE = Cache(TAG, exp=10_800)
2026-04-21 15:55:40 -04:00
BASE_URL = "https://livetv.sx/export/webmasters.php"
2026-03-02 20:02:26 -05:00
2026-01-26 22:39:47 -05:00
async def process_event(
url: str,
url_num: int,
page: Page,
) -> str | None:
captured: list[str] = []
got_one = asyncio.Event()
handler = partial(
network.capture_req,
captured=captured,
got_one=got_one,
)
page.on("request", handler)
try:
2026-03-02 20:02:26 -05:00
resp = await page.goto(
2026-01-26 22:39:47 -05:00
url,
wait_until="domcontentloaded",
timeout=10_000,
2026-01-26 22:39:47 -05:00
)
2026-03-03 16:59:09 -05:00
if not resp or resp.status != 200:
log.warning(
f"URL {url_num}) Status Code: {resp.status if resp else 'None'}"
)
2026-03-02 20:02:26 -05:00
return
2026-01-26 22:39:47 -05:00
2026-03-02 20:02:26 -05:00
try:
event_a = page.locator('a[title*="Aliez"]').first
2026-01-26 22:39:47 -05:00
2026-03-02 20:02:26 -05:00
href = await event_a.get_attribute("href", timeout=1_250)
except TimeoutError:
log.warning(f"URL {url_num}) No valid sources found.")
2026-01-26 22:39:47 -05:00
return
2026-03-03 19:48:15 -05:00
event_url = href if href.startswith("http") else f"https:{href}"
2026-02-05 16:22:51 -05:00
2026-01-26 22:39:47 -05:00
await page.goto(
2026-03-02 20:02:26 -05:00
event_url,
2026-01-26 22:39:47 -05:00
wait_until="domcontentloaded",
timeout=5_000,
)
wait_task = asyncio.create_task(got_one.wait())
try:
await asyncio.wait_for(wait_task, timeout=6)
except asyncio.TimeoutError:
log.warning(f"URL {url_num}) Timed out waiting for M3U8.")
return
finally:
if not wait_task.done():
wait_task.cancel()
try:
await wait_task
except asyncio.CancelledError:
pass
if captured:
log.info(f"URL {url_num}) Captured M3U8")
return captured[0]
log.warning(f"URL {url_num}) No M3U8 captured after waiting.")
return
except Exception as e:
log.warning(f"URL {url_num}) {e}")
2026-01-26 22:39:47 -05:00
return
finally:
page.remove_listener("request", handler)
2026-04-21 15:55:40 -04:00
async def get_events(cached_keys: list[str]) -> list[dict[str, str]]:
events = []
2026-01-26 22:39:47 -05:00
2026-04-21 15:55:40 -04:00
php_data = await network.unvd_client.get(BASE_URL, params={"lang": "en"})
2026-01-26 22:39:47 -05:00
2026-04-21 15:55:40 -04:00
if php_data.status_code != 200:
2026-01-26 22:39:47 -05:00
return events
2026-04-21 15:55:40 -04:00
soup = HTMLParser(php_data.content)
2026-01-26 22:39:47 -05:00
2026-04-21 15:55:40 -04:00
if not (table := soup.css_first("table.tbl")):
return events
2026-01-26 22:39:47 -05:00
2026-04-21 15:55:40 -04:00
for row in table.css("tr > td"):
if not (event_tbl := row.css_first("table")):
2026-01-26 22:39:47 -05:00
continue
2026-04-21 15:55:40 -04:00
sport_elem = event_tbl.css_first(".spr")
league_elem = event_tbl.css_first(".cmp")
link_elem = event_tbl.css_first("a.title")
event_id_elem = row.css_first("div[id^='el']")
2026-01-29 17:37:15 -05:00
2026-04-21 15:55:40 -04:00
if not (league_elem and sport_elem and link_elem and event_id_elem):
continue
2026-01-26 22:39:47 -05:00
2026-04-21 15:55:40 -04:00
elif not (event_id := event_id_elem.attributes.get("id")):
2026-01-26 22:39:47 -05:00
continue
2026-04-21 15:55:40 -04:00
sport = sport_elem.text(strip=True)
league = league_elem.text(strip=True)
event_name = link_elem.text(strip=True)
2026-02-09 13:43:06 -05:00
2026-04-21 15:55:40 -04:00
if f"[{sport} - {league}] {event_name} ({TAG})" in cached_keys:
2026-01-26 22:39:47 -05:00
continue
2026-04-21 15:55:40 -04:00
events.append(
{
"sport": sport,
"league": league,
"event": event_name,
"link": f"https://cdn.livetv872.me/cache/links/en.{event_id[2:]}.html",
}
)
2026-01-26 22:39:47 -05:00
2026-04-21 15:55:40 -04:00
return events
2026-01-26 22:39:47 -05:00
async def scrape(browser: Browser) -> None:
cached_urls = CACHE_FILE.load()
valid_urls = {k: v for k, v in cached_urls.items() if v["url"]}
valid_count = cached_count = len(valid_urls)
urls.update(valid_urls)
log.info(f"Loaded {cached_count} event(s) from cache")
2026-01-26 22:55:04 -05:00
log.info('Scraping from "https://livetv.sx/enx/"')
2026-01-26 22:39:47 -05:00
2026-03-02 00:50:28 -05:00
if events := await get_events(cached_urls.keys()):
log.info(f"Processing {len(events)} new URL(s)")
2026-04-21 15:55:40 -04:00
now = Time.clean(Time.now())
2026-01-28 19:00:08 -05:00
async with network.event_context(browser, ignore_https=True) as context:
2026-01-26 22:39:47 -05:00
for i, ev in enumerate(events, start=1):
async with network.event_page(context) as page:
handler = partial(
process_event,
url=(link := ev["link"]),
2026-01-26 22:39:47 -05:00
url_num=i,
page=page,
)
url = await network.safe_process(
handler,
url_num=i,
semaphore=network.PW_S,
log=log,
2026-02-05 16:22:51 -05:00
timeout=20,
2026-01-26 22:39:47 -05:00
)
2026-04-21 15:55:40 -04:00
sport, league, event = (
2026-01-26 22:39:47 -05:00
ev["sport"],
2026-01-29 17:52:18 -05:00
ev["league"],
2026-01-26 22:39:47 -05:00
ev["event"],
)
2026-01-29 17:52:18 -05:00
key = f"[{sport} - {league}] {event} ({TAG})"
2026-01-26 22:39:47 -05:00
tvg_id, logo = leagues.get_tvg_info(sport, event)
entry = {
"url": url,
"logo": logo,
"base": "https://livetv.sx/enx/",
2026-04-21 15:55:40 -04:00
"timestamp": now.timestamp(),
2026-01-26 22:39:47 -05:00
"id": tvg_id or "Live.Event.us",
2026-01-27 16:34:41 -05:00
"link": link,
2026-01-26 22:39:47 -05:00
}
cached_urls[key] = entry
if url:
valid_count += 1
urls[key] = entry
2026-03-02 00:50:28 -05:00
log.info(f"Collected and cached {valid_count - cached_count} new event(s)")
2026-01-26 22:39:47 -05:00
else:
log.info("No new events found")
CACHE_FILE.write(cached_urls)