rankingCalendar.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. package core
  2. import (
  3. "fmt"
  4. "html/template"
  5. "net/http"
  6. "time"
  7. session "codeberg.org/vnpower/pixivfe/v2/core/session"
  8. url "codeberg.org/vnpower/pixivfe/v2/core/http"
  9. "github.com/gofiber/fiber/v2"
  10. "golang.org/x/net/html"
  11. )
  12. func get_weekday(n time.Weekday) int {
  13. switch n {
  14. case time.Sunday:
  15. return 1
  16. case time.Monday:
  17. return 2
  18. case time.Tuesday:
  19. return 3
  20. case time.Wednesday:
  21. return 4
  22. case time.Thursday:
  23. return 5
  24. case time.Friday:
  25. return 6
  26. case time.Saturday:
  27. return 7
  28. }
  29. return 0
  30. }
  31. // note(@iacore):
  32. // so the funny thing about Pixiv is that they will return this month's data for a request of a future date
  33. // is it a bug or a feature?
  34. func GetRankingCalendar(c *fiber.Ctx, mode string, year, month int) (template.HTML, error) {
  35. token := session.GetPixivToken(c)
  36. URL := url.GetRankingCalendarURL(mode, year, month)
  37. req, err := http.NewRequest("GET", URL, nil)
  38. if err != nil {
  39. return template.HTML(""), err
  40. }
  41. req = req.WithContext(c.Context())
  42. req.Header.Add("User-Agent", "Mozilla/5.0")
  43. req.Header.Add("Cookie", "PHPSESSID="+token)
  44. // req.AddCookie(&http.Cookie{
  45. // Name: "PHPSESSID",
  46. // Value: token,
  47. // })
  48. resp, err := http.DefaultClient.Do(req)
  49. if err != nil {
  50. return "", err
  51. }
  52. defer resp.Body.Close()
  53. // Use the html package to parse the response body from the request
  54. doc, err := html.Parse(resp.Body)
  55. if err != nil {
  56. return "", err
  57. }
  58. // Find and print all links on the web page
  59. var links []string
  60. var link func(*html.Node)
  61. link = func(n *html.Node) {
  62. if n.Type == html.ElementNode && n.Data == "img" {
  63. for _, a := range n.Attr {
  64. if a.Key == "data-src" {
  65. // adds a new link entry when the attribute matches
  66. links = append(links, session.ProxyImageUrlNoEscape(c, a.Val))
  67. }
  68. }
  69. }
  70. // traverses the HTML of the webpage from the first child node
  71. for c := n.FirstChild; c != nil; c = c.NextSibling {
  72. link(c)
  73. }
  74. }
  75. link(doc)
  76. // now := c.Context().Time()
  77. // yearNow := now.Year()
  78. // monthNow := now.Month()
  79. lastMonth := time.Date(year, time.Month(month), 0, 0, 0, 0, 0, time.UTC)
  80. thisMonth := time.Date(year, time.Month(month+1), 0, 0, 0, 0, 0, time.UTC)
  81. renderString := ""
  82. for i := 0; i < get_weekday(lastMonth.Weekday()); i++ {
  83. renderString += "<div class=\"calendar-node calendar-node-empty\"></div>"
  84. }
  85. for i := 0; i < thisMonth.Day(); i++ {
  86. date := fmt.Sprintf("%d%02d%02d", year, month, i+1)
  87. if len(links) > i {
  88. renderString += fmt.Sprintf(`<a href="/ranking?mode=%s&date=%s"><div class="calendar-node"><img src="%s" alt="Day %d" /><span>%d</span></div></a>`, mode, date, links[i], i+1, i+1)
  89. } else {
  90. renderString += fmt.Sprintf(`<div class="calendar-node"><span>%d</span></div>`, i+1)
  91. }
  92. }
  93. return template.HTML(renderString), nil
  94. }