2020-01-07 11:23:09 +00:00
// Copyright 2019 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package queue
import (
"fmt"
2020-02-02 23:19:58 +00:00
"strings"
2020-01-07 11:23:09 +00:00
2021-07-25 00:03:58 +08:00
"code.gitea.io/gitea/modules/json"
2020-01-07 11:23:09 +00:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
)
func validType ( t string ) ( Type , error ) {
if len ( t ) == 0 {
return PersistableChannelQueueType , nil
}
for _ , typ := range RegisteredTypes ( ) {
if t == string ( typ ) {
return typ , nil
}
}
2022-01-24 22:54:35 +00:00
return PersistableChannelQueueType , fmt . Errorf ( "unknown queue type: %s defaulting to %s" , t , string ( PersistableChannelQueueType ) )
2020-01-07 11:23:09 +00:00
}
2020-01-29 01:01:06 +00:00
func getQueueSettings ( name string ) ( setting . QueueSettings , [ ] byte ) {
2020-01-07 11:23:09 +00:00
q := setting . GetQueueSettings ( name )
2020-10-15 22:40:03 +01:00
cfg , err := json . Marshal ( q )
2020-01-07 11:23:09 +00:00
if err != nil {
2020-10-15 22:40:03 +01:00
log . Error ( "Unable to marshall generic options: %v Error: %v" , q , err )
2020-01-07 11:23:09 +00:00
log . Error ( "Unable to create queue for %s" , name , err )
2020-01-29 01:01:06 +00:00
return q , [ ] byte { }
}
return q , cfg
}
// CreateQueue for name with provided handler and exemplar
func CreateQueue ( name string , handle HandlerFunc , exemplar interface { } ) Queue {
q , cfg := getQueueSettings ( name )
if len ( cfg ) == 0 {
2020-01-07 11:23:09 +00:00
return nil
}
2020-01-29 01:01:06 +00:00
typ , err := validType ( q . Type )
if err != nil {
log . Error ( "Invalid type %s provided for queue named %s defaulting to %s" , q . Type , name , string ( typ ) )
}
2020-01-07 11:23:09 +00:00
returnable , err := NewQueue ( typ , handle , cfg , exemplar )
if q . WrapIfNecessary && err != nil {
log . Warn ( "Unable to create queue for %s: %v" , name , err )
log . Warn ( "Attempting to create wrapped queue" )
returnable , err = NewQueue ( WrappedQueueType , handle , WrappedQueueConfiguration {
2020-01-29 01:01:06 +00:00
Underlying : typ ,
2020-01-07 11:23:09 +00:00
Timeout : q . Timeout ,
MaxAttempts : q . MaxAttempts ,
Config : cfg ,
2020-10-15 22:40:03 +01:00
QueueLength : q . QueueLength ,
2020-07-05 20:38:03 +01:00
Name : name ,
2020-01-07 11:23:09 +00:00
} , exemplar )
}
if err != nil {
log . Error ( "Unable to create queue for %s: %v" , name , err )
return nil
}
2022-01-22 21:22:14 +00:00
// Sanity check configuration
if q . Workers == 0 && ( q . BoostTimeout == 0 || q . BoostWorkers == 0 || q . MaxWorkers == 0 ) {
log . Warn ( "Queue: %s is configured to be non-scaling and have no workers\n - this configuration is likely incorrect and could cause Gitea to block" , q . Name )
if pausable , ok := returnable . ( Pausable ) ; ok {
log . Warn ( "Queue: %s is being paused to prevent data-loss, add workers manually and unpause." , q . Name )
pausable . Pause ( )
}
}
2020-01-07 11:23:09 +00:00
return returnable
}
2020-02-02 23:19:58 +00:00
// CreateUniqueQueue for name with provided handler and exemplar
func CreateUniqueQueue ( name string , handle HandlerFunc , exemplar interface { } ) UniqueQueue {
q , cfg := getQueueSettings ( name )
if len ( cfg ) == 0 {
return nil
}
2021-09-03 11:20:57 +01:00
if len ( q . Type ) > 0 && q . Type != "dummy" && q . Type != "immediate" && ! strings . HasPrefix ( q . Type , "unique-" ) {
2020-02-02 23:19:58 +00:00
q . Type = "unique-" + q . Type
}
typ , err := validType ( q . Type )
if err != nil || typ == PersistableChannelQueueType {
typ = PersistableChannelUniqueQueueType
if err != nil {
log . Error ( "Invalid type %s provided for queue named %s defaulting to %s" , q . Type , name , string ( typ ) )
}
}
returnable , err := NewQueue ( typ , handle , cfg , exemplar )
if q . WrapIfNecessary && err != nil {
log . Warn ( "Unable to create unique queue for %s: %v" , name , err )
log . Warn ( "Attempting to create wrapped queue" )
returnable , err = NewQueue ( WrappedUniqueQueueType , handle , WrappedUniqueQueueConfiguration {
Underlying : typ ,
Timeout : q . Timeout ,
MaxAttempts : q . MaxAttempts ,
Config : cfg ,
2020-10-15 22:40:03 +01:00
QueueLength : q . QueueLength ,
2020-02-02 23:19:58 +00:00
} , exemplar )
}
if err != nil {
log . Error ( "Unable to create unique queue for %s: %v" , name , err )
return nil
}
2022-01-22 21:22:14 +00:00
// Sanity check configuration
if q . Workers == 0 && ( q . BoostTimeout == 0 || q . BoostWorkers == 0 || q . MaxWorkers == 0 ) {
log . Warn ( "Queue: %s is configured to be non-scaling and have no workers\n - this configuration is likely incorrect and could cause Gitea to block" , q . Name )
if pausable , ok := returnable . ( Pausable ) ; ok {
log . Warn ( "Queue: %s is being paused to prevent data-loss, add workers manually and unpause." , q . Name )
pausable . Pause ( )
}
}
2020-02-02 23:19:58 +00:00
return returnable . ( UniqueQueue )
}