nitter/src/api.nim

99 lines
2.9 KiB
Nim
Raw Normal View History

2019-06-20 14:16:20 +00:00
import httpclient, asyncdispatch, htmlparser, times
import sequtils, strutils, strformat, json, xmltree, uri
import nimquery, regex
import ./types, ./parser
2019-06-21 01:51:14 +00:00
const
base = parseUri("https://twitter.com/")
agent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36"
timelineUrl = "i/profiles/show/$1/timeline/tweets?include_available_features=1&include_entities=1&include_new_items_bar=true"
profilePopupUrl = "i/profiles/popup"
profileIntentUrl = "intent/user"
tweetUrl = "i/status/"
proc fetchHtml(url: Uri; headers: HttpHeaders; jsonKey = ""): Future[XmlNode] {.async.} =
var client = newAsyncHttpClient()
defer: client.close()
2019-06-20 14:16:20 +00:00
2019-06-21 01:51:14 +00:00
client.headers = headers
2019-06-20 14:16:20 +00:00
2019-06-21 00:15:46 +00:00
var resp = ""
try:
resp = await client.getContent($url)
except:
return nil
if jsonKey.len > 0:
let json = parseJson(resp)[jsonKey].str
return parseHtml(json)
else:
return parseHtml(resp)
2019-06-21 01:51:14 +00:00
proc getProfileFallback(username: string; headers: HttpHeaders): Future[Profile] {.async.} =
2019-06-21 00:15:46 +00:00
let
2019-06-21 01:51:14 +00:00
url = base / profileIntentUrl ? {"screen_name": username}
html = await fetchHtml(url, headers)
2019-06-21 00:15:46 +00:00
result = parseIntentProfile(html)
2019-06-20 14:16:20 +00:00
proc getProfile*(username: string): Future[Profile] {.async.} =
2019-06-21 01:51:14 +00:00
let headers = newHttpHeaders({
2019-06-20 14:16:20 +00:00
"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9",
"Referer": $(base / username),
"User-Agent": agent,
"X-Twitter-Active-User": "yes",
"X-Requested-With": "XMLHttpRequest",
"Accept-Language": "en-US,en;q=0.9"
})
let
2019-06-21 00:15:46 +00:00
params = {
"screen_name": username,
"wants_hovercard": "true",
"_": $(epochTime().int)
}
url = base / profilePopupUrl ? params
2019-06-21 01:51:14 +00:00
html = await fetchHtml(url, headers, jsonKey="html")
2019-06-20 14:16:20 +00:00
2019-06-21 00:15:46 +00:00
if not html.querySelector(".ProfileCard-sensitiveWarningContainer").isNil:
2019-06-21 01:51:14 +00:00
return await getProfileFallback(username, headers)
2019-06-21 00:15:46 +00:00
result = parsePopupProfile(html)
2019-06-20 14:16:20 +00:00
proc getTimeline*(username: string; after=""): Future[Tweets] {.async.} =
2019-06-21 01:51:14 +00:00
let headers = newHttpHeaders({
2019-06-20 14:16:20 +00:00
"Accept": "application/json, text/javascript, */*; q=0.01",
"Referer": $(base / username),
"User-Agent": agent,
"X-Twitter-Active-User": "yes",
"X-Requested-With": "XMLHttpRequest",
"Accept-Language": "en-US,en;q=0.9"
})
var url = timelineUrl % username
if after.len > 0:
2019-06-20 14:16:20 +00:00
url &= "&max_position=" & after
2019-06-21 01:51:14 +00:00
let html = await fetchHtml(base / url, headers, jsonKey="items_html")
2019-06-20 14:16:20 +00:00
result = parseTweets(html)
proc getTweet*(id: string): Future[Conversation] {.async.} =
2019-06-21 01:51:14 +00:00
let headers = newHttpHeaders({
2019-06-20 14:16:20 +00:00
"Accept": "application/json, text/javascript, */*; q=0.01",
"Referer": $base,
"User-Agent": agent,
"X-Twitter-Active-User": "yes",
"X-Requested-With": "XMLHttpRequest",
"Accept-Language": "en-US,en;q=0.9",
"pragma": "no-cache",
"x-previous-page-name": "profile"
})
2019-06-21 00:15:46 +00:00
let
url = base / tweetUrl / id
2019-06-21 01:51:14 +00:00
html = await fetchHtml(url, headers)
2019-06-20 14:16:20 +00:00
result = parseConversation(html)