xvideos SSR's JSON-LD VideoObject (duration/title/uploadDate) + on-page /models/ (perf) + /tags/. Sample: median ~10.5min, 93% >=3min. Pilot (2 pages): 29 new, 100% playable + visible + tagged (performers sparse — xvideos 'new' is amateur-heavy; /models/ tagged mostly on studio rips). - XVideosBrowseScraper (JSON-LD + page-parse models/tags), in ALL_BROWSE_SCRAPERS. - deep_crawl._PAGE_CAP: per-sitetag depth cap; xvideoscom=1800 (~newest 50k). At the cap the tube is marked exhausted (reset -> incremental re-sweep) so a mega-tube cannot monopolize the round-robin or balloon the DB. - ported yesporn.py into the public repo (was prod-only, like hdporngg) ending the __init__ public/prod divergence. youporn rejected: JSON-LD lacks actor/keywords, its /pornstar//category/ links are A-Z nav not scene-specific. xhamster: 429/Cloudflare from the VPS IP. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
321 lines
12 KiB
Python
321 lines
12 KiB
Python
"""yesporn.vip — latest-vids browse scraper.
|
|
|
|
Dołączony 2026-05-27. Identyfikowany przez user audit jako "scraper-of-paysites"
|
|
(DogFart / HardX / TeamSkeet / Vixen / Brazzers content). Wcześniejszy theporndude
|
|
audit pomylił domeny: `yespornvip.com` (z theporndude rankingu) redirectuje przez
|
|
pdude.link do `porndudecams.com` affiliate spam — kanoniczna domena ma TLD `.vip`.
|
|
|
|
Czemu wart (parity z porndoe):
|
|
- **JSON-LD VideoObject** w każdym scene page: name, description, uploadDate
|
|
(ISO `YYYY-MM-DDTHH:MM:SS`), duration (ISO `PT0H39M00S`), thumbnailUrl
|
|
(BunnyCDN: `yesnn.b-cdn.net/contents/videos_screenshots/...`).
|
|
- **`<meta property="video:duration" content="2340">`** — durations już w sekundach
|
|
(fallback gdy ISO-duration parse fail).
|
|
- **`<meta property="video:release_date">`** — ISO 8601 z timezone, redundant z
|
|
JSON-LD uploadDate ale czystszy format.
|
|
- **`<meta property="video:tag">`** (multiple) — kanoniczna lista tagów (np.
|
|
"Big Ass", "Threesome"). Główne źródło tagów; alternatywnie DOM ma `btn gold`
|
|
linki ale te miksują performerów/studio z tagami.
|
|
- **Studio + Performers**: oba w sekcji `<a class="btn gold" href="/channels/<slug>/">`
|
|
(studio, singular) i `<a class="btn gold" href="/models/<slug>/">` (performerzy,
|
|
multiple). Slugi mają stable per-type salt (`*-i459s7` dla modeli, `*-7p72tp`
|
|
dla channels) — zachowują się jak hash z site-version, ale stabilne przez
|
|
sesje.
|
|
|
|
External_id strategia: `yespornvip:<numeric_video_id>` (`/video/69841/...` → `69841`).
|
|
Slug w URL ma `*-npu57w` suffix który wygląda na stałe-per-page-type, ale id
|
|
numeryczne jest bezpieczniejsze gdyby site zmienił salt.
|
|
|
|
URL patterns:
|
|
- Listing: `/latest-updates/` (page 1) / `/latest-updates/N/` (page>1)
|
|
- Scene: `/video/<id>/<slug>/` (id numeryczny, slug = title slug + 6-char salt)
|
|
- Studio: `/channels/<slug>/`
|
|
- Performer: `/models/<slug>/`
|
|
- Search: `/search/<query>/` (nie używane w browse-mode — można dorobić jako
|
|
osobny tryb dla performer-driven backfill jeśli będzie potrzeba)
|
|
|
|
Playback: download endpoint `/view_video_download.php?id=<id>&format=<480|720|1080>`
|
|
z `data-attach-session="PHPSESSID"` — wymaga session cookie, więc nie direct mp4
|
|
z server-side. Plus jest `embedUrl: /embed/<id>` w JSON-LD. Extractor →
|
|
`_vps_blocked_fallback.extract` (zgodne z pre-public bandwidth/anonymity policy):
|
|
mobile WebView fetcha embed z phone IP, INJECTED_JS scrape'uje `<video>.src`.
|
|
"""
|
|
from __future__ import annotations
|
|
|
|
import html as html_mod
|
|
import json
|
|
import logging
|
|
import re
|
|
from datetime import date, datetime
|
|
|
|
from app.connectors.base import (
|
|
RawFingerprint,
|
|
RawPerformer,
|
|
RawPlaybackSource,
|
|
RawScene,
|
|
RawStudio,
|
|
RawTag,
|
|
)
|
|
from app.connectors.direct_scrapers._browse_base import (
|
|
BaseBrowseScraper,
|
|
compute_thumbnail_phash,
|
|
meta_content,
|
|
)
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
_BASE = "https://yesporn.vip"
|
|
|
|
# Listing — scene URLs w listing HTML: `<a href="https://yesporn.vip/video/<id>/<slug>/">`.
|
|
# Slug `<slug>` zawiera stable per-type salt (`*-npu57w` dla videos).
|
|
_SCENE_URL_RE = re.compile(
|
|
r'href="(https://yesporn\.vip/video/(\d+)/[a-z0-9\-]+/)"',
|
|
re.IGNORECASE,
|
|
)
|
|
_VIDEO_ID_RE = re.compile(r"/video/(\d+)/", re.IGNORECASE)
|
|
|
|
# Studio (singular) i performerzy (multiple) w `<a class="btn gold" href="...">`.
|
|
# Studio: `/channels/<slug>/`. Performer: `/models/<slug>/`. Tekst linka =
|
|
# nazwa wyświetlana (może zawierać CSS-y/inne tagi, więc strip tagów po fakcie).
|
|
_STUDIO_LINK_RE = re.compile(
|
|
r'<a\s+class="btn\s+gold"\s+href="https://yesporn\.vip/channels/([a-z0-9\-]+)/"[^>]*>(.*?)</a>',
|
|
re.IGNORECASE | re.DOTALL,
|
|
)
|
|
_PERFORMER_LINK_RE = re.compile(
|
|
r'<a\s+class="btn\s+gold"\s+href="https://yesporn\.vip/models/([a-z0-9\-]+)/"[^>]*>(.*?)</a>',
|
|
re.IGNORECASE | re.DOTALL,
|
|
)
|
|
_HTML_TAG_RE = re.compile(r"<[^>]+>")
|
|
|
|
# JSON-LD VideoObject — pełny blok między `<script type="application/ld+json">` tagami.
|
|
_JSONLD_RE = re.compile(
|
|
r'<script[^>]+type=["\']application/ld\+json["\'][^>]*>(.*?)</script>',
|
|
re.IGNORECASE | re.DOTALL,
|
|
)
|
|
|
|
# `<meta property="video:tag" content="Big Ass">` — multiple, jeden tag per meta.
|
|
_META_TAG_RE = re.compile(
|
|
r'<meta\s+property=["\']video:tag["\']\s+content=["\']([^"\']+)["\']',
|
|
re.IGNORECASE,
|
|
)
|
|
|
|
# ISO 8601 duration `PT0H39M00S` / `PT39M0S` / `PT45S`.
|
|
_ISO_DUR_RE = re.compile(
|
|
r"^P?T?(?:(\d+)H)?(?:(\d+)M)?(?:(\d+)S)?$", re.IGNORECASE
|
|
)
|
|
|
|
|
|
def _parse_iso_duration(value: str | None) -> int | None:
|
|
if not value:
|
|
return None
|
|
m = _ISO_DUR_RE.match(value.strip())
|
|
if not m:
|
|
return None
|
|
h = int(m.group(1) or 0)
|
|
mn = int(m.group(2) or 0)
|
|
s = int(m.group(3) or 0)
|
|
total = h * 3600 + mn * 60 + s
|
|
return total or None
|
|
|
|
|
|
def _parse_iso_date(value: str | None) -> date | None:
|
|
"""`2026-05-26T19:23:29Z` / `2026-05-26T19:23:29.EDT` → date."""
|
|
if not value:
|
|
return None
|
|
# yesporn emituje `.EDT` jako "timezone" w JSON-LD uploadDate — strip żeby
|
|
# `fromisoformat` nie crash'ował. video:release_date meta ma czysty `Z`.
|
|
cleaned = re.sub(r"\.[A-Z]{2,4}$", "", value.strip())
|
|
try:
|
|
return datetime.fromisoformat(cleaned.replace("Z", "+00:00")).date()
|
|
except ValueError:
|
|
m = re.match(r"(\d{4}-\d{2}-\d{2})", cleaned)
|
|
if m:
|
|
try:
|
|
return date.fromisoformat(m.group(1))
|
|
except ValueError:
|
|
return None
|
|
return None
|
|
|
|
|
|
def _iter_jsonld_objects(data: object):
|
|
"""Spłaszcza JSON-LD: dict / list / @graph → strumień dict-ów."""
|
|
if isinstance(data, dict):
|
|
graph = data.get("@graph")
|
|
if isinstance(graph, list):
|
|
for item in graph:
|
|
yield from _iter_jsonld_objects(item)
|
|
else:
|
|
yield data
|
|
elif isinstance(data, list):
|
|
for item in data:
|
|
yield from _iter_jsonld_objects(item)
|
|
|
|
|
|
def _extract_video_object(html: str) -> dict | None:
|
|
for m in _JSONLD_RE.finditer(html):
|
|
raw = m.group(1).strip()
|
|
if not raw:
|
|
continue
|
|
try:
|
|
data = json.loads(raw)
|
|
except (json.JSONDecodeError, ValueError):
|
|
continue
|
|
for obj in _iter_jsonld_objects(data):
|
|
if obj.get("@type") == "VideoObject":
|
|
return obj
|
|
return None
|
|
|
|
|
|
def _clean_link_text(raw: str) -> str:
|
|
"""Strip HTML tagów + decode entities + whitespace normalize."""
|
|
text = _HTML_TAG_RE.sub("", raw)
|
|
text = html_mod.unescape(text)
|
|
return " ".join(text.split()).strip()
|
|
|
|
|
|
class YesPornVipScraper(BaseBrowseScraper):
|
|
sitetag = "yespornvip"
|
|
|
|
def _listing_url(self, page: int) -> str:
|
|
if page <= 1:
|
|
return f"{_BASE}/latest-updates/"
|
|
return f"{_BASE}/latest-updates/{page}/"
|
|
|
|
def _extract_scene_urls(self, listing_html: str) -> list[str]:
|
|
seen: set[str] = set()
|
|
out: list[str] = []
|
|
for m in _SCENE_URL_RE.finditer(listing_html):
|
|
url = m.group(1)
|
|
if url in seen:
|
|
continue
|
|
seen.add(url)
|
|
out.append(url)
|
|
return out
|
|
|
|
def _parse_detail(self, scene_url: str, detail_html: str) -> RawScene | None:
|
|
video = _extract_video_object(detail_html)
|
|
if not video:
|
|
log.info("yesporn: no JSON-LD VideoObject on %s", scene_url)
|
|
return None
|
|
|
|
title = (video.get("name") or "").strip()
|
|
if not title:
|
|
return None
|
|
|
|
video_id_m = _VIDEO_ID_RE.search(scene_url)
|
|
video_id = video_id_m.group(1) if video_id_m else None
|
|
|
|
description = (video.get("description") or "").strip() or None
|
|
|
|
# Duration: preferuj `<meta property="video:duration">` (czyste sekundy),
|
|
# fallback do JSON-LD ISO format.
|
|
duration_sec: int | None = None
|
|
meta_dur = meta_content(detail_html, property="video:duration")
|
|
if meta_dur and meta_dur.isdigit():
|
|
duration_sec = int(meta_dur) or None
|
|
if duration_sec is None:
|
|
duration_sec = _parse_iso_duration(video.get("duration"))
|
|
|
|
# Release date: preferuj `<meta property="video:release_date">` (czystszy
|
|
# format z timezone), fallback do JSON-LD uploadDate.
|
|
release_date = _parse_iso_date(
|
|
meta_content(detail_html, property="video:release_date")
|
|
or video.get("uploadDate")
|
|
)
|
|
|
|
thumbnail_url = video.get("thumbnailUrl") or None
|
|
|
|
# Studio: pierwszy `btn gold` link do `/channels/<slug>/`. Strona renderuje
|
|
# tylko jednego per scenę (logo studia obok performerów).
|
|
studio: RawStudio | None = None
|
|
for m in _STUDIO_LINK_RE.finditer(detail_html):
|
|
slug = m.group(1).strip()
|
|
name = _clean_link_text(m.group(2))
|
|
if not name:
|
|
continue
|
|
studio = RawStudio(
|
|
external_id=f"{self.sitetag}:channel:{slug}",
|
|
name=name,
|
|
slug=slug,
|
|
)
|
|
break
|
|
|
|
# Performers: wszystkie `btn gold` linki do `/models/<slug>/` (multiple).
|
|
performers: list[RawPerformer] = []
|
|
seen_perf: set[str] = set()
|
|
for m in _PERFORMER_LINK_RE.finditer(detail_html):
|
|
slug = m.group(1).strip()
|
|
if slug in seen_perf:
|
|
continue
|
|
name = _clean_link_text(m.group(2))
|
|
if not name:
|
|
continue
|
|
seen_perf.add(slug)
|
|
performers.append(
|
|
RawPerformer(
|
|
external_id=f"{self.sitetag}:performer:{slug}",
|
|
name=name,
|
|
)
|
|
)
|
|
|
|
# Tagi: `<meta property="video:tag" content="...">` (multiple).
|
|
# Deny-list: pomiń wszystkie all-lowercase tagi. yesporn.vip SEO-stuffuje
|
|
# `meta video:tag` tokenami z tytułu i imionami performerów + gibberish
|
|
# ("bella", "rose", "reverse", "deep", "throat", "ddca"), wszystkie always
|
|
# lowercase. Legit kategorie są zawsze Title Case ("Big Ass", "Deep
|
|
# Throat", "Blonde", "Gangbang") lub UPPER ("MILF", "BBW"). Potwierdzone
|
|
# w 20-scene dry-run 2026-05-27. Trade-off: stracimy hipotetyczne legit
|
|
# lowercase tagi (np. "interracial" gdyby site je nie capitalize'ował) —
|
|
# akceptowalne bo tags mają wagę tylko 0.05 w composite scoring resolvera.
|
|
tags: list[RawTag] = []
|
|
seen_tag: set[str] = set()
|
|
for m in _META_TAG_RE.finditer(detail_html):
|
|
name = html_mod.unescape(m.group(1)).strip()
|
|
if not name:
|
|
continue
|
|
if name == name.lower():
|
|
continue
|
|
slug = re.sub(r"[^a-z0-9]+", "-", name.lower()).strip("-")
|
|
if slug in seen_tag:
|
|
continue
|
|
seen_tag.add(slug)
|
|
tags.append(
|
|
RawTag(external_id=f"{self.sitetag}:tag:{slug}", name=name, slug=slug)
|
|
)
|
|
|
|
# Phash z thumbnailUrl — BunnyCDN `yesnn.b-cdn.net` hostuje 1.jpg per scene.
|
|
# Hit-rate vs canonical TPDB/StashDB nieznany do pilot run; graceful: brak
|
|
# phash → resolver spada do composite scoring (studio + performer + date +
|
|
# duration + title token-set) — wszystkie dostępne dzięki JSON-LD.
|
|
fingerprints: list[RawFingerprint] = []
|
|
if thumbnail_url:
|
|
ph = compute_thumbnail_phash(thumbnail_url, referer=_BASE + "/")
|
|
if ph:
|
|
fingerprints.append(RawFingerprint(kind="phash", value=ph))
|
|
|
|
# Playback — page_url do strony sceny. Direct mp4 (`view_video_download.php`)
|
|
# wymaga PHPSESSID cookie (data-attach-session attribute), więc nie usable
|
|
# server-side. Extractor `yespornvip` → `_vps_blocked_fallback.extract`:
|
|
# mobile WebView z phone IP łapie session natively, INJECTED_JS scrape.
|
|
playback_sources = [
|
|
RawPlaybackSource(
|
|
origin=f"tube:{self.sitetag}",
|
|
page_url=scene_url,
|
|
duration_sec=duration_sec,
|
|
thumbnail_url=thumbnail_url,
|
|
)
|
|
]
|
|
|
|
return RawScene(
|
|
external_id=f"{self.sitetag}:{video_id or scene_url}",
|
|
title=title,
|
|
description=description,
|
|
release_date=release_date,
|
|
duration_sec=duration_sec,
|
|
url=scene_url,
|
|
studio=studio,
|
|
performers=performers,
|
|
tags=tags,
|
|
fingerprints=fingerprints,
|
|
playback_sources=playback_sources,
|
|
)
|