requests.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. package core
  2. import (
  3. "bytes"
  4. "context"
  5. "errors"
  6. "fmt"
  7. "io"
  8. "net/http"
  9. "time"
  10. "codeberg.org/vnpower/pixivfe/v2/audit"
  11. config "codeberg.org/vnpower/pixivfe/v2/config"
  12. "codeberg.org/vnpower/pixivfe/v2/request_context"
  13. "codeberg.org/vnpower/pixivfe/v2/utils"
  14. "github.com/tidwall/gjson"
  15. )
  16. type SimpleHTTPResponse struct {
  17. StatusCode int
  18. Body string
  19. }
  20. // send GET
  21. func API_GET(context context.Context, url string, token string) (SimpleHTTPResponse, error) {
  22. start_time := time.Now()
  23. res, resp, err := _API_GET(context, url, token)
  24. end_time := time.Now()
  25. audit.LogAPIRoundTrip(audit.APIRequestSpan{
  26. RequestId: request_context.GetFromContext(context).RequestId,
  27. Response: resp, Error: err, Method: "GET", Url: url, Token: token, Body: res.Body, StartTime: start_time, EndTime: end_time})
  28. if err != nil {
  29. return SimpleHTTPResponse{}, fmt.Errorf("While GET %s: %w", url, err)
  30. }
  31. return res, nil
  32. }
  33. func _API_GET(context context.Context, url string, token string) (SimpleHTTPResponse, *http.Response, error) {
  34. var res SimpleHTTPResponse
  35. req, err := http.NewRequestWithContext(context, "GET", url, nil)
  36. if err != nil {
  37. return res, nil, err
  38. }
  39. req.Header.Add("User-Agent", config.GlobalConfig.UserAgent)
  40. req.Header.Add("Accept-Language", config.GlobalConfig.AcceptLanguage)
  41. if token == "" {
  42. req.AddCookie(&http.Cookie{
  43. Name: "PHPSESSID",
  44. Value: config.GetRandomDefaultToken(),
  45. })
  46. } else {
  47. req.AddCookie(&http.Cookie{
  48. Name: "PHPSESSID",
  49. Value: token,
  50. })
  51. }
  52. // Make the request
  53. resp, err := utils.HttpClient.Do(req)
  54. if err != nil {
  55. return res, nil, err
  56. }
  57. defer resp.Body.Close()
  58. body, err := io.ReadAll(resp.Body)
  59. if err != nil {
  60. return res, resp, err
  61. }
  62. res = SimpleHTTPResponse{
  63. StatusCode: resp.StatusCode,
  64. Body: string(body),
  65. }
  66. return res, resp, nil
  67. }
  68. func API_GET_UnwrapJson(context context.Context, url string, token string) (string, error) {
  69. resp, err := API_GET(context, url, token)
  70. if err != nil {
  71. return "", err
  72. }
  73. if !gjson.Valid(resp.Body) {
  74. return "", fmt.Errorf("Invalid JSON: %v", resp.Body)
  75. }
  76. err2 := gjson.Get(resp.Body, "error")
  77. if !err2.Exists() {
  78. return "", errors.New("Incompatible request body")
  79. }
  80. if err2.Bool() {
  81. return "", errors.New(gjson.Get(resp.Body, "message").String())
  82. }
  83. return gjson.Get(resp.Body, "body").String(), nil
  84. }
  85. // send POST
  86. func API_POST(context context.Context, url, payload, token, csrf string, isJSON bool) error {
  87. start_time := time.Now()
  88. resp, err := _API_POST(context, url, payload, token, csrf, isJSON)
  89. end_time := time.Now()
  90. audit.LogAPIRoundTrip(audit.APIRequestSpan{
  91. RequestId: request_context.GetFromContext(context).RequestId,
  92. Response: resp, Error: err, Method: "POST", Url: url, Token: token, Body: "", StartTime: start_time, EndTime: end_time})
  93. if err != nil {
  94. return fmt.Errorf("While POST %s: %w", url, err)
  95. }
  96. return err
  97. }
  98. func _API_POST(context context.Context, url, payload, token, csrf string, isJSON bool) (*http.Response, error) {
  99. requestBody := []byte(payload)
  100. req, err := http.NewRequestWithContext(context, "POST", url, bytes.NewBuffer(requestBody))
  101. if err != nil {
  102. return nil, err
  103. }
  104. req.Header.Add("User-Agent", "Mozilla/5.0")
  105. req.Header.Add("Accept", "application/json")
  106. req.Header.Add("x-csrf-token", csrf)
  107. req.AddCookie(&http.Cookie{
  108. Name: "PHPSESSID",
  109. Value: token,
  110. })
  111. if isJSON {
  112. req.Header.Add("Content-Type", "application/json; charset=utf-8")
  113. } else {
  114. req.Header.Add("Content-Type", "application/x-www-form-urlencoded; charset=utf-8")
  115. }
  116. resp, err := utils.HttpClient.Do(req)
  117. if err != nil {
  118. return nil, err
  119. }
  120. defer resp.Body.Close()
  121. body, err := io.ReadAll(resp.Body)
  122. if err != nil {
  123. return resp, err
  124. }
  125. body_s := string(body)
  126. if !gjson.Valid(body_s) {
  127. return resp, fmt.Errorf("Invalid JSON: %v", body_s)
  128. }
  129. err2 := gjson.Get(body_s, "error")
  130. if !err2.Exists() {
  131. return resp, fmt.Errorf("Incompatible request body.")
  132. }
  133. if err2.Bool() {
  134. return resp, fmt.Errorf("Pixiv: Invalid request.")
  135. }
  136. return resp, nil
  137. }
  138. func ProxyRequest(w http.ResponseWriter, req *http.Request) error {
  139. // Make the request
  140. resp, err := utils.HttpClient.Do(req)
  141. if err != nil {
  142. return err
  143. }
  144. defer resp.Body.Close()
  145. // copy headers
  146. header := w.Header()
  147. for k, v := range resp.Header {
  148. header[k] = v
  149. }
  150. // copy body
  151. _, err = io.Copy(w, resp.Body)
  152. return err
  153. }