mirror of
https://github.com/drakkan/sftpgo.git
synced 2024-11-22 07:30:25 +00:00
6ef85d6026
Verifying argon2 passwords has a high memory and computational cost, by enabling, in memory, password caching you reduce this cost
183 lines
4.2 KiB
Go
183 lines
4.2 KiB
Go
package dataprovider
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/drakkan/sftpgo/logger"
|
|
)
|
|
|
|
var delayedQuotaUpdater quotaUpdater
|
|
|
|
func init() {
|
|
delayedQuotaUpdater = newQuotaUpdater()
|
|
}
|
|
|
|
type quotaObject struct {
|
|
size int64
|
|
files int
|
|
}
|
|
|
|
type quotaUpdater struct {
|
|
paramsMutex sync.RWMutex
|
|
waitTime time.Duration
|
|
sync.RWMutex
|
|
pendingUserQuotaUpdates map[string]quotaObject
|
|
pendingFolderQuotaUpdates map[string]quotaObject
|
|
}
|
|
|
|
func newQuotaUpdater() quotaUpdater {
|
|
return quotaUpdater{
|
|
pendingUserQuotaUpdates: make(map[string]quotaObject),
|
|
pendingFolderQuotaUpdates: make(map[string]quotaObject),
|
|
}
|
|
}
|
|
|
|
func (q *quotaUpdater) start() {
|
|
q.setWaitTime(config.DelayedQuotaUpdate)
|
|
|
|
go q.loop()
|
|
}
|
|
|
|
func (q *quotaUpdater) loop() {
|
|
waitTime := q.getWaitTime()
|
|
providerLog(logger.LevelDebug, "delayed quota update loop started, wait time: %v", waitTime)
|
|
for waitTime > 0 {
|
|
// We do this with a time.Sleep instead of a time.Ticker because we don't know
|
|
// how long each quota processing cycle will take, and we want to make
|
|
// sure we wait the configured seconds between each iteration
|
|
time.Sleep(waitTime)
|
|
providerLog(logger.LevelDebug, "delayed quota update check start")
|
|
q.storeUsersQuota()
|
|
q.storeFoldersQuota()
|
|
providerLog(logger.LevelDebug, "delayed quota update check end")
|
|
waitTime = q.getWaitTime()
|
|
}
|
|
providerLog(logger.LevelDebug, "delayed quota update loop ended, wait time: %v", waitTime)
|
|
}
|
|
|
|
func (q *quotaUpdater) setWaitTime(secs int) {
|
|
q.paramsMutex.Lock()
|
|
defer q.paramsMutex.Unlock()
|
|
|
|
q.waitTime = time.Duration(secs) * time.Second
|
|
}
|
|
|
|
func (q *quotaUpdater) getWaitTime() time.Duration {
|
|
q.paramsMutex.RLock()
|
|
defer q.paramsMutex.RUnlock()
|
|
|
|
return q.waitTime
|
|
}
|
|
|
|
func (q *quotaUpdater) resetUserQuota(username string) {
|
|
q.Lock()
|
|
defer q.Unlock()
|
|
|
|
delete(q.pendingUserQuotaUpdates, username)
|
|
}
|
|
|
|
func (q *quotaUpdater) updateUserQuota(username string, files int, size int64) {
|
|
q.Lock()
|
|
defer q.Unlock()
|
|
|
|
obj := q.pendingUserQuotaUpdates[username]
|
|
obj.size += size
|
|
obj.files += files
|
|
if obj.files == 0 && obj.size == 0 {
|
|
delete(q.pendingUserQuotaUpdates, username)
|
|
return
|
|
}
|
|
q.pendingUserQuotaUpdates[username] = obj
|
|
}
|
|
|
|
func (q *quotaUpdater) getUserPendingQuota(username string) (int, int64) {
|
|
q.RLock()
|
|
defer q.RUnlock()
|
|
|
|
obj := q.pendingUserQuotaUpdates[username]
|
|
|
|
return obj.files, obj.size
|
|
}
|
|
|
|
func (q *quotaUpdater) resetFolderQuota(name string) {
|
|
q.Lock()
|
|
defer q.Unlock()
|
|
|
|
delete(q.pendingFolderQuotaUpdates, name)
|
|
}
|
|
|
|
func (q *quotaUpdater) updateFolderQuota(name string, files int, size int64) {
|
|
q.Lock()
|
|
defer q.Unlock()
|
|
|
|
obj := q.pendingFolderQuotaUpdates[name]
|
|
obj.size += size
|
|
obj.files += files
|
|
if obj.files == 0 && obj.size == 0 {
|
|
delete(q.pendingFolderQuotaUpdates, name)
|
|
return
|
|
}
|
|
q.pendingFolderQuotaUpdates[name] = obj
|
|
}
|
|
|
|
func (q *quotaUpdater) getFolderPendingQuota(name string) (int, int64) {
|
|
q.RLock()
|
|
defer q.RUnlock()
|
|
|
|
obj := q.pendingFolderQuotaUpdates[name]
|
|
|
|
return obj.files, obj.size
|
|
}
|
|
|
|
func (q *quotaUpdater) getUsernames() []string {
|
|
q.RLock()
|
|
defer q.RUnlock()
|
|
|
|
result := make([]string, 0, len(q.pendingUserQuotaUpdates))
|
|
for username := range q.pendingUserQuotaUpdates {
|
|
result = append(result, username)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func (q *quotaUpdater) getFoldernames() []string {
|
|
q.RLock()
|
|
defer q.RUnlock()
|
|
|
|
result := make([]string, 0, len(q.pendingFolderQuotaUpdates))
|
|
for name := range q.pendingFolderQuotaUpdates {
|
|
result = append(result, name)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func (q *quotaUpdater) storeUsersQuota() {
|
|
for _, username := range q.getUsernames() {
|
|
files, size := q.getUserPendingQuota(username)
|
|
if size != 0 || files != 0 {
|
|
err := provider.updateQuota(username, files, size, false)
|
|
if err != nil {
|
|
providerLog(logger.LevelWarn, "unable to update quota delayed for user %#v: %v", username, err)
|
|
continue
|
|
}
|
|
q.updateUserQuota(username, -files, -size)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (q *quotaUpdater) storeFoldersQuota() {
|
|
for _, name := range q.getFoldernames() {
|
|
files, size := q.getFolderPendingQuota(name)
|
|
if size != 0 || files != 0 {
|
|
err := provider.updateFolderQuota(name, files, size, false)
|
|
if err != nil {
|
|
providerLog(logger.LevelWarn, "unable to update quota delayed for folder %#v: %v", name, err)
|
|
continue
|
|
}
|
|
q.updateFolderQuota(name, -files, -size)
|
|
}
|
|
}
|
|
}
|