2017-10-08 11:28:03 -07:00
|
|
|
package limiter
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
|
2023-10-01 11:40:12 +02:00
|
|
|
"github.com/restic/restic/internal/backend"
|
2017-10-08 11:28:03 -07:00
|
|
|
)
|
|
|
|
|
2023-10-01 11:40:12 +02:00
|
|
|
func WrapBackendConstructor[B backend.Backend, C any](constructor func(ctx context.Context, cfg C) (B, error)) func(ctx context.Context, cfg C, lim Limiter) (backend.Backend, error) {
|
|
|
|
return func(ctx context.Context, cfg C, lim Limiter) (backend.Backend, error) {
|
|
|
|
var be backend.Backend
|
2023-06-08 13:06:25 +02:00
|
|
|
be, err := constructor(ctx, cfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if lim != nil {
|
|
|
|
be = LimitBackend(be, lim)
|
|
|
|
}
|
|
|
|
return be, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-08 11:28:03 -07:00
|
|
|
// LimitBackend wraps a Backend and applies rate limiting to Load() and Save()
|
|
|
|
// calls on the backend.
|
2023-10-01 11:40:12 +02:00
|
|
|
func LimitBackend(be backend.Backend, l Limiter) backend.Backend {
|
2017-10-08 11:28:03 -07:00
|
|
|
return rateLimitedBackend{
|
|
|
|
Backend: be,
|
|
|
|
limiter: l,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type rateLimitedBackend struct {
|
2023-10-01 11:40:12 +02:00
|
|
|
backend.Backend
|
2017-10-08 11:28:03 -07:00
|
|
|
limiter Limiter
|
|
|
|
}
|
|
|
|
|
2023-10-01 11:40:12 +02:00
|
|
|
func (r rateLimitedBackend) Save(ctx context.Context, h backend.Handle, rd backend.RewindReader) error {
|
2018-03-03 14:20:54 +01:00
|
|
|
limited := limitedRewindReader{
|
|
|
|
RewindReader: rd,
|
|
|
|
limited: r.limiter.Upstream(rd),
|
|
|
|
}
|
|
|
|
|
|
|
|
return r.Backend.Save(ctx, h, limited)
|
|
|
|
}
|
|
|
|
|
|
|
|
type limitedRewindReader struct {
|
2023-10-01 11:40:12 +02:00
|
|
|
backend.RewindReader
|
2018-03-03 14:20:54 +01:00
|
|
|
|
|
|
|
limited io.Reader
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l limitedRewindReader) Read(b []byte) (int, error) {
|
|
|
|
return l.limited.Read(b)
|
2017-10-08 11:28:03 -07:00
|
|
|
}
|
|
|
|
|
2023-10-01 11:40:12 +02:00
|
|
|
func (r rateLimitedBackend) Load(ctx context.Context, h backend.Handle, length int, offset int64, consumer func(rd io.Reader) error) error {
|
2018-01-16 23:59:16 -05:00
|
|
|
return r.Backend.Load(ctx, h, length, offset, func(rd io.Reader) error {
|
2021-04-24 11:46:06 +02:00
|
|
|
return consumer(newDownstreamLimitedReader(rd, r.limiter))
|
2018-01-16 23:59:16 -05:00
|
|
|
})
|
2017-10-08 11:28:03 -07:00
|
|
|
}
|
|
|
|
|
2023-10-01 11:40:12 +02:00
|
|
|
func (r rateLimitedBackend) Unwrap() backend.Backend { return r.Backend }
|
2023-04-08 12:53:43 +02:00
|
|
|
|
2021-04-24 11:46:06 +02:00
|
|
|
type limitedReader struct {
|
2021-01-01 12:46:20 +01:00
|
|
|
io.Reader
|
|
|
|
writerTo io.WriterTo
|
|
|
|
limiter Limiter
|
|
|
|
}
|
|
|
|
|
2021-04-24 11:46:06 +02:00
|
|
|
func newDownstreamLimitedReader(rd io.Reader, limiter Limiter) io.Reader {
|
|
|
|
lrd := limiter.Downstream(rd)
|
|
|
|
if wt, ok := rd.(io.WriterTo); ok {
|
|
|
|
lrd = &limitedReader{
|
|
|
|
Reader: lrd,
|
|
|
|
writerTo: wt,
|
|
|
|
limiter: limiter,
|
2021-01-01 12:46:20 +01:00
|
|
|
}
|
|
|
|
}
|
2021-04-24 11:46:06 +02:00
|
|
|
return lrd
|
2017-10-08 11:28:03 -07:00
|
|
|
}
|
|
|
|
|
2021-04-24 11:46:06 +02:00
|
|
|
func (l *limitedReader) WriteTo(w io.Writer) (int64, error) {
|
2021-01-01 12:46:20 +01:00
|
|
|
return l.writerTo.WriteTo(l.limiter.DownstreamWriter(w))
|
|
|
|
}
|
|
|
|
|
2023-10-01 11:40:12 +02:00
|
|
|
var _ backend.Backend = (*rateLimitedBackend)(nil)
|