nitter/src/routes/timeline.nim

102 lines
3.4 KiB
Nim
Raw Normal View History

2019-09-06 00:42:35 +00:00
import asyncdispatch, strutils, sequtils, uri
import jester
import router_utils
2019-09-20 23:08:30 +00:00
import ".."/[api, types, cache, formatters, agents, query]
2019-09-13 20:24:58 +00:00
import ../views/[general, profile, timeline, status, search]
2019-09-06 00:42:35 +00:00
export uri, sequtils
export router_utils
2019-09-13 20:24:58 +00:00
export api, cache, formatters, query, agents
2019-09-06 00:42:35 +00:00
export profile, timeline, status
type ProfileTimeline = (Profile, Timeline, seq[GalleryPhoto])
proc fetchSingleTimeline*(name, after, agent: string; query: Query;
media=true): Future[ProfileTimeline] {.async.} =
2019-09-06 00:42:35 +00:00
let railFut = getPhotoRail(name, agent)
var timeline: Timeline
var profile: Profile
var cachedProfile = hasCachedProfile(name)
if cachedProfile.isSome:
profile = get(cachedProfile)
2019-09-19 00:23:22 +00:00
if query.kind == posts:
2019-09-06 00:42:35 +00:00
if cachedProfile.isSome:
timeline = await getTimeline(name, after, agent, media)
2019-09-06 00:42:35 +00:00
else:
(profile, timeline) = await getProfileAndTimeline(name, agent, after, media)
2019-09-06 00:42:35 +00:00
cache(profile)
else:
var timelineFut = getSearch[Tweet](query, after, agent, media)
2019-09-06 00:42:35 +00:00
if cachedProfile.isNone:
profile = await getCachedProfile(name, agent)
timeline = await timelineFut
if profile.username.len == 0: return
return (profile, timeline, await railFut)
2019-09-06 00:42:35 +00:00
proc fetchMultiTimeline*(names: seq[string]; after, agent: string;
2019-09-19 00:23:22 +00:00
query: Query): Future[Timeline] {.async.} =
2019-09-06 00:42:35 +00:00
var q = query
2019-09-19 00:23:22 +00:00
q.fromUser = names
if q.kind == posts and "replies" notin q.excludes:
q.excludes.add "replies"
return await getSearch[Tweet](q, after, agent)
2019-09-06 00:42:35 +00:00
2019-09-20 20:56:27 +00:00
proc get*(req: Request; key: string): string =
if key in params(req): params(req)[key]
else: ""
proc showTimeline*(request: Request; query: Query; cfg: Config; rss: string): Future[string] {.async.} =
2019-09-20 20:56:27 +00:00
let
agent = getAgent()
prefs = cookiePrefs()
name = request.get("name")
after = request.get("max_position")
2019-09-20 20:56:27 +00:00
names = name.strip(chars={'/'}).split(",").filterIt(it.len > 0)
2019-09-06 00:42:35 +00:00
if names.len == 1:
let (p, t, r) = await fetchSingleTimeline(names[0], after, agent, query)
if p.username.len == 0: return
2019-09-20 20:56:27 +00:00
let pHtml = renderProfile(p, t, r, prefs, getPath())
return renderMain(pHtml, request, cfg, pageTitle(p), pageDesc(p), rss=rss)
2019-09-06 00:42:35 +00:00
else:
let
timeline = await fetchMultiTimeline(names, after, agent, query)
2019-09-20 20:56:27 +00:00
html = renderTweetSearch(timeline, prefs, getPath())
return renderMain(html, request, cfg, "Multi")
2019-09-06 00:42:35 +00:00
template respTimeline*(timeline: typed) =
if timeline.len == 0:
2019-10-21 05:59:22 +00:00
resp Http404, showError("User \"" & @"name" & "\" not found", cfg)
2019-09-06 00:42:35 +00:00
resp timeline
proc createTimelineRouter*(cfg: Config) =
setProfileCacheTime(cfg.profileCacheTime)
router timeline:
get "/@name/?":
cond '.' notin @"name"
let rss = "/$1/rss" % @"name"
respTimeline(await showTimeline(request, Query(), cfg, rss))
2019-09-06 00:42:35 +00:00
get "/@name/with_replies":
2019-09-06 00:42:35 +00:00
cond '.' notin @"name"
let rss = "/$1/with_replies/rss" % @"name"
respTimeline(await showTimeline(request, getReplyQuery(@"name"), cfg, rss))
2019-09-06 00:42:35 +00:00
get "/@name/media":
cond '.' notin @"name"
let rss = "/$1/media/rss" % @"name"
respTimeline(await showTimeline(request, getMediaQuery(@"name"), cfg, rss))
2019-09-15 07:57:45 +00:00
2019-09-20 01:35:27 +00:00
get "/@name/search":
cond '.' notin @"name"
let query = initQuery(params(request), name=(@"name"))
2019-09-20 20:56:27 +00:00
let rss = "/$1/search/rss?$2" % [@"name", genQueryUrl(query)]
respTimeline(await showTimeline(request, query, cfg, rss))