Два API-эндпоинта

This commit is contained in:
lost+skunk 2024-09-03 15:36:19 +03:00
parent 1464584264
commit db53a8bd90
10 changed files with 115 additions and 48 deletions

View File

@ -1,11 +1,13 @@
# v1.3.x # v1.3.x
* Почистить говнокод * Почистить говнокод
* Добавить фильтры поиска
* ~~Сделать порт под FreeBSD~~ ✔️ * ~~Сделать порт под FreeBSD~~ ✔️
* **Доделать парсинг описания** * **Доделать парсинг описания**
* ~~Реализовать стрипы в ежедневных артах~~ ✔️ * ~~Реализовать стрипы в ежедневных артах~~ ✔️
* ~~Исправить баг с навигацией по страницам~~ ✔️ * ~~Исправить баг с навигацией по страницам~~ ✔️
* Сделать нормальное отображение ошибок * Сделать нормальное отображение ошибок
* ~~Сделать единицы в конфиге более понятными~~ ✔️ * ~~Сделать единицы в конфиге более понятными~~ ✔️
* Добавить чекер инстанса на работоспособность
* ~~Добавить просмотр понравившихся артов пользователю~~ ✔️ * ~~Добавить просмотр понравившихся артов пользователю~~ ✔️
* Добавить возможность включить темплейты в бинарник [P] * Добавить возможность включить темплейты в бинарник [P]
* ~~Реализовать миниатюры и оптимизировать CSS под маленькие экраны~~ ✔️ * ~~Реализовать миниатюры и оптимизировать CSS под маленькие экраны~~ ✔️
@ -13,9 +15,10 @@
* **Реализовать отображение контента, отличного от картинок (видео, аудио, etc)** * **Реализовать отображение контента, отличного от картинок (видео, аудио, etc)**
* Исправить баг с эмоджи, когда некоторые кастомные эмоции могут не отображаться * Исправить баг с эмоджи, когда некоторые кастомные эмоции могут не отображаться
* ~~Добавить аргумент &filename, который будет выдавать файл с нормально выглядещем именем~~ ✔️ * ~~Добавить аргумент &filename, который будет выдавать файл с нормально выглядещем именем~~ ✔️
* ~~Улучшить систему кеширования: добавить рейтинг для удаления и копирование изображений в ОЗУ~~ BUG: почему-то всё переодически встаёт раком * ~~Улучшить систему кеширования: добавить рейтинг для удаления и копирование изображений в ОЗУ~~ ✔️
# v1.4 # v1.4
* Реализовать API * Реализовать API
* Реализовать темы * Реализовать темы
* Перейти на арены в кеше * Перейти на арены в кеше
* Реализовать многоязычный интерфейс * Реализовать многоязычный интерфейс

View File

@ -1,6 +1,7 @@
package app package app
import ( import (
"encoding/json"
"math/rand" "math/rand"
"strings" "strings"
@ -8,27 +9,65 @@ import (
) )
type API struct { type API struct {
skunkyartLink *skunkyart main *skunkyart
}
type info struct {
Version string `json:"version"`
Settings settingsParams `json:"settings"`
}
func (a API) Info() {
json, err := json.Marshal(info{
Version: a.main.Version,
Settings: settingsParams{
Nsfw: CFG.Nsfw,
Proxy: CFG.Proxy,
},
})
try(err)
a.main.Writer.Write(json)
return
}
func (a API) Error(description string, status int) {
a.main.Writer.WriteHeader(status)
var response strings.Builder
response.WriteString(`{"error":"`)
response.WriteString(description)
response.WriteString(`"}`)
wr(a.main.Writer, response.String())
} }
func (a API) sendMedia(d *devianter.Deviation) { func (a API) sendMedia(d *devianter.Deviation) {
mediaUrl, name := devianter.UrlFromMedia(d.Media) mediaUrl, name := devianter.UrlFromMedia(d.Media)
a.main.SetFilename(name)
var filename strings.Builder
filename.WriteString(`filename="`)
filename.WriteString(name)
filename.WriteString(`"`)
a.skunkyartLink.Writer.Header().Add("Content-Disposition", filename.String())
if len(mediaUrl) != 0 { if len(mediaUrl) != 0 {
mediaUrl = mediaUrl[21:] mediaUrl = mediaUrl[21:]
dot := strings.Index(mediaUrl, ".") dot := strings.Index(mediaUrl, ".")
a.skunkyartLink.DownloadAndSendMedia(mediaUrl[:dot], mediaUrl[dot+11:]) a.main.Writer.Header().Del("Content-Type")
a.main.DownloadAndSendMedia(mediaUrl[:dot], mediaUrl[dot+11:])
} }
} }
// TODO: сделать фильтры
func (a API) Random() { func (a API) Random() {
s, err := devianter.PerformSearch(string(rand.Intn(999)), rand.Intn(30), 'a') for attempt := 1;; {
try(err) if attempt > 3 {
a.sendMedia(&s.Results[rand.Intn(len(s.Results))]) a.Error("Sorry, butt NSFW on this are disabled, and the instance failed to find a random art without NSFW", 500)
}
s, err := devianter.PerformSearch(string(rand.Intn(999)), rand.Intn(30), 'a')
try(err)
deviation := &s.Results[rand.Intn(len(s.Results))]
if deviation.NSFW && !CFG.Nsfw {
attempt++
continue
}
a.sendMedia(deviation)
return
}
} }

View File

@ -4,19 +4,20 @@ import (
"bufio" "bufio"
"bytes" "bytes"
"encoding/json" "encoding/json"
"html/template"
"os" "os"
"time" "time"
) )
func ExecuteCommandLineArguments() { func ExecuteCommandLineArguments() {
const helpmsg = `SkunkyArt v1.3.1 [CSS improvements for mobile and the strips on Daily Deviations] var helpmsg = `SkunkyArt v{{.Version}} [{{.Description}}]
Usage: Usage:
- [-c|--config] | path to config - [-c|--config] | path to config
- [-a|--add-instance] | generates 'instances.json' and 'INSTANCES.md' files with ur instance - [-a|--add-instance] | generates 'instances.json' and 'INSTANCES.md' files with ur instance
- [-h|--help] | returns this message - [-h|--help] | returns this message
Example: Example:
./skunkyart -c config.json ./skunkyart -c config.json
Copyright lost+skunk, X11. https://git.macaw.me/skunky/skunkyart/src/tag/v1.3.1` Copyright lost+skunk, X11. https://git.macaw.me/skunky/skunkyart/src/tag/v{{.Version}}`
a := os.Args[1:] a := os.Args[1:]
for n, x := range a { for n, x := range a {
@ -28,7 +29,11 @@ Copyright lost+skunk, X11. https://git.macaw.me/skunky/skunkyart/src/tag/v1.3.1`
exit("Not enought arguments", 1) exit("Not enought arguments", 1)
} }
case "-h", "--help": case "-h", "--help":
exit(helpmsg, 0) var buf bytes.Buffer
t := template.New("help")
t.Parse(helpmsg)
t.Execute(&buf, &Release)
exit(buf.String(), 0)
case "-a", "--add-instance": case "-a", "--add-instance":
addInstance() addInstance()
} }

View File

@ -11,6 +11,11 @@ import (
"git.macaw.me/skunky/devianter" "git.macaw.me/skunky/devianter"
) )
var Release struct {
Version string
Description string
}
type cache_config struct { type cache_config struct {
Enabled bool Enabled bool
Path string Path string

View File

@ -13,10 +13,9 @@ var Host, Path string
func Router() { func Router() {
parsepath := func(path string) map[int]string { parsepath := func(path string) map[int]string {
path = "/"
if l := len(CFG.URI); len(path) > l { if l := len(CFG.URI); len(path) > l {
path = path[l-1:] path = path[l-1:]
} else {
path = "/"
} }
parsedpath := make(map[int]string) parsedpath := make(map[int]string)
@ -54,44 +53,42 @@ func Router() {
// функция, что управляет всем // функция, что управляет всем
handle := func(w http.ResponseWriter, r *http.Request) { handle := func(w http.ResponseWriter, r *http.Request) {
if h := r.Header["X-Forwarded-Proto"]; len(h) != 0 && h[0] == "https" {
Host = h[0] + "://" + r.Host
} else {
Host = "http://" + r.Host
}
Path = r.URL.Path Path = r.URL.Path
path := parsepath(Path) path := parsepath(Path)
// структура с функциями Host = "http://" + r.Host
var skunky skunkyart
if h := r.Header["X-Forwarded-Proto"]; len(h) != 0 && h[0] == "https" {
Host = "https://" + r.Host
}
var skunky = skunkyart{Version: Release.Version}
arg := skunky.Args.Get
p, _ := strconv.Atoi(arg("p"))
skunky.Endpoint = path[1]
skunky.API.main = &skunky
skunky.Writer = w skunky.Writer = w
skunky.Args = r.URL.Query() skunky.Args = r.URL.Query()
skunky.BasePath = CFG.URI skunky.BasePath = CFG.URI
arg := skunky.Args.Get
skunky.QueryRaw = arg("q") skunky.QueryRaw = arg("q")
skunky.Query = u.QueryEscape(skunky.QueryRaw) skunky.Query = u.QueryEscape(skunky.QueryRaw)
skunky.Page = p
if t := arg("type"); len(t) > 0 { if t := arg("type"); len(t) > 0 {
skunky.Type = rune(t[0]) skunky.Type = rune(t[0])
} }
p, _ := strconv.Atoi(arg("p"))
skunky.Page = p
if arg("atom") == "true" { if arg("atom") == "true" {
skunky.Atom = true skunky.Atom = true
} }
skunky.Endpoint = path[1]
skunky.API.skunkyartLink = &skunky
// пути
switch skunky.Endpoint { switch skunky.Endpoint {
default: // main
skunky.ReturnHTTPError(404)
case "": case "":
skunky.ExecuteTemplate("index.htm", "html", &CFG.URI) skunky.ExecuteTemplate("index.htm", "html", &CFG.URI)
case "about":
skunky.About()
case "post": case "post":
skunky.Deviation(path[2], path[3]) skunky.Deviation(path[2], path[3])
case "search": case "search":
@ -101,33 +98,38 @@ func Router() {
case "group_user": case "group_user":
skunky.GRUser() skunky.GRUser()
// media
case "media": case "media":
switch path[2] { switch path[2] {
case "file": case "file":
if a := arg("filename"); a != "" { if a := arg("filename"); a != "" {
var filename strings.Builder skunky.SetFilename(a)
filename.WriteString(`filename="`)
filename.WriteString(a)
filename.WriteString(`"`)
w.Header().Add("Content-Disposition", filename.String())
} }
skunky.DownloadAndSendMedia(path[3], next(path, 4)) skunky.DownloadAndSendMedia(path[3], next(path, 4))
case "emojitar": case "emojitar":
skunky.Emojitar(path[3]) skunky.Emojitar(path[3])
} }
case "about":
skunky.About()
case "stylesheet": case "stylesheet":
w.Header().Add("content-type", "text/css") w.Header().Add("content-type", "text/css")
w.Write(open("css/skunky.css")) w.Write(open("css/skunky.css"))
case "favicon.ico": case "favicon.ico":
w.Write(open("images/logo.png")) w.Write(open("images/logo.png"))
// API
case "api": case "api":
w.Header().Add("Content-Type", "application/json")
switch path[2] { switch path[2] {
case "instance":
skunky.API.Info()
case "random": case "random":
skunky.API.Random() skunky.API.Random()
default:
skunky.API.Error("Not Found", 404)
} }
// 404
default:
skunky.ReturnHTTPError(404)
} }
} }

View File

@ -64,6 +64,7 @@ type skunkyart struct {
Query, QueryRaw string Query, QueryRaw string
API API API API
Version string
Templates struct { Templates struct {
About struct { About struct {
@ -131,7 +132,7 @@ func UrlBuilder(strs ...string) string {
str.WriteString(CFG.URI) str.WriteString(CFG.URI)
for n, x := range strs { for n, x := range strs {
str.WriteString(x) str.WriteString(x)
if n+1 < l && !(strs[n+1][0] == '?' || strs[n+1][0] == '&') && !(x[0] == '?' || x[0] == '&') { if n := n+1; n < l && len(strs[n]) != 0 && !(strs[n][0] == '?' || strs[n][0] == '&') && !(x[0] == '?' || x[0] == '&') {
str.WriteString("/") str.WriteString("/")
} }
} }
@ -153,6 +154,14 @@ func (s skunkyart) ReturnHTTPError(status int) {
wr(s.Writer, msg.String()) wr(s.Writer, msg.String())
} }
func (s skunkyart) SetFilename(name string) {
var filename strings.Builder
filename.WriteString(`filename="`)
filename.WriteString(name)
filename.WriteString(`"`)
s.Writer.Header().Add("Content-Disposition", filename.String())
}
type Downloaded struct { type Downloaded struct {
Headers http.Header Headers http.Header
Status int Status int
@ -189,6 +198,9 @@ func ParseMedia(media devianter.Media, thumb ...int) string {
if len(mediaUrl) != 0 && CFG.Proxy { if len(mediaUrl) != 0 && CFG.Proxy {
mediaUrl = mediaUrl[21:] mediaUrl = mediaUrl[21:]
dot := strings.Index(mediaUrl, ".") dot := strings.Index(mediaUrl, ".")
if filename == "" {
filename = "image.gif"
}
return UrlBuilder("media", "file", mediaUrl[:dot], mediaUrl[dot+11:], "&filename=", filename) return UrlBuilder("media", "file", mediaUrl[:dot], mediaUrl[dot+11:], "&filename=", filename)
} }
return mediaUrl return mediaUrl

View File

@ -42,7 +42,6 @@ func (s skunkyart) GRUser() {
group.About.A = x.ModuleData.About group.About.A = x.ModuleData.About
var about = &group.About.A var about = &group.About.A
group.CreationDate = time.Unix(time.Now().Unix()-x.ModuleData.About.RegDate, 0).UTC().String() group.CreationDate = time.Unix(time.Now().Unix()-x.ModuleData.About.RegDate, 0).UTC().String()
group.About.DescriptionFormatted = ParseDescription(about.Description) group.About.DescriptionFormatted = ParseDescription(about.Description)
for _, val := range x.ModuleData.About.SocialLinks { for _, val := range x.ModuleData.About.SocialLinks {

2
go.mod
View File

@ -2,7 +2,7 @@ module skunkyart
go 1.18 go 1.18
// replace git.macaw.me/skunky/devianter v0.2.5 => /home/skunk/projects/devianter replace git.macaw.me/skunky/devianter v0.2.5 => /home/skunk/projects/devianter
require ( require (
git.macaw.me/skunky/devianter v0.2.5 git.macaw.me/skunky/devianter v0.2.5
golang.org/x/net v0.27.0 golang.org/x/net v0.27.0

View File

@ -9,6 +9,8 @@ import (
) )
func main() { func main() {
app.Release.Version = "1.3.2-alpha"
app.Release.Description = "Two API endpoints and template embedding into binary"
go app.RefreshInstances() go app.RefreshInstances()
app.ExecuteCommandLineArguments() app.ExecuteCommandLineArguments()

View File

@ -44,6 +44,6 @@
{{end}} {{end}}
</ul> </ul>
</details> </details>
<p>Copyright <a href="https://go.kde.org/matrix/#/@softpigeones:ebloid.ru" target="_blank">lost+skunk</a>, X11. <a href="https://git.macaw.me/skunky/skunkyart/src/tag/v1.3.1" target="_blank">SkunkyArt v1.3.1</a></p> <p>Copyright <a href="https://go.kde.org/matrix/#/@softpigeones:ebloid.ru" target="_blank">lost+skunk</a>, X11. <a href="https://git.macaw.me/skunky/skunkyart/src/tag/v{{.Version}}" target="_blank">SkunkyArt v{{.Version}}</a></p>
</main> </main>
</html> </html>