mirror of
https://codeberg.org/Codeberg/pages-server.git
synced 2025-01-19 08:57:55 +00:00
magic
This commit is contained in:
parent
4a2a14272b
commit
94cb43508c
2 changed files with 57 additions and 30 deletions
|
@ -1,8 +1,30 @@
|
||||||
package gitea
|
package gitea
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"codeberg.org/codeberg/pages/server/cache"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// defaultBranchCacheTimeout specifies the timeout for the default branch cache. It can be quite long.
|
||||||
|
defaultBranchCacheTimeout = 15 * time.Minute
|
||||||
|
|
||||||
|
// branchExistenceCacheTimeout specifies the timeout for the branch timestamp & existence cache. It should be shorter
|
||||||
|
// than fileCacheTimeout, as that gets invalidated if the branch timestamp has changed. That way, repo changes will be
|
||||||
|
// picked up faster, while still allowing the content to be cached longer if nothing changes.
|
||||||
|
branchExistenceCacheTimeout = 5 * time.Minute
|
||||||
|
|
||||||
|
// fileCacheTimeout specifies the timeout for the file content cache - you might want to make this quite long, depending
|
||||||
|
// on your available memory.
|
||||||
|
// TODO: move as option into cache interface
|
||||||
|
fileCacheTimeout = 5 * time.Minute
|
||||||
|
|
||||||
|
// fileCacheSizeLimit limits the maximum file size that will be cached, and is set to 1 MB by default.
|
||||||
|
fileCacheSizeLimit = int64(1024 * 1024)
|
||||||
)
|
)
|
||||||
|
|
||||||
type FileResponse struct {
|
type FileResponse struct {
|
||||||
|
@ -42,20 +64,39 @@ type BranchTimestamp struct {
|
||||||
Timestamp time.Time
|
Timestamp time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
const (
|
type writeCacheReader struct {
|
||||||
// defaultBranchCacheTimeout specifies the timeout for the default branch cache. It can be quite long.
|
r io.Reader
|
||||||
defaultBranchCacheTimeout = 15 * time.Minute
|
buff *bytes.Buffer
|
||||||
|
f *FileResponse
|
||||||
|
cacheKey string
|
||||||
|
cache cache.SetGetKey
|
||||||
|
hasErr bool
|
||||||
|
}
|
||||||
|
|
||||||
// branchExistenceCacheTimeout specifies the timeout for the branch timestamp & existence cache. It should be shorter
|
func (t *writeCacheReader) Read(p []byte) (n int, err error) {
|
||||||
// than fileCacheTimeout, as that gets invalidated if the branch timestamp has changed. That way, repo changes will be
|
n, err = t.r.Read(p)
|
||||||
// picked up faster, while still allowing the content to be cached longer if nothing changes.
|
if err != nil {
|
||||||
branchExistenceCacheTimeout = 5 * time.Minute
|
t.hasErr = true
|
||||||
|
} else if n > 0 {
|
||||||
|
_, _ = t.buff.Write(p[:n])
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// fileCacheTimeout specifies the timeout for the file content cache - you might want to make this quite long, depending
|
func (t *writeCacheReader) Close() error {
|
||||||
// on your available memory.
|
if !t.hasErr {
|
||||||
// TODO: move as option into cache interface
|
t.f.Body = t.buff.Bytes()
|
||||||
fileCacheTimeout = 5 * time.Minute
|
t.cache.Set(t.cacheKey, *t.f, fileCacheTimeout)
|
||||||
|
}
|
||||||
|
return t.Close()
|
||||||
|
}
|
||||||
|
|
||||||
// fileCacheSizeLimit limits the maximum file size that will be cached, and is set to 1 MB by default.
|
func (f FileResponse) CreateCacheReader(r io.ReadCloser, cache cache.SetGetKey, cacheKey string) io.ReadCloser {
|
||||||
fileCacheSizeLimit = int64(1024 * 1024)
|
buf := []byte{}
|
||||||
)
|
return &writeCacheReader{
|
||||||
|
r: r,
|
||||||
|
buff: bytes.NewBuffer(buf),
|
||||||
|
f: &f,
|
||||||
|
cacheKey: cacheKey,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -158,26 +158,12 @@ func (client *Client) ServeRawContent(targetOwner, targetRepo, ref, resource str
|
||||||
}
|
}
|
||||||
|
|
||||||
// now we write to cache and respond at the sime time
|
// now we write to cache and respond at the sime time
|
||||||
|
fileResp := FileResponse{
|
||||||
// TODO: at the sime time !!!
|
|
||||||
/*
|
|
||||||
we create a new type that implement an writer witch write to cache based on key etc ...
|
|
||||||
// TODO: cache is half-empty if request is cancelled - does the ctx.Err() below do the trick?
|
|
||||||
// err = res.BodyWriteTo(io.MultiWriter(ctx.Response().BodyWriter(), &cacheBodyWriter))
|
|
||||||
*/
|
|
||||||
body, err := io.ReadAll(io.LimitReader(reader, fileCacheSizeLimit))
|
|
||||||
if err != nil {
|
|
||||||
log.Error().Err(err).Msg("not expected")
|
|
||||||
}
|
|
||||||
if err := client.responseCache.Set(cacheKey, FileResponse{
|
|
||||||
Exists: true,
|
Exists: true,
|
||||||
ETag: resp.Header.Get(eTagHeader),
|
ETag: resp.Header.Get(eTagHeader),
|
||||||
MimeType: mimeType,
|
MimeType: mimeType,
|
||||||
Body: body,
|
|
||||||
}, fileCacheTimeout); err != nil {
|
|
||||||
log.Error().Err(err).Msg("could not save content in cache")
|
|
||||||
}
|
}
|
||||||
return io.NopCloser(bytes.NewReader(body)), resp.Response, nil
|
return fileResp.CreateCacheReader(reader, client.responseCache, cacheKey), resp.Response, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
case http.StatusNotFound:
|
case http.StatusNotFound:
|
||||||
|
|
Loading…
Reference in a new issue