mirror of
https://github.com/OpenListTeam/OpenList.git
synced 2025-11-25 03:15:19 +08:00
* feat(cache): improve cache management * feat(disk-usage): add cache * feat(disk-usage): add refresh * fix(disk-usage): cache with ttl * feat(cache): implement KeyedCache and TypedCache for improved caching mechanism * fix(copy): update object retrieval to use Get instead of GetUnwrap * refactor(cache): simplify DirectoryCache structure and improve object management * fix(cache): correct cache entry initialization and key deletion logic in TypedCache * refactor(driver): remove GetObjInfo interface and simplify Link function logic https://github.com/OpenListTeam/OpenList/pull/888/files#r2430925783 * fix(link): optimize link retrieval and caching logic * refactor(cache): consolidate cache management and improve directory cache handling * fix(cache): add cache control based on storage configuration in List function * . * refactor: replace fmt.Sprintf with strconv for integer conversions * refactor(cache): enhance cache entry management with Expirable interface * fix(cache): improve link reference acquisition logic to handle expiration * refactor: replace OnlyLinkMFile with NoLinkSF in driver configurations and logic * refactor(link): enhance link caching logic with dynamic type keys based on IP and User-Agent * feat(drivers): add LinkCacheType to driver configurations for enhanced caching * refactor(cache): streamline directory object management in cache operations * refactor(cache): remove unnecessary 'dirty' field from CacheEntry structure * refactor(cache): replace 'dirty' field with bitwise flags * refactor(io): 调高SyncClosers.AcquireReference的优先级 * refactor(link): 优化链接获取逻辑,增加重 * refactor(link): 添加RequireReference字段以增强链接管理 * refactor(link): 移除MFile字段,改用RangeReader * refactor: 移除不必要的NoLinkSF字段 * refactor(cache): 修改目录缓存的脏标志定义和更新逻辑 * feat(cache): add expiration gc --------- Co-authored-by: KirCute <951206789@qq.com> Co-authored-by: KirCute <kircute@foxmail.com> Co-authored-by: j2rong4cn <j2rong@qq.com>
178 lines
5.0 KiB
Go
178 lines
5.0 KiB
Go
package handles
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
stdpath "path"
|
|
"strconv"
|
|
|
|
"github.com/OpenListTeam/OpenList/v4/internal/conf"
|
|
"github.com/OpenListTeam/OpenList/v4/internal/driver"
|
|
"github.com/OpenListTeam/OpenList/v4/internal/errs"
|
|
"github.com/OpenListTeam/OpenList/v4/internal/fs"
|
|
"github.com/OpenListTeam/OpenList/v4/internal/model"
|
|
"github.com/OpenListTeam/OpenList/v4/internal/net"
|
|
"github.com/OpenListTeam/OpenList/v4/internal/setting"
|
|
"github.com/OpenListTeam/OpenList/v4/pkg/utils"
|
|
"github.com/OpenListTeam/OpenList/v4/server/common"
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/microcosm-cc/bluemonday"
|
|
log "github.com/sirupsen/logrus"
|
|
"github.com/yuin/goldmark"
|
|
)
|
|
|
|
func Down(c *gin.Context) {
|
|
rawPath := c.Request.Context().Value(conf.PathKey).(string)
|
|
filename := stdpath.Base(rawPath)
|
|
storage, err := fs.GetStorage(rawPath, &fs.GetStoragesArgs{})
|
|
if err != nil {
|
|
common.ErrorPage(c, err, 500)
|
|
return
|
|
}
|
|
if common.ShouldProxy(storage, filename) {
|
|
Proxy(c)
|
|
return
|
|
} else {
|
|
link, _, err := fs.Link(c.Request.Context(), rawPath, model.LinkArgs{
|
|
IP: c.ClientIP(),
|
|
Header: c.Request.Header,
|
|
Type: c.Query("type"),
|
|
Redirect: true,
|
|
})
|
|
if err != nil {
|
|
common.ErrorPage(c, err, 500)
|
|
return
|
|
}
|
|
redirect(c, link)
|
|
}
|
|
}
|
|
|
|
func Proxy(c *gin.Context) {
|
|
rawPath := c.Request.Context().Value(conf.PathKey).(string)
|
|
filename := stdpath.Base(rawPath)
|
|
storage, err := fs.GetStorage(rawPath, &fs.GetStoragesArgs{})
|
|
if err != nil {
|
|
common.ErrorPage(c, err, 500)
|
|
return
|
|
}
|
|
if canProxy(storage, filename) {
|
|
if _, ok := c.GetQuery("d"); !ok {
|
|
if url := common.GenerateDownProxyURL(storage.GetStorage(), rawPath); url != "" {
|
|
c.Redirect(302, url)
|
|
return
|
|
}
|
|
}
|
|
link, file, err := fs.Link(c.Request.Context(), rawPath, model.LinkArgs{
|
|
Header: c.Request.Header,
|
|
Type: c.Query("type"),
|
|
})
|
|
if err != nil {
|
|
common.ErrorPage(c, err, 500)
|
|
return
|
|
}
|
|
proxy(c, link, file, storage.GetStorage().ProxyRange)
|
|
} else {
|
|
common.ErrorPage(c, errors.New("proxy not allowed"), 403)
|
|
return
|
|
}
|
|
}
|
|
|
|
func redirect(c *gin.Context, link *model.Link) {
|
|
defer link.Close()
|
|
var err error
|
|
c.Header("Referrer-Policy", "no-referrer")
|
|
c.Header("Cache-Control", "max-age=0, no-cache, no-store, must-revalidate")
|
|
if setting.GetBool(conf.ForwardDirectLinkParams) {
|
|
query := c.Request.URL.Query()
|
|
for _, v := range conf.SlicesMap[conf.IgnoreDirectLinkParams] {
|
|
query.Del(v)
|
|
}
|
|
link.URL, err = utils.InjectQuery(link.URL, query)
|
|
if err != nil {
|
|
common.ErrorPage(c, err, 500)
|
|
return
|
|
}
|
|
}
|
|
c.Redirect(302, link.URL)
|
|
}
|
|
|
|
func proxy(c *gin.Context, link *model.Link, file model.Obj, proxyRange bool) {
|
|
defer link.Close()
|
|
var err error
|
|
if link.URL != "" && setting.GetBool(conf.ForwardDirectLinkParams) {
|
|
query := c.Request.URL.Query()
|
|
for _, v := range conf.SlicesMap[conf.IgnoreDirectLinkParams] {
|
|
query.Del(v)
|
|
}
|
|
link.URL, err = utils.InjectQuery(link.URL, query)
|
|
if err != nil {
|
|
common.ErrorPage(c, err, 500)
|
|
return
|
|
}
|
|
}
|
|
if proxyRange {
|
|
link = common.ProxyRange(c, link, file.GetSize())
|
|
}
|
|
Writer := &common.WrittenResponseWriter{ResponseWriter: c.Writer}
|
|
raw, _ := strconv.ParseBool(c.DefaultQuery("raw", "false"))
|
|
if utils.Ext(file.GetName()) == "md" && setting.GetBool(conf.FilterReadMeScripts) && !raw {
|
|
buf := bytes.NewBuffer(make([]byte, 0, file.GetSize()))
|
|
w := &common.InterceptResponseWriter{ResponseWriter: Writer, Writer: buf}
|
|
err = common.Proxy(w, c.Request, link, file)
|
|
if err == nil && buf.Len() > 0 {
|
|
if c.Writer.Status() < 200 || c.Writer.Status() > 300 {
|
|
c.Writer.Write(buf.Bytes())
|
|
return
|
|
}
|
|
|
|
var html bytes.Buffer
|
|
if err = goldmark.Convert(buf.Bytes(), &html); err != nil {
|
|
err = fmt.Errorf("markdown conversion failed: %w", err)
|
|
} else {
|
|
buf.Reset()
|
|
err = bluemonday.UGCPolicy().SanitizeReaderToWriter(&html, buf)
|
|
if err == nil {
|
|
Writer.Header().Set("Content-Length", strconv.FormatInt(int64(buf.Len()), 10))
|
|
Writer.Header().Set("Content-Type", "text/html; charset=utf-8")
|
|
_, err = utils.CopyWithBuffer(Writer, buf)
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
err = common.Proxy(Writer, c.Request, link, file)
|
|
}
|
|
if err == nil {
|
|
return
|
|
}
|
|
if Writer.IsWritten() {
|
|
log.Errorf("%s %s local proxy error: %+v", c.Request.Method, c.Request.URL.Path, err)
|
|
} else {
|
|
if statusCode, ok := errs.UnwrapOrSelf(err).(net.HttpStatusCodeError); ok {
|
|
common.ErrorPage(c, err, int(statusCode), true)
|
|
} else {
|
|
common.ErrorPage(c, err, 500, true)
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO need optimize
|
|
// when can be proxy?
|
|
// 1. text file
|
|
// 2. config.MustProxy()
|
|
// 3. storage.WebProxy
|
|
// 4. proxy_types
|
|
// solution: text_file + shouldProxy()
|
|
func canProxy(storage driver.Driver, filename string) bool {
|
|
if storage.Config().MustProxy() || storage.GetStorage().WebProxy || storage.GetStorage().WebdavProxyURL() {
|
|
return true
|
|
}
|
|
if utils.SliceContains(conf.SlicesMap[conf.ProxyTypes], utils.Ext(filename)) {
|
|
return true
|
|
}
|
|
if utils.SliceContains(conf.SlicesMap[conf.TextTypes], utils.Ext(filename)) {
|
|
return true
|
|
}
|
|
return false
|
|
}
|