mirror of
https://github.com/fnproject/fn.git
synced 2022-10-28 21:29:17 +03:00
Inverting deps on SQL, Log and MQ plugins to make them optional dependencies of extended servers, Removing some dead code that brought in unused dependencies Filtering out some non-linux transitive deps. (#1057)
* initial Db helper split - make SQL and datastore packages optional * abstracting log store * break out DB, MQ and log drivers as extensions * cleanup * fewer deps * fixing docker test * hmm dbness * updating db startup * Consolidate all your extensions into one convenient package * cleanup * clean up dep constraints
This commit is contained in:
208
api/mqs/memory/memory.go
Normal file
208
api/mqs/memory/memory.go
Normal file
@@ -0,0 +1,208 @@
|
||||
package memory
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/fnproject/fn/api/common"
|
||||
"github.com/fnproject/fn/api/models"
|
||||
"github.com/fnproject/fn/api/mqs"
|
||||
"github.com/google/btree"
|
||||
"github.com/sirupsen/logrus"
|
||||
"net/url"
|
||||
)
|
||||
|
||||
type memoryProvider int
|
||||
|
||||
type MemoryMQ struct {
|
||||
// WorkQueue A buffered channel that we can send work requests on.
|
||||
PriorityQueues []chan *models.Call
|
||||
Ticker *time.Ticker
|
||||
BTree *btree.BTree
|
||||
Timeouts map[string]*callItem
|
||||
// Protects B-tree and Timeouts
|
||||
// If this becomes a bottleneck, consider separating the two mutexes. The
|
||||
// goroutine to clear up timed out messages could also become a bottleneck at
|
||||
// some point. May need to switch to bucketing of some sort.
|
||||
Mutex sync.Mutex
|
||||
}
|
||||
|
||||
const NumPriorities = 3
|
||||
|
||||
func (memoryProvider) Supports(url *url.URL) bool {
|
||||
switch url.Scheme {
|
||||
case "memory":
|
||||
return true
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func (memoryProvider) New(url *url.URL) (models.MessageQueue, error) {
|
||||
var queues []chan *models.Call
|
||||
for i := 0; i < NumPriorities; i++ {
|
||||
queues = append(queues, make(chan *models.Call, 5000))
|
||||
}
|
||||
ticker := time.NewTicker(time.Second)
|
||||
mq := &MemoryMQ{
|
||||
PriorityQueues: queues,
|
||||
Ticker: ticker,
|
||||
BTree: btree.New(2),
|
||||
Timeouts: make(map[string]*callItem, 0),
|
||||
}
|
||||
mq.start()
|
||||
logrus.Info("MemoryMQ initialized")
|
||||
return mq, nil
|
||||
}
|
||||
|
||||
func (memoryProvider) String() string {
|
||||
return "memory"
|
||||
}
|
||||
|
||||
func (mq *MemoryMQ) start() {
|
||||
// start goroutine to check for delayed jobs and put them onto regular queue when ready
|
||||
go func() {
|
||||
for range mq.Ticker.C {
|
||||
ji := &callItem{
|
||||
StartAt: time.Now(),
|
||||
}
|
||||
mq.Mutex.Lock()
|
||||
mq.BTree.AscendLessThan(ji, func(a btree.Item) bool {
|
||||
logrus.WithFields(logrus.Fields{"queue": a}).Debug("delayed job move to queue")
|
||||
ji2 := mq.BTree.Delete(a).(*callItem)
|
||||
// put it onto the regular queue now
|
||||
_, err := mq.pushForce(ji2.Call)
|
||||
if err != nil {
|
||||
logrus.WithError(err).Error("Couldn't push delayed message onto main queue")
|
||||
}
|
||||
return true
|
||||
})
|
||||
mq.Mutex.Unlock()
|
||||
}
|
||||
}()
|
||||
// start goroutine to check for messages that have timed out and put them back onto regular queue
|
||||
// TODO: this should be like the delayed messages above. Could even be the same thing as delayed messages, but remove them if job is completed.
|
||||
go func() {
|
||||
for range mq.Ticker.C {
|
||||
ji := &callItem{
|
||||
StartAt: time.Now(),
|
||||
}
|
||||
mq.Mutex.Lock()
|
||||
for _, jobItem := range mq.Timeouts {
|
||||
if jobItem.Less(ji) {
|
||||
delete(mq.Timeouts, jobItem.Call.ID)
|
||||
_, err := mq.pushForce(jobItem.Call)
|
||||
if err != nil {
|
||||
logrus.WithError(err).Error("Couldn't push timed out message onto main queue")
|
||||
}
|
||||
}
|
||||
}
|
||||
mq.Mutex.Unlock()
|
||||
}
|
||||
}()
|
||||
}
|
||||
|
||||
// callItem is for the Btree, implements btree.Item
|
||||
type callItem struct {
|
||||
Call *models.Call
|
||||
StartAt time.Time
|
||||
}
|
||||
|
||||
func (ji *callItem) Less(than btree.Item) bool {
|
||||
// TODO: this could lose jobs: https://godoc.org/github.com/google/btree#Item
|
||||
ji2 := than.(*callItem)
|
||||
return ji.StartAt.Before(ji2.StartAt)
|
||||
}
|
||||
|
||||
func (mq *MemoryMQ) Push(ctx context.Context, job *models.Call) (*models.Call, error) {
|
||||
_, log := common.LoggerWithFields(ctx, logrus.Fields{"call_id": job.ID})
|
||||
log.Debugln("Pushed to MQ")
|
||||
|
||||
// It seems to me that using the job ID in the reservation is acceptable since each job can only have one outstanding reservation.
|
||||
// job.MsgId = randSeq(20)
|
||||
if job.Delay > 0 {
|
||||
// then we'll put into short term storage until ready
|
||||
ji := &callItem{
|
||||
Call: job,
|
||||
StartAt: time.Now().Add(time.Second * time.Duration(job.Delay)),
|
||||
}
|
||||
mq.Mutex.Lock()
|
||||
replaced := mq.BTree.ReplaceOrInsert(ji)
|
||||
mq.Mutex.Unlock()
|
||||
if replaced != nil {
|
||||
log.Warn("Ooops! an item was replaced and therefore lost, not good.")
|
||||
}
|
||||
return job, nil
|
||||
}
|
||||
|
||||
// Push the work onto the queue.
|
||||
return mq.pushForce(job)
|
||||
}
|
||||
|
||||
func (mq *MemoryMQ) pushTimeout(job *models.Call) error {
|
||||
|
||||
ji := &callItem{
|
||||
Call: job,
|
||||
StartAt: time.Now().Add(time.Minute),
|
||||
}
|
||||
mq.Mutex.Lock()
|
||||
mq.Timeouts[job.ID] = ji
|
||||
mq.Mutex.Unlock()
|
||||
return nil
|
||||
}
|
||||
|
||||
func (mq *MemoryMQ) pushForce(job *models.Call) (*models.Call, error) {
|
||||
mq.PriorityQueues[*job.Priority] <- job
|
||||
return job, nil
|
||||
}
|
||||
|
||||
// This is recursive, so be careful how many channels you pass in.
|
||||
func pickEarliestNonblocking(channels ...chan *models.Call) *models.Call {
|
||||
if len(channels) == 0 {
|
||||
return nil
|
||||
}
|
||||
|
||||
select {
|
||||
case job := <-channels[0]:
|
||||
return job
|
||||
default:
|
||||
return pickEarliestNonblocking(channels[1:]...)
|
||||
}
|
||||
}
|
||||
|
||||
func (mq *MemoryMQ) Reserve(ctx context.Context) (*models.Call, error) {
|
||||
job := pickEarliestNonblocking(mq.PriorityQueues[2], mq.PriorityQueues[1], mq.PriorityQueues[0])
|
||||
if job == nil {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
_, log := common.LoggerWithFields(ctx, logrus.Fields{"call_id": job.ID})
|
||||
log.Debugln("Reserved")
|
||||
return job, mq.pushTimeout(job)
|
||||
}
|
||||
|
||||
func (mq *MemoryMQ) Delete(ctx context.Context, job *models.Call) error {
|
||||
_, log := common.LoggerWithFields(ctx, logrus.Fields{"call_id": job.ID})
|
||||
|
||||
mq.Mutex.Lock()
|
||||
defer mq.Mutex.Unlock()
|
||||
_, exists := mq.Timeouts[job.ID]
|
||||
if !exists {
|
||||
return errors.New("Not reserved")
|
||||
}
|
||||
|
||||
delete(mq.Timeouts, job.ID)
|
||||
log.Debugln("Deleted")
|
||||
return nil
|
||||
}
|
||||
|
||||
// Close stops the associated goroutines by stopping the ticker
|
||||
func (mq *MemoryMQ) Close() error {
|
||||
mq.Ticker.Stop()
|
||||
return nil
|
||||
}
|
||||
|
||||
func init() {
|
||||
mqs.AddProvider(memoryProvider(0))
|
||||
}
|
||||
Reference in New Issue
Block a user