Два API-эндпоинта
This commit is contained in:
parent
1464584264
commit
db53a8bd90
5
TODO.md
5
TODO.md
@ -1,11 +1,13 @@
|
||||
# v1.3.x
|
||||
* Почистить говнокод
|
||||
* Добавить фильтры поиска
|
||||
* ~~Сделать порт под FreeBSD~~ ✔️
|
||||
* **Доделать парсинг описания**
|
||||
* ~~Реализовать стрипы в ежедневных артах~~ ✔️
|
||||
* ~~Исправить баг с навигацией по страницам~~ ✔️
|
||||
* Сделать нормальное отображение ошибок
|
||||
* ~~Сделать единицы в конфиге более понятными~~ ✔️
|
||||
* Добавить чекер инстанса на работоспособность
|
||||
* ~~Добавить просмотр понравившихся артов пользователю~~ ✔️
|
||||
* Добавить возможность включить темплейты в бинарник [P]
|
||||
* ~~Реализовать миниатюры и оптимизировать CSS под маленькие экраны~~ ✔️
|
||||
@ -13,9 +15,10 @@
|
||||
* **Реализовать отображение контента, отличного от картинок (видео, аудио, etc)**
|
||||
* Исправить баг с эмоджи, когда некоторые кастомные эмоции могут не отображаться
|
||||
* ~~Добавить аргумент &filename, который будет выдавать файл с нормально выглядещем именем~~ ✔️
|
||||
* ~~Улучшить систему кеширования: добавить рейтинг для удаления и копирование изображений в ОЗУ~~ BUG: почему-то всё переодически встаёт раком
|
||||
* ~~Улучшить систему кеширования: добавить рейтинг для удаления и копирование изображений в ОЗУ~~ ✔️
|
||||
# v1.4
|
||||
* Реализовать API
|
||||
* Реализовать темы
|
||||
* Перейти на арены в кеше
|
||||
* Реализовать многоязычный интерфейс
|
||||
|
||||
|
61
app/api.go
61
app/api.go
@ -1,6 +1,7 @@
|
||||
package app
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"math/rand"
|
||||
"strings"
|
||||
|
||||
@ -8,27 +9,65 @@ import (
|
||||
)
|
||||
|
||||
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) {
|
||||
mediaUrl, name := devianter.UrlFromMedia(d.Media)
|
||||
|
||||
var filename strings.Builder
|
||||
filename.WriteString(`filename="`)
|
||||
filename.WriteString(name)
|
||||
filename.WriteString(`"`)
|
||||
a.skunkyartLink.Writer.Header().Add("Content-Disposition", filename.String())
|
||||
a.main.SetFilename(name)
|
||||
|
||||
if len(mediaUrl) != 0 {
|
||||
mediaUrl = mediaUrl[21:]
|
||||
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() {
|
||||
s, err := devianter.PerformSearch(string(rand.Intn(999)), rand.Intn(30), 'a')
|
||||
try(err)
|
||||
a.sendMedia(&s.Results[rand.Intn(len(s.Results))])
|
||||
for attempt := 1;; {
|
||||
if attempt > 3 {
|
||||
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
|
||||
}
|
||||
}
|
||||
|
11
app/cli.go
11
app/cli.go
@ -4,19 +4,20 @@ import (
|
||||
"bufio"
|
||||
"bytes"
|
||||
"encoding/json"
|
||||
"html/template"
|
||||
"os"
|
||||
"time"
|
||||
)
|
||||
|
||||
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:
|
||||
- [-c|--config] | path to config
|
||||
- [-a|--add-instance] | generates 'instances.json' and 'INSTANCES.md' files with ur instance
|
||||
- [-h|--help] | returns this message
|
||||
Example:
|
||||
./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:]
|
||||
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)
|
||||
}
|
||||
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":
|
||||
addInstance()
|
||||
}
|
||||
|
@ -11,6 +11,11 @@ import (
|
||||
"git.macaw.me/skunky/devianter"
|
||||
)
|
||||
|
||||
var Release struct {
|
||||
Version string
|
||||
Description string
|
||||
}
|
||||
|
||||
type cache_config struct {
|
||||
Enabled bool
|
||||
Path string
|
||||
|
@ -13,10 +13,9 @@ var Host, Path string
|
||||
|
||||
func Router() {
|
||||
parsepath := func(path string) map[int]string {
|
||||
path = "/"
|
||||
if l := len(CFG.URI); len(path) > l {
|
||||
path = path[l-1:]
|
||||
} else {
|
||||
path = "/"
|
||||
}
|
||||
|
||||
parsedpath := make(map[int]string)
|
||||
@ -54,44 +53,42 @@ func Router() {
|
||||
|
||||
// функция, что управляет всем
|
||||
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 := parsepath(Path)
|
||||
// структура с функциями
|
||||
var skunky skunkyart
|
||||
Host = "http://" + r.Host
|
||||
|
||||
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.Args = r.URL.Query()
|
||||
skunky.BasePath = CFG.URI
|
||||
|
||||
arg := skunky.Args.Get
|
||||
skunky.QueryRaw = arg("q")
|
||||
skunky.Query = u.QueryEscape(skunky.QueryRaw)
|
||||
skunky.Page = p
|
||||
|
||||
if t := arg("type"); len(t) > 0 {
|
||||
skunky.Type = rune(t[0])
|
||||
}
|
||||
p, _ := strconv.Atoi(arg("p"))
|
||||
skunky.Page = p
|
||||
|
||||
if arg("atom") == "true" {
|
||||
skunky.Atom = true
|
||||
}
|
||||
|
||||
skunky.Endpoint = path[1]
|
||||
skunky.API.skunkyartLink = &skunky
|
||||
|
||||
// пути
|
||||
switch skunky.Endpoint {
|
||||
default:
|
||||
skunky.ReturnHTTPError(404)
|
||||
|
||||
// main
|
||||
case "":
|
||||
skunky.ExecuteTemplate("index.htm", "html", &CFG.URI)
|
||||
case "about":
|
||||
skunky.About()
|
||||
case "post":
|
||||
skunky.Deviation(path[2], path[3])
|
||||
case "search":
|
||||
@ -101,33 +98,38 @@ func Router() {
|
||||
case "group_user":
|
||||
skunky.GRUser()
|
||||
|
||||
// media
|
||||
case "media":
|
||||
switch path[2] {
|
||||
case "file":
|
||||
if a := arg("filename"); a != "" {
|
||||
var filename strings.Builder
|
||||
filename.WriteString(`filename="`)
|
||||
filename.WriteString(a)
|
||||
filename.WriteString(`"`)
|
||||
w.Header().Add("Content-Disposition", filename.String())
|
||||
skunky.SetFilename(a)
|
||||
}
|
||||
skunky.DownloadAndSendMedia(path[3], next(path, 4))
|
||||
case "emojitar":
|
||||
skunky.Emojitar(path[3])
|
||||
}
|
||||
case "about":
|
||||
skunky.About()
|
||||
case "stylesheet":
|
||||
w.Header().Add("content-type", "text/css")
|
||||
w.Write(open("css/skunky.css"))
|
||||
case "favicon.ico":
|
||||
w.Write(open("images/logo.png"))
|
||||
|
||||
// API
|
||||
case "api":
|
||||
w.Header().Add("Content-Type", "application/json")
|
||||
switch path[2] {
|
||||
case "instance":
|
||||
skunky.API.Info()
|
||||
case "random":
|
||||
skunky.API.Random()
|
||||
default:
|
||||
skunky.API.Error("Not Found", 404)
|
||||
}
|
||||
|
||||
// 404
|
||||
default:
|
||||
skunky.ReturnHTTPError(404)
|
||||
}
|
||||
}
|
||||
|
||||
|
14
app/util.go
14
app/util.go
@ -64,6 +64,7 @@ type skunkyart struct {
|
||||
Query, QueryRaw string
|
||||
|
||||
API API
|
||||
Version string
|
||||
|
||||
Templates struct {
|
||||
About struct {
|
||||
@ -131,7 +132,7 @@ func UrlBuilder(strs ...string) string {
|
||||
str.WriteString(CFG.URI)
|
||||
for n, x := range strs {
|
||||
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("/")
|
||||
}
|
||||
}
|
||||
@ -153,6 +154,14 @@ func (s skunkyart) ReturnHTTPError(status int) {
|
||||
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 {
|
||||
Headers http.Header
|
||||
Status int
|
||||
@ -189,6 +198,9 @@ func ParseMedia(media devianter.Media, thumb ...int) string {
|
||||
if len(mediaUrl) != 0 && CFG.Proxy {
|
||||
mediaUrl = mediaUrl[21:]
|
||||
dot := strings.Index(mediaUrl, ".")
|
||||
if filename == "" {
|
||||
filename = "image.gif"
|
||||
}
|
||||
return UrlBuilder("media", "file", mediaUrl[:dot], mediaUrl[dot+11:], "&filename=", filename)
|
||||
}
|
||||
return mediaUrl
|
||||
|
@ -42,7 +42,6 @@ func (s skunkyart) GRUser() {
|
||||
group.About.A = x.ModuleData.About
|
||||
var about = &group.About.A
|
||||
group.CreationDate = time.Unix(time.Now().Unix()-x.ModuleData.About.RegDate, 0).UTC().String()
|
||||
|
||||
group.About.DescriptionFormatted = ParseDescription(about.Description)
|
||||
|
||||
for _, val := range x.ModuleData.About.SocialLinks {
|
||||
|
2
go.mod
2
go.mod
@ -2,7 +2,7 @@ module skunkyart
|
||||
|
||||
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 (
|
||||
git.macaw.me/skunky/devianter v0.2.5
|
||||
golang.org/x/net v0.27.0
|
||||
|
2
main.go
2
main.go
@ -9,6 +9,8 @@ import (
|
||||
)
|
||||
|
||||
func main() {
|
||||
app.Release.Version = "1.3.2-alpha"
|
||||
app.Release.Description = "Two API endpoints and template embedding into binary"
|
||||
go app.RefreshInstances()
|
||||
|
||||
app.ExecuteCommandLineArguments()
|
||||
|
@ -44,6 +44,6 @@
|
||||
{{end}}
|
||||
</ul>
|
||||
</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>
|
||||
</html>
|
||||
</html>
|
||||
|
Loading…
Reference in New Issue
Block a user