Небольшие улучшения в группах
This commit is contained in:
parent
949bef2c5d
commit
0a6260b2e0
106
misc.go
106
misc.go
@ -1,79 +1,14 @@
|
|||||||
package devianter
|
package devianter
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
|
||||||
"log"
|
"log"
|
||||||
"math"
|
"math"
|
||||||
"net/http"
|
|
||||||
u "net/url"
|
u "net/url"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
// функция для высера ошибки в stderr
|
|
||||||
func err(txt error) {
|
|
||||||
if txt != nil {
|
|
||||||
println(txt.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// сокращение для вызова щенка и парсинга жсона
|
|
||||||
func ujson(data string, output any) {
|
|
||||||
input, e := puppy(data)
|
|
||||||
err(e)
|
|
||||||
|
|
||||||
eee := json.Unmarshal([]byte(input), output)
|
|
||||||
err(eee)
|
|
||||||
}
|
|
||||||
|
|
||||||
/* REQUEST SECTION */
|
|
||||||
// структура для ответа сервера
|
|
||||||
type reqrt struct {
|
|
||||||
Body string
|
|
||||||
Status int
|
|
||||||
Cookies []*http.Cookie
|
|
||||||
Headers http.Header
|
|
||||||
}
|
|
||||||
|
|
||||||
// функция для совершения запроса
|
|
||||||
func request(uri string, other ...string) reqrt {
|
|
||||||
var r reqrt
|
|
||||||
|
|
||||||
// создаём новый запрос
|
|
||||||
cli := &http.Client{}
|
|
||||||
req, e := http.NewRequest("GET", uri, nil)
|
|
||||||
err(e)
|
|
||||||
|
|
||||||
req.Header.Set("User-Agent", "Mozilla/5.0 (X11; Linux x86_64; rv:123.0) Gecko/20100101 Firefox/123.0.0")
|
|
||||||
|
|
||||||
// куки и UA-шник
|
|
||||||
for num, rng := range other {
|
|
||||||
switch num {
|
|
||||||
case 1:
|
|
||||||
req.Header.Set("User-Agent", rng)
|
|
||||||
case 0:
|
|
||||||
req.Header.Set("Cookie", rng)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
resp, e := cli.Do(req)
|
|
||||||
err(e)
|
|
||||||
defer resp.Body.Close()
|
|
||||||
|
|
||||||
body, e := io.ReadAll(resp.Body)
|
|
||||||
err(e)
|
|
||||||
|
|
||||||
// заполняем структуру
|
|
||||||
r.Body = string(body)
|
|
||||||
r.Cookies = resp.Cookies()
|
|
||||||
r.Headers = resp.Header
|
|
||||||
r.Status = resp.StatusCode
|
|
||||||
|
|
||||||
return r
|
|
||||||
}
|
|
||||||
|
|
||||||
/* AVATARS AND EMOJIS */
|
/* AVATARS AND EMOJIS */
|
||||||
func AEmedia(name string, t rune) (string, error) {
|
func AEmedia(name string, t rune) (string, error) {
|
||||||
// список всех возможных расширений
|
// список всех возможных расширений
|
||||||
@ -189,44 +124,3 @@ func SearchFunc(query string, page int, scope rune, user ...string) (ss Search,
|
|||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
/* PUPPY aka DeviantArt API */
|
|
||||||
// получение или обновление токена
|
|
||||||
var cookie string
|
|
||||||
var token string
|
|
||||||
|
|
||||||
func UpdateCSRF() error {
|
|
||||||
if cookie == "" {
|
|
||||||
req := request("https://www.deviantart.com/_puppy")
|
|
||||||
|
|
||||||
for _, content := range req.Cookies {
|
|
||||||
cookie = content.Raw
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
req := request("https://www.deviantart.com", cookie)
|
|
||||||
if req.Status != 200 {
|
|
||||||
return errors.New(req.Body)
|
|
||||||
}
|
|
||||||
token = req.Body[strings.Index(req.Body, "window.__CSRF_TOKEN__ = '")+25 : strings.Index(req.Body, "window.__XHR_LOCAL__")-3]
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func puppy(data string) (string, error) {
|
|
||||||
var url strings.Builder
|
|
||||||
url.WriteString("https://www.deviantart.com/_puppy/")
|
|
||||||
url.WriteString(data)
|
|
||||||
url.WriteString("&csrf_token=")
|
|
||||||
url.WriteString(token)
|
|
||||||
url.WriteString("&da_minor_version=20230710")
|
|
||||||
|
|
||||||
body := request(url.String(), cookie)
|
|
||||||
|
|
||||||
// если код ответа не 200, возвращается ошибка
|
|
||||||
if body.Status != 200 {
|
|
||||||
return "", errors.New(body.Body)
|
|
||||||
}
|
|
||||||
|
|
||||||
return body.Body, nil
|
|
||||||
}
|
|
||||||
|
@ -6,13 +6,14 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// структура группы или пользователя
|
// структура группы или пользователя
|
||||||
type groups struct {
|
type GroupAbout struct {
|
||||||
GroupAbout struct {
|
|
||||||
FoundatedAt time `json:"foundationTs"`
|
FoundatedAt time `json:"foundationTs"`
|
||||||
Description Text
|
Description Text
|
||||||
}
|
}
|
||||||
GroupAdmins struct {
|
type GroupAdmins struct {
|
||||||
Results []struct {
|
Results []struct {
|
||||||
|
TypeId int
|
||||||
|
User struct {
|
||||||
Username string
|
Username string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -50,23 +51,9 @@ type GRuser struct {
|
|||||||
Modules []struct {
|
Modules []struct {
|
||||||
Name string
|
Name string
|
||||||
ModuleData struct {
|
ModuleData struct {
|
||||||
groups
|
GroupAbout GroupAbout
|
||||||
|
GroupAdmins GroupAdmins
|
||||||
users
|
users
|
||||||
|
|
||||||
// группы
|
|
||||||
Folders struct {
|
|
||||||
Results []struct {
|
|
||||||
FolderId int
|
|
||||||
Name string
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// галерея
|
|
||||||
Folder struct {
|
|
||||||
Username string
|
|
||||||
Pages int `json:"totalPageCount"`
|
|
||||||
Deviations []Deviation
|
|
||||||
} `json:"folderDeviations"`
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -80,9 +67,40 @@ type GRuser struct {
|
|||||||
} `json:"pageExtraData"`
|
} `json:"pageExtraData"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Gallery struct {
|
||||||
|
Gruser struct {
|
||||||
|
ID int `json:"gruserId"`
|
||||||
|
Page struct {
|
||||||
|
Modules []struct {
|
||||||
|
Name string
|
||||||
|
ModuleData struct {
|
||||||
|
// группы
|
||||||
|
Folders struct {
|
||||||
|
HasMore bool
|
||||||
|
Results []struct {
|
||||||
|
FolderId int
|
||||||
|
Name string
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// галерея
|
||||||
|
Folder struct {
|
||||||
|
HasMore bool
|
||||||
|
Username string
|
||||||
|
Pages int `json:"totalPageCount"`
|
||||||
|
Deviations []Deviation
|
||||||
|
} `json:"folderDeviations"`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
HasMore bool
|
||||||
|
Results []Deviation
|
||||||
|
}
|
||||||
|
|
||||||
type Group struct {
|
type Group struct {
|
||||||
Name string // обязательно заполнить
|
Name string // обязательно заполнить
|
||||||
Content GRuser
|
Content Gallery
|
||||||
}
|
}
|
||||||
|
|
||||||
// подходит как группа, так и пользователь
|
// подходит как группа, так и пользователь
|
||||||
@ -93,14 +111,27 @@ func (s Group) GroupFunc() (g GRuser) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// гарелея пользователя или группы
|
// гарелея пользователя или группы
|
||||||
func (s Group) Gallery(page int) (g Group) {
|
func (s Group) Gallery(page int, folderid ...int) (g Group) {
|
||||||
var url strings.Builder
|
var url strings.Builder
|
||||||
|
if folderid[0] > 0 {
|
||||||
|
page--
|
||||||
|
url.WriteString("dashared/gallection/contents?username=")
|
||||||
|
url.WriteString(s.Name)
|
||||||
|
url.WriteString("&folderid=")
|
||||||
|
url.WriteString(strconv.Itoa(folderid[0]))
|
||||||
|
url.WriteString("&offset=")
|
||||||
|
url.WriteString(strconv.Itoa(page * 50))
|
||||||
|
url.WriteString("&type=gallery&")
|
||||||
|
} else {
|
||||||
url.WriteString("dauserprofile/init/gallery?username=")
|
url.WriteString("dauserprofile/init/gallery?username=")
|
||||||
url.WriteString(s.Name)
|
url.WriteString(s.Name)
|
||||||
url.WriteString("&page=")
|
url.WriteString("&page=")
|
||||||
url.WriteString(strconv.Itoa(page))
|
url.WriteString(strconv.Itoa(page))
|
||||||
url.WriteString("&deviations_limit=50&with_subfolders=false")
|
url.WriteString("&deviations_")
|
||||||
|
}
|
||||||
|
url.WriteString("limit=50")
|
||||||
|
url.WriteString("&with_subfolders=false")
|
||||||
|
|
||||||
ujson(url.String(), &g)
|
ujson(url.String(), &g.Content)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
112
util.go
Normal file
112
util.go
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
package devianter
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// функция для высера ошибки в stderr
|
||||||
|
func err(txt error) {
|
||||||
|
if txt != nil {
|
||||||
|
println(txt.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// сокращение для вызова щенка и парсинга жсона
|
||||||
|
func ujson(data string, output any) {
|
||||||
|
input, e := puppy(data)
|
||||||
|
err(e)
|
||||||
|
|
||||||
|
eee := json.Unmarshal([]byte(input), output)
|
||||||
|
err(eee)
|
||||||
|
}
|
||||||
|
|
||||||
|
/* REQUEST SECTION */
|
||||||
|
// структура для ответа сервера
|
||||||
|
type reqrt struct {
|
||||||
|
Body string
|
||||||
|
Status int
|
||||||
|
Cookies []*http.Cookie
|
||||||
|
Headers http.Header
|
||||||
|
}
|
||||||
|
|
||||||
|
// функция для совершения запроса
|
||||||
|
func request(uri string, other ...string) reqrt {
|
||||||
|
var r reqrt
|
||||||
|
|
||||||
|
// создаём новый запрос
|
||||||
|
cli := &http.Client{}
|
||||||
|
req, e := http.NewRequest("GET", uri, nil)
|
||||||
|
err(e)
|
||||||
|
|
||||||
|
req.Header.Set("User-Agent", "Mozilla/5.0 (X11; Linux x86_64; rv:123.0) Gecko/20100101 Firefox/123.0.0")
|
||||||
|
|
||||||
|
// куки и UA-шник
|
||||||
|
for num, rng := range other {
|
||||||
|
switch num {
|
||||||
|
case 1:
|
||||||
|
req.Header.Set("User-Agent", rng)
|
||||||
|
case 0:
|
||||||
|
req.Header.Set("Cookie", rng)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, e := cli.Do(req)
|
||||||
|
err(e)
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
body, e := io.ReadAll(resp.Body)
|
||||||
|
err(e)
|
||||||
|
|
||||||
|
// заполняем структуру
|
||||||
|
r.Body = string(body)
|
||||||
|
r.Cookies = resp.Cookies()
|
||||||
|
r.Headers = resp.Header
|
||||||
|
r.Status = resp.StatusCode
|
||||||
|
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
/* PUPPY aka DeviantArt API */
|
||||||
|
// получение или обновление токена
|
||||||
|
var cookie string
|
||||||
|
var token string
|
||||||
|
|
||||||
|
func UpdateCSRF() error {
|
||||||
|
if cookie == "" {
|
||||||
|
req := request("https://www.deviantart.com/_puppy")
|
||||||
|
|
||||||
|
for _, content := range req.Cookies {
|
||||||
|
cookie = content.Raw
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
req := request("https://www.deviantart.com", cookie)
|
||||||
|
if req.Status != 200 {
|
||||||
|
return errors.New(req.Body)
|
||||||
|
}
|
||||||
|
token = req.Body[strings.Index(req.Body, "window.__CSRF_TOKEN__ = '")+25 : strings.Index(req.Body, "window.__XHR_LOCAL__")-3]
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func puppy(data string) (string, error) {
|
||||||
|
var url strings.Builder
|
||||||
|
url.WriteString("https://www.deviantart.com/_puppy/")
|
||||||
|
url.WriteString(data)
|
||||||
|
url.WriteString("&csrf_token=")
|
||||||
|
url.WriteString(token)
|
||||||
|
url.WriteString("&da_minor_version=20230710")
|
||||||
|
|
||||||
|
body := request(url.String(), cookie)
|
||||||
|
|
||||||
|
// если код ответа не 200, возвращается ошибка
|
||||||
|
if body.Status != 200 {
|
||||||
|
return "", errors.New(body.Body)
|
||||||
|
}
|
||||||
|
|
||||||
|
return body.Body, nil
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user