Trending page (#128)
Fixes #46 Reviewed-on: https://codeberg.org/rimgo/rimgo/pulls/128 Co-authored-by: orangix <orangix@noreply.codeberg.org> Co-committed-by: orangix <orangix@noreply.codeberg.org>
This commit is contained in:
parent
d69d8dba0e
commit
a8abb43f3a
|
@ -0,0 +1,107 @@
|
||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"codeberg.org/rimgo/rimgo/utils"
|
||||||
|
"github.com/patrickmn/go-cache"
|
||||||
|
"github.com/tidwall/gjson"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (client *Client) FetchTrending(section, sort, page string) ([]Submission, error) {
|
||||||
|
cacheData, found := client.Cache.Get(fmt.Sprintf("trending-%s-%s-%s", section, sort, page))
|
||||||
|
if found {
|
||||||
|
return cacheData.([]Submission), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("GET", "https://api.imgur.com/post/v1/posts", nil)
|
||||||
|
if err != nil {
|
||||||
|
return []Submission{}, err
|
||||||
|
}
|
||||||
|
utils.SetReqHeaders(req)
|
||||||
|
|
||||||
|
q := req.URL.Query()
|
||||||
|
q.Add("client_id", client.ClientID)
|
||||||
|
q.Add("include", "cover")
|
||||||
|
q.Add("page", page)
|
||||||
|
|
||||||
|
switch sort {
|
||||||
|
case "newest":
|
||||||
|
q.Add("filter[window]", "week")
|
||||||
|
q.Add("sort", "-time")
|
||||||
|
case "best":
|
||||||
|
q.Add("filter[window]", "all")
|
||||||
|
q.Add("sort", "-top")
|
||||||
|
case "popular":
|
||||||
|
fallthrough
|
||||||
|
default:
|
||||||
|
q.Add("filter[window]", "week")
|
||||||
|
q.Add("sort", "-viral")
|
||||||
|
sort = "popular"
|
||||||
|
}
|
||||||
|
switch section {
|
||||||
|
case "hot":
|
||||||
|
q.Add("filter[section]", "eq:hot")
|
||||||
|
case "new":
|
||||||
|
q.Add("filter[section]", "eq:new")
|
||||||
|
case "top":
|
||||||
|
q.Add("filter[section]", "eq:top")
|
||||||
|
q.Add("filter[window]", "day")
|
||||||
|
default:
|
||||||
|
q.Add("filter[section]", "eq:hot")
|
||||||
|
section = "hot"
|
||||||
|
}
|
||||||
|
|
||||||
|
req.URL.RawQuery = q.Encode()
|
||||||
|
|
||||||
|
res, err := http.DefaultClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return []Submission{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := io.ReadAll(res.Body)
|
||||||
|
if err != nil {
|
||||||
|
return []Submission{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
data := gjson.Parse(string(body))
|
||||||
|
|
||||||
|
wg := sync.WaitGroup{}
|
||||||
|
posts := make([]Submission, 0)
|
||||||
|
data.ForEach(
|
||||||
|
func(key, value gjson.Result) bool {
|
||||||
|
wg.Add(1)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
defer wg.Done()
|
||||||
|
posts = append(posts, Submission{
|
||||||
|
Id: value.Get("id").String(),
|
||||||
|
Title: value.Get("title").String(),
|
||||||
|
Link: strings.ReplaceAll(value.Get("url").String(), "https://imgur.com", ""),
|
||||||
|
Cover: Media{
|
||||||
|
Id: value.Get("cover_id").String(),
|
||||||
|
Type: value.Get("cover.type").String(),
|
||||||
|
Url: strings.ReplaceAll(value.Get("cover.url").String(), "https://i.imgur.com", ""),
|
||||||
|
},
|
||||||
|
Points: value.Get("point_count").Int(),
|
||||||
|
Upvotes: value.Get("upvote_count").Int(),
|
||||||
|
Downvotes: value.Get("downvote_count").Int(),
|
||||||
|
Comments: value.Get("comment_count").Int(),
|
||||||
|
Views: value.Get("view_count").Int(),
|
||||||
|
IsAlbum: value.Get("is_album").Bool(),
|
||||||
|
})
|
||||||
|
}()
|
||||||
|
|
||||||
|
return true
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
|
client.Cache.Set(fmt.Sprintf("trending-%s-%s-%s", section, sort, page), posts, cache.DefaultExpiration)
|
||||||
|
return posts, nil
|
||||||
|
}
|
17
main.go
17
main.go
|
@ -24,7 +24,7 @@ func main() {
|
||||||
envPath := flag.String("c", ".env", "Path to env file")
|
envPath := flag.String("c", ".env", "Path to env file")
|
||||||
godotenv.Load(*envPath)
|
godotenv.Load(*envPath)
|
||||||
utils.LoadConfig()
|
utils.LoadConfig()
|
||||||
|
|
||||||
pages.InitializeApiClient()
|
pages.InitializeApiClient()
|
||||||
|
|
||||||
views := http.FS(views.GetFiles())
|
views := http.FS(views.GetFiles())
|
||||||
|
@ -32,7 +32,7 @@ func main() {
|
||||||
views = http.Dir("./views")
|
views = http.Dir("./views")
|
||||||
}
|
}
|
||||||
engine := handlebars.NewFileSystem(views, ".hbs")
|
engine := handlebars.NewFileSystem(views, ".hbs")
|
||||||
|
|
||||||
engine.AddFunc("noteq", func(a interface{}, b interface{}, options *raymond.Options) interface{} {
|
engine.AddFunc("noteq", func(a interface{}, b interface{}, options *raymond.Options) interface{} {
|
||||||
if raymond.Str(a) != raymond.Str(b) {
|
if raymond.Str(a) != raymond.Str(b) {
|
||||||
return options.Fn()
|
return options.Fn()
|
||||||
|
@ -69,11 +69,11 @@ func main() {
|
||||||
fmt.Println(e)
|
fmt.Println(e)
|
||||||
},
|
},
|
||||||
}))
|
}))
|
||||||
|
|
||||||
if os.Getenv("ENV") == "dev" {
|
if os.Getenv("ENV") == "dev" {
|
||||||
app.Use("/static", filesystem.New(filesystem.Config{
|
app.Use("/static", filesystem.New(filesystem.Config{
|
||||||
MaxAge: 2592000,
|
MaxAge: 2592000,
|
||||||
Root: http.Dir("./static"),
|
Root: http.Dir("./static"),
|
||||||
}))
|
}))
|
||||||
app.Get("/errors/429", func(c *fiber.Ctx) error {
|
app.Get("/errors/429", func(c *fiber.Ctx) error {
|
||||||
return c.Render("errors/429", nil)
|
return c.Render("errors/429", nil)
|
||||||
|
@ -91,11 +91,11 @@ func main() {
|
||||||
Root: http.FS(static.GetFiles()),
|
Root: http.FS(static.GetFiles()),
|
||||||
}))
|
}))
|
||||||
app.Use(cache.New(cache.Config{
|
app.Use(cache.New(cache.Config{
|
||||||
Expiration: 30 * time.Minute,
|
Expiration: 30 * time.Minute,
|
||||||
MaxBytes: 25000000,
|
MaxBytes: 25000000,
|
||||||
KeyGenerator: func(c *fiber.Ctx) string {
|
KeyGenerator: func(c *fiber.Ctx) string {
|
||||||
return c.OriginalURL()
|
return c.OriginalURL()
|
||||||
},
|
},
|
||||||
CacheControl: true,
|
CacheControl: true,
|
||||||
StoreResponseHeaders: true,
|
StoreResponseHeaders: true,
|
||||||
}))
|
}))
|
||||||
|
@ -116,6 +116,7 @@ func main() {
|
||||||
app.Get("/about", pages.HandleAbout)
|
app.Get("/about", pages.HandleAbout)
|
||||||
app.Get("/privacy", pages.HandlePrivacy)
|
app.Get("/privacy", pages.HandlePrivacy)
|
||||||
app.Get("/search", pages.HandleSearch)
|
app.Get("/search", pages.HandleSearch)
|
||||||
|
app.Get("/trending", pages.HandleTrending)
|
||||||
app.Get("/a/:postID", pages.HandlePost)
|
app.Get("/a/:postID", pages.HandlePost)
|
||||||
app.Get("/a/:postID/embed", pages.HandleEmbed)
|
app.Get("/a/:postID/embed", pages.HandleEmbed)
|
||||||
app.Get("/t/:tag", pages.HandleTag)
|
app.Get("/t/:tag", pages.HandleTag)
|
||||||
|
|
|
@ -0,0 +1,58 @@
|
||||||
|
package pages
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"codeberg.org/rimgo/rimgo/utils"
|
||||||
|
"github.com/gofiber/fiber/v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
func HandleTrending(c *fiber.Ctx) error {
|
||||||
|
utils.SetHeaders(c)
|
||||||
|
c.Set("X-Frame-Options", "DENY")
|
||||||
|
c.Set("Cache-Control", "public,max-age=604800")
|
||||||
|
c.Set("Content-Security-Policy", "default-src 'none'; frame-ancestors 'none'; base-uri 'none'; form-action 'self'; style-src 'unsafe-inline' 'self'; media-src 'self'; img-src 'self'; manifest-src 'self'; block-all-mixed-content")
|
||||||
|
|
||||||
|
page := "1"
|
||||||
|
if c.Query("page") != "" {
|
||||||
|
page = c.Query("page")
|
||||||
|
}
|
||||||
|
|
||||||
|
pageNumber, err := strconv.Atoi(c.Query("page"))
|
||||||
|
if err != nil {
|
||||||
|
pageNumber = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
section := c.Query("section")
|
||||||
|
switch section {
|
||||||
|
case "hot", "new", "top":
|
||||||
|
default:
|
||||||
|
section = "hot"
|
||||||
|
}
|
||||||
|
sort := c.Query("sort")
|
||||||
|
switch sort {
|
||||||
|
case "newest", "best", "popular":
|
||||||
|
default:
|
||||||
|
sort = "popular"
|
||||||
|
}
|
||||||
|
|
||||||
|
displayPrevPage := true
|
||||||
|
if page == "1" {
|
||||||
|
displayPrevPage = false
|
||||||
|
}
|
||||||
|
|
||||||
|
results, err := ApiClient.FetchTrending(section, sort, page)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.Render("trending", fiber.Map{
|
||||||
|
"results": results,
|
||||||
|
"section": section,
|
||||||
|
"sort": sort,
|
||||||
|
"page": pageNumber,
|
||||||
|
"displayPrev": displayPrevPage,
|
||||||
|
"nextPage": pageNumber + 1,
|
||||||
|
"prevPage": pageNumber - 1,
|
||||||
|
})
|
||||||
|
}
|
|
@ -0,0 +1 @@
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="1em" height="1em" viewBox="0 0 256 256"><path fill="currentColor" d="M200 64v104a8 8 0 0 1-16 0V83.31L69.66 197.66a8 8 0 0 1-11.32-11.32L172.69 72H88a8 8 0 0 1 0-16h104a8 8 0 0 1 8 8Z"/></svg>
|
After Width: | Height: | Size: 239 B |
|
@ -13,6 +13,7 @@
|
||||||
<header class="my-8 p-8 rounded-xl flex flex-col gap-4 items-center justify-center bg-gradient-to-r from-blue-400 to-emerald-400">
|
<header class="my-8 p-8 rounded-xl flex flex-col gap-4 items-center justify-center bg-gradient-to-r from-blue-400 to-emerald-400">
|
||||||
<h2 class="font-bold text-white text-2xl">The fast, private image viewer for Imgur.</h2>
|
<h2 class="font-bold text-white text-2xl">The fast, private image viewer for Imgur.</h2>
|
||||||
{{> partials/searchBar }}
|
{{> partials/searchBar }}
|
||||||
|
<a class="flex gap-1 items-center" href="/trending">Or see what's trending <img class="invert" src="/static/icons/PhArrowUpRight.svg" alt="" height="18" width="18" /></a>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<main class="my-8">
|
<main class="my-8">
|
||||||
|
|
|
@ -0,0 +1,79 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<title>Trending - rimgo</title>
|
||||||
|
|
||||||
|
{{> partials/head }}
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body class="font-sans text-lg bg-slate-800 text-white">
|
||||||
|
{{> partials/nav }}
|
||||||
|
|
||||||
|
<section class="my-4 w-full flex flex-col items-center">
|
||||||
|
{{> partials/searchBar }}
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<header class="p-4 rounded-xl text-white mb-4 bg-gradient-to-r from-blue-400 to-emerald-400">
|
||||||
|
<div class="flex flex-col items-center justify-center text-center">
|
||||||
|
<h2 class="text-2xl font-extrabold">Trending</h2>
|
||||||
|
</div>
|
||||||
|
<div class="flex flex-col sm:flex-row sm:justify-between">
|
||||||
|
<div class="flex flex-col">
|
||||||
|
{{#equal section "hot"}}
|
||||||
|
<a href="?section=hot&sort={{sort}}"><b>Hot</b></a>
|
||||||
|
<a href="?section=new&sort={{sort}}">New</a>
|
||||||
|
<a href="?section=top&sort={{sort}}">Top</a>
|
||||||
|
{{/equal}}
|
||||||
|
{{#equal section "new"}}
|
||||||
|
<a href="?section=hot&sort={{sort}}">Hot</a>
|
||||||
|
<a href="?section=new&sort={{sort}}"><b>New</b></a>
|
||||||
|
<a href="?section=top&sort={{sort}}">Top</a>
|
||||||
|
{{/equal}}
|
||||||
|
{{#equal section "top"}}
|
||||||
|
<a href="?section=hot&sort={{sort}}">Hot</a>
|
||||||
|
<a href="?section=new&sort={{sort}}">New</a>
|
||||||
|
<a href="?section=top&sort={{sort}}"><b>Top</b></a>
|
||||||
|
{{/equal}}
|
||||||
|
</div>
|
||||||
|
<hr class="sm:hidden my-2" />
|
||||||
|
<div class="flex flex-col sm:items-end">
|
||||||
|
{{#equal sort "popular"}}
|
||||||
|
<a href="?section={{section}}&sort=popular"><b>Popular</b></a>
|
||||||
|
<a href="?section={{section}}&sort=newest">Newest</a>
|
||||||
|
<a href="?section={{section}}&sort=best">Best</a>
|
||||||
|
{{/equal}}
|
||||||
|
{{#equal sort "newest"}}
|
||||||
|
<a href="?section={{section}}&sort=popular">Popular</a>
|
||||||
|
<a href="?section={{section}}&sort=newest"><b>Newest</b></a>
|
||||||
|
<a href="?section={{section}}&sort=best">Best</a>
|
||||||
|
{{/equal}}
|
||||||
|
{{#equal sort "best"}}
|
||||||
|
<a href="?section={{section}}&sort=popular">Popular</a>
|
||||||
|
<a href="?section={{section}}&sort=newest">Newest</a>
|
||||||
|
<a href="?section={{section}}&sort=best"><b>Best</b></a>
|
||||||
|
{{/equal}}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main>
|
||||||
|
<div class="posts">
|
||||||
|
{{#each results}}
|
||||||
|
{{> partials/post }}
|
||||||
|
{{/each}}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="flex justify-between mt-4 font-bold">
|
||||||
|
{{#if displayPrev}}
|
||||||
|
<a href="/trending?section={{section}}&sort={{sort}}&page={{prevPage}}">Previous page</a>
|
||||||
|
{{/if}}
|
||||||
|
<p>Page {{page}}</p>
|
||||||
|
<a href="/trending?section={{section}}&sort={{sort}}&page={{nextPage}}">Next page</a>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
|
||||||
|
{{> partials/footer }}
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
Loading…
Reference in New Issue