1
0
mirror of https://github.com/spaam/svtplay-dl.git synced 2024-11-28 06:04:17 +01:00
svtplay-dl/lib/svtplay_dl/service/dplay.py

233 lines
9.7 KiB
Python
Raw Normal View History

2015-10-04 14:40:00 +02:00
# ex:ts=4:sw=4:sts=4:et
# -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*-
import datetime
import hashlib
2018-05-27 16:05:54 +02:00
import logging
2019-08-25 00:40:39 +02:00
import random
import re
2018-01-30 22:07:21 +01:00
from urllib.parse import urlparse
2015-10-04 14:40:00 +02:00
from svtplay_dl.error import ServiceError
2019-08-25 00:40:39 +02:00
from svtplay_dl.fetcher.hls import hlsparse
from svtplay_dl.service import Service
2018-05-27 16:05:54 +02:00
from svtplay_dl.subtitle import subtitle
2015-10-04 14:40:00 +02:00
2018-01-30 20:11:37 +01:00
country = {"sv": ".se", "da": ".dk", "no": ".no"}
2021-01-18 21:44:05 +01:00
REALMS = {"discoveryplus.se": "dplayse", "discoveryplus.no": "dplayno", "discoveryplus.dk": "dplaydk"}
2018-05-27 16:10:22 +02:00
2015-10-04 14:40:00 +02:00
class Dplay(Service):
2021-01-18 21:44:05 +01:00
supported_domains = ["discoveryplus.se", "discoveryplus.no", "discoveryplus.dk"]
packages = []
2015-10-04 14:40:00 +02:00
def get(self):
parse = urlparse(self.url)
2021-01-18 21:44:05 +01:00
self.domain = re.search(r"(discoveryplus\.\w\w)", parse.netloc).group(1)
2015-10-04 14:40:00 +02:00
if not self._token():
2018-05-27 16:05:54 +02:00
logging.error("Something went wrong getting token for requests")
2015-10-29 22:42:27 +01:00
if not self._login():
yield ServiceError("You need the 'st' cookie from your web brower for the site to make it work")
return
channel = False
if "kanaler" in parse.path:
match = re.search("kanaler/([^/]+)$", parse.path)
2020-09-29 20:39:27 +02:00
if not match:
yield ServiceError("Can't detect 'kanaler'")
return
2021-04-27 19:44:09 +02:00
path = f"/channels/{match.group(1)}"
2021-02-28 22:05:15 +01:00
url = f"https://disco-api.{self.domain}/content{path}"
channel = True
2018-05-13 13:06:45 +02:00
self.config.set("live", True)
elif "program" in parse.path:
match = re.search("(programmer|program)/([^/]+)$", parse.path)
2020-09-29 20:39:27 +02:00
if not match:
yield ServiceError("Can't find program url")
return
2021-04-27 19:44:09 +02:00
path = f"/shows/{match.group(2)}"
2021-02-28 22:05:15 +01:00
url = f"https://disco-api.{self.domain}/content{path}"
res = self.http.get(url, headers={"x-disco-client": "WEB:UNKNOWN:dplay-client:0.0.1"})
programid = res.json()["data"]["id"]
2019-08-25 00:27:31 +02:00
qyerystring = (
"include=primaryChannel,show&filter[videoType]=EPISODE&filter[show.id]={}&"
"page[size]=100&sort=seasonNumber,episodeNumber,-earliestPlayableStart".format(programid)
2019-08-25 00:27:31 +02:00
)
2021-02-28 22:05:15 +01:00
res = self.http.get(f"https://disco-api.{self.domain}/content/videos?{qyerystring}")
janson = res.json()
vid = 0
slug = None
for i in janson["data"]:
if int(i["id"]) > vid:
vid = int(i["id"])
slug = i["attributes"]["path"]
if slug:
2021-02-28 22:05:15 +01:00
url = f"https://disco-api.{self.domain}/content/videos/{slug}"
else:
yield ServiceError("Cant find latest video on program url")
return
else:
match = re.search("(videos|videoer)/(.*)$", parse.path)
2021-04-27 19:44:09 +02:00
url = f"https://disco-api.{self.domain}/content/videos/{match.group(2)}"
res = self.http.get(url, headers={"x-disco-client": "WEB:UNKNOWN:dplay-client:0.0.1"})
janson = res.json()
if "errors" in janson:
yield ServiceError("Cant find any videos on this url")
return
2018-05-13 13:06:45 +02:00
if channel:
name = janson["data"]["attributes"]["name"]
self.output["title"] = name
else:
name = self._autoname(janson)
if name is None:
yield ServiceError("Cant find vid id for autonaming")
2018-02-24 20:59:23 +01:00
return
2018-05-13 13:06:45 +02:00
self.output["id"] = janson["data"]["id"]
2018-02-24 20:59:23 +01:00
api = "https://disco-api.{}/playback/videoPlaybackInfo/{}?usePreAuth=true".format(self.domain, janson["data"]["id"])
res = self.http.get(api)
if res.status_code > 400:
yield ServiceError("You dont have permission to watch this")
return
2019-08-25 00:27:31 +02:00
streams = hlsparse(
self.config,
self.http.request("get", res.json()["data"]["attributes"]["streaming"]["hls"]["url"]),
2019-08-25 00:27:31 +02:00
res.json()["data"]["attributes"]["streaming"]["hls"]["url"],
httpobject=self.http,
output=self.output,
)
2018-05-08 22:48:55 +02:00
for n in list(streams.keys()):
if isinstance(streams[n], subtitle): # we get the subtitles from the hls playlist.
if self.config.get("get_all_subtitles"):
yield streams[n]
else:
if streams[n].subfix in country and country[streams[n].subfix] in self.domain:
yield streams[n]
else:
yield streams[n]
2015-10-29 19:32:38 +01:00
def _autoname(self, jsondata):
2019-08-25 00:27:31 +02:00
match = re.search("^([^/]+)/", jsondata["data"]["attributes"]["path"])
2018-05-13 13:06:45 +02:00
self.output["title"] = match.group(1)
self.output["season"] = int(jsondata["data"]["attributes"]["seasonNumber"])
self.output["episode"] = int(jsondata["data"]["attributes"]["episodeNumber"])
self.output["episodename"] = jsondata["data"]["attributes"]["name"]
return self.output["title"]
2018-05-08 22:46:11 +02:00
def find_all_episodes(self, config):
parse = urlparse(self.url)
2021-01-18 21:44:05 +01:00
self.domain = re.search(r"(discoveryplus\.\w\w)", parse.netloc).group(1)
programid = None
seasons = []
episodes = []
match = re.search("^/(program|programmer|videos|videoer)/([^/]+)", parse.path)
if not match:
2018-05-27 16:05:54 +02:00
logging.error("Can't find show name")
return None
if not self._login():
logging.error("Need the 'st' cookie to work")
return None
if not self._token():
2018-05-27 16:05:54 +02:00
logging.error("Something went wrong getting token for requests")
self._getpackages()
2020-11-19 22:21:52 +01:00
urllocal = ""
if self.domain in ["discoveryplus.no", "discoveryplus.dk"]:
2020-11-19 22:21:52 +01:00
urllocal = "mer"
2021-04-27 19:44:09 +02:00
url = f"http://disco-api.{self.domain}/cms/routes/program{urllocal}/{match.group(2)}?decorators=viewingHistory&include=default"
res = self.http.get(url)
if res.status_code > 400:
logging.error("Cant find any videos. wrong url?")
return episodes
showid = None
for what in res.json()["included"]:
if "attributes" in what and "component" in what["attributes"] and what["attributes"]["component"]["id"] == "tabbed-content":
programid = what["id"]
for ses in what["attributes"]["component"]["filters"]:
if ses["id"] == "seasonNumber":
for opt in ses["options"]:
if "value" not in opt:
continue
seasons.append(opt["value"])
if "mandatoryParams" in what["attributes"]["component"]:
showid = what["attributes"]["component"]["mandatoryParams"]
if programid:
for season in seasons:
page = 1
totalpages = 1
while page <= totalpages:
querystring = "decorators=viewingHistory&include=default&page[items.number]={}&pf[seasonNumber]={}".format(
page,
season,
)
if showid:
2021-02-28 22:05:15 +01:00
querystring += f"&{showid}"
res = self.http.get(f"https://disco-api.{self.domain}/cms/collections/{programid}?{querystring}")
janson = res.json()
if "meta" not in janson["data"]:
page += 1
continue
totalpages = janson["data"]["meta"]["itemsTotalPages"]
for i in janson["included"]:
if i["type"] != "video":
continue
if i["attributes"]["videoType"] == "EPISODE":
if not self._playablefile(i["attributes"]["availabilityWindows"]):
continue
episodes.append("https://www.{}/videos/{}".format(self.domain, i["attributes"]["path"]))
page += 1
if not episodes:
2018-05-27 16:05:54 +02:00
logging.error("Cant find any playable files")
2018-05-08 22:46:11 +02:00
if config.get("all_last") > 0:
2019-08-25 00:27:31 +02:00
return episodes[: config.get("all_last")]
return episodes
def _login(self):
2021-02-28 22:05:15 +01:00
res = self.http.get(f"https://disco-api.{self.domain}/users/me", headers={"authority": f"disco-api.{self.domain}"})
if res.status_code >= 400:
return False
if not res.json()["data"]["attributes"]["anonymous"]:
return True
return False
def _token(self) -> bool:
# random device id for cookietoken
2018-02-26 00:05:21 +01:00
deviceid = hashlib.sha256(bytes(int(random.random() * 1000))).hexdigest()
2021-04-27 19:44:09 +02:00
url = f"https://disco-api.{self.domain}/token?realm={REALMS[self.domain]}&deviceId={deviceid}&shortlived=true"
res = self.http.get(url)
if res.status_code >= 400:
return False
return True
def _getpackages(self):
2021-02-28 22:05:15 +01:00
res = self.http.get(f"https://disco-api.{self.domain}/users/me", headers={"authority": f"disco-api.{self.domain}"})
if res.status_code < 400:
self.packages.extend(res.json()["data"]["attributes"]["packages"])
def _playablefile(self, needs):
playable = False
now = datetime.datetime.utcnow()
for package in self.packages:
for need in needs:
if package != need["package"]:
continue
start = datetime.datetime.strptime(need["playableStart"], "%Y-%m-%dT%H:%M:%SZ").replace(tzinfo=None)
if now > start:
if "playableEnd" in need:
end = datetime.datetime.strptime(need["playableEnd"], "%Y-%m-%dT%H:%M:%SZ").replace(tzinfo=None)
if now < end:
playable = True
else:
playable = True
return playable