2019-08-23 07:03:30 +00:00
|
|
|
"""
|
|
|
|
Twitch API access.
|
|
|
|
"""
|
|
|
|
|
2018-01-25 10:09:20 +00:00
|
|
|
import requests
|
|
|
|
|
|
|
|
from twitchdl import CLIENT_ID
|
2019-08-12 13:14:13 +00:00
|
|
|
from twitchdl.exceptions import ConsoleError
|
2018-01-25 10:09:20 +00:00
|
|
|
|
|
|
|
|
2019-08-23 07:03:30 +00:00
|
|
|
def authenticated_get(url, params={}, headers={}):
|
|
|
|
headers['Client-ID'] = CLIENT_ID
|
2018-01-25 10:09:20 +00:00
|
|
|
|
|
|
|
response = requests.get(url, params, headers=headers)
|
2019-08-12 13:14:13 +00:00
|
|
|
if response.status_code == 400:
|
|
|
|
data = response.json()
|
|
|
|
raise ConsoleError(data["message"])
|
|
|
|
|
2018-01-25 10:09:20 +00:00
|
|
|
response.raise_for_status()
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
2020-04-11 14:07:17 +00:00
|
|
|
def authenticated_post(url, data=None, json=None, headers={}):
|
|
|
|
headers['Client-ID'] = CLIENT_ID
|
|
|
|
|
|
|
|
response = requests.post(url, data=data, json=json, headers=headers)
|
|
|
|
if response.status_code == 400:
|
|
|
|
data = response.json()
|
|
|
|
raise ConsoleError(data["message"])
|
|
|
|
|
|
|
|
response.raise_for_status()
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
2019-08-23 07:03:30 +00:00
|
|
|
def kraken_get(url, params={}, headers={}):
|
|
|
|
"""
|
|
|
|
Add accept header required by kraken API v5.
|
|
|
|
see: https://discuss.dev.twitch.tv/t/change-in-access-to-deprecated-kraken-twitch-apis/22241
|
|
|
|
"""
|
|
|
|
headers["Accept"] = "application/vnd.twitchtv.v5+json"
|
|
|
|
return authenticated_get(url, params, headers)
|
|
|
|
|
|
|
|
|
|
|
|
def get_user(login):
|
|
|
|
"""
|
|
|
|
https://dev.twitch.tv/docs/api/reference/#get-users
|
|
|
|
"""
|
|
|
|
response = authenticated_get("https://api.twitch.tv/helix/users", {
|
|
|
|
"login": login
|
|
|
|
})
|
|
|
|
|
|
|
|
users = response.json()["data"]
|
|
|
|
return users[0] if users else None
|
|
|
|
|
|
|
|
|
2018-01-25 10:09:20 +00:00
|
|
|
def get_video(video_id):
|
|
|
|
"""
|
|
|
|
https://dev.twitch.tv/docs/v5/reference/videos#get-video
|
|
|
|
"""
|
2020-04-11 14:07:17 +00:00
|
|
|
url = "https://api.twitch.tv/kraken/videos/{}".format(video_id)
|
2018-01-25 10:09:20 +00:00
|
|
|
|
2019-08-23 07:03:30 +00:00
|
|
|
return kraken_get(url).json()
|
2018-01-25 10:09:20 +00:00
|
|
|
|
|
|
|
|
2020-04-11 14:07:17 +00:00
|
|
|
def get_clip(slug):
|
|
|
|
url = "https://gql.twitch.tv/gql"
|
|
|
|
|
|
|
|
query = """
|
|
|
|
{{
|
|
|
|
clip(slug: "{}") {{
|
|
|
|
title
|
|
|
|
durationSeconds
|
|
|
|
game {{
|
|
|
|
name
|
|
|
|
}}
|
|
|
|
broadcaster {{
|
|
|
|
login
|
|
|
|
displayName
|
|
|
|
}}
|
|
|
|
videoQualities {{
|
|
|
|
frameRate
|
|
|
|
quality
|
|
|
|
sourceURL
|
|
|
|
}}
|
|
|
|
}}
|
|
|
|
}}
|
|
|
|
"""
|
|
|
|
|
|
|
|
payload = {"query": query.format(slug)}
|
|
|
|
data = authenticated_post(url, json=payload).json()
|
|
|
|
return data["data"]["clip"]
|
|
|
|
|
|
|
|
|
2019-08-23 07:03:30 +00:00
|
|
|
def get_channel_videos(channel_id, limit, offset, sort):
|
2018-01-25 10:09:20 +00:00
|
|
|
"""
|
|
|
|
https://dev.twitch.tv/docs/v5/reference/channels#get-channel-videos
|
|
|
|
"""
|
2019-08-23 07:03:30 +00:00
|
|
|
url = "https://api.twitch.tv/kraken/channels/{}/videos".format(channel_id)
|
2018-01-25 10:09:20 +00:00
|
|
|
|
2019-08-23 07:03:30 +00:00
|
|
|
return kraken_get(url, {
|
2018-01-25 10:09:20 +00:00
|
|
|
"broadcast_type": "archive",
|
|
|
|
"limit": limit,
|
2019-08-13 10:25:25 +00:00
|
|
|
"offset": offset,
|
|
|
|
"sort": sort,
|
2018-01-25 10:09:20 +00:00
|
|
|
}).json()
|
|
|
|
|
|
|
|
|
|
|
|
def get_access_token(video_id):
|
2020-04-11 14:07:17 +00:00
|
|
|
url = "https://api.twitch.tv/api/vods/{}/access_token".format(video_id)
|
2018-01-25 10:09:20 +00:00
|
|
|
|
|
|
|
return authenticated_get(url).json()
|
|
|
|
|
|
|
|
|
|
|
|
def get_playlists(video_id, access_token):
|
2019-08-23 10:36:05 +00:00
|
|
|
"""
|
|
|
|
For a given video return a playlist which contains possible video qualities.
|
|
|
|
"""
|
2018-01-25 10:09:20 +00:00
|
|
|
url = "http://usher.twitch.tv/vod/{}".format(video_id)
|
|
|
|
|
|
|
|
response = requests.get(url, params={
|
|
|
|
"nauth": access_token['token'],
|
|
|
|
"nauthsig": access_token['sig'],
|
|
|
|
"allow_source": "true",
|
|
|
|
"player": "twitchweb",
|
|
|
|
})
|
|
|
|
response.raise_for_status()
|
2019-08-23 10:36:05 +00:00
|
|
|
return response.content.decode('utf-8')
|