mirror of
https://github.com/tailscale/tailscale.git
synced 2025-04-03 14:55:47 +00:00

Expose enough metrics to get a sense of queue depth, use and if it has stalled. Updates tailscale/corp#26058 Signed-off-by: Andrew Dunham <andrew@du.nham.ca> Change-Id: I271ac8d03f3db587a33aca6964fe92f2833e1251
145 lines
2.8 KiB
Go
145 lines
2.8 KiB
Go
// Copyright (c) Tailscale Inc & AUTHORS
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
// Package execqueue implements an ordered asynchronous queue for executing functions.
|
|
package execqueue
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"expvar"
|
|
"fmt"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
)
|
|
|
|
type ExecQueue struct {
|
|
mu sync.Mutex
|
|
closed bool
|
|
inFlight bool // whether a goroutine is running q.run
|
|
doneWaiter chan struct{} // non-nil if waiter is waiting, then closed
|
|
queue []func()
|
|
|
|
// metrics follow
|
|
metricsRegisterOnce sync.Once
|
|
metricInserts expvar.Int
|
|
metricRemovals expvar.Int
|
|
metricQueueLastDrain expvar.Int // unix millis
|
|
}
|
|
|
|
// This is extremely silly but is for debugging
|
|
var metricsCounter atomic.Int64
|
|
|
|
// registerMetrics registers the queue's metrics with expvar, using a unique name.
|
|
func (q *ExecQueue) registerMetrics() {
|
|
q.metricsRegisterOnce.Do(func() {
|
|
m := new(expvar.Map).Init()
|
|
m.Set("inserts", &q.metricInserts)
|
|
m.Set("removals", &q.metricRemovals)
|
|
m.Set("length", expvar.Func(func() any {
|
|
return q.metricInserts.Value() - q.metricRemovals.Value()
|
|
}))
|
|
m.Set("last_drain", &q.metricQueueLastDrain)
|
|
|
|
name := fmt.Sprintf("execqueue-%d", metricsCounter.Add(1))
|
|
expvar.Publish(name, m)
|
|
})
|
|
}
|
|
|
|
func (q *ExecQueue) Add(f func()) {
|
|
q.registerMetrics()
|
|
|
|
q.mu.Lock()
|
|
defer q.mu.Unlock()
|
|
if q.closed {
|
|
return
|
|
}
|
|
if q.inFlight {
|
|
q.queue = append(q.queue, f)
|
|
q.metricInserts.Add(1)
|
|
} else {
|
|
q.inFlight = true
|
|
go q.run(f)
|
|
}
|
|
}
|
|
|
|
// RunSync waits for the queue to be drained and then synchronously runs f.
|
|
// It returns an error if the queue is closed before f is run or ctx expires.
|
|
func (q *ExecQueue) RunSync(ctx context.Context, f func()) error {
|
|
q.registerMetrics()
|
|
|
|
for {
|
|
if err := q.Wait(ctx); err != nil {
|
|
return err
|
|
}
|
|
q.mu.Lock()
|
|
if q.inFlight {
|
|
q.mu.Unlock()
|
|
continue
|
|
}
|
|
defer q.mu.Unlock()
|
|
if q.closed {
|
|
return errors.New("closed")
|
|
}
|
|
f()
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (q *ExecQueue) run(f func()) {
|
|
f()
|
|
|
|
q.mu.Lock()
|
|
for len(q.queue) > 0 && !q.closed {
|
|
f := q.queue[0]
|
|
q.queue[0] = nil
|
|
q.queue = q.queue[1:]
|
|
q.metricRemovals.Add(1)
|
|
q.mu.Unlock()
|
|
f()
|
|
q.mu.Lock()
|
|
}
|
|
q.inFlight = false
|
|
q.metricQueueLastDrain.Set(int64(time.Now().UnixMilli()))
|
|
q.queue = nil
|
|
if q.doneWaiter != nil {
|
|
close(q.doneWaiter)
|
|
q.doneWaiter = nil
|
|
}
|
|
q.mu.Unlock()
|
|
}
|
|
|
|
// Shutdown asynchronously signals the queue to stop.
|
|
func (q *ExecQueue) Shutdown() {
|
|
q.registerMetrics()
|
|
|
|
q.mu.Lock()
|
|
defer q.mu.Unlock()
|
|
q.closed = true
|
|
}
|
|
|
|
// Wait waits for the queue to be empty.
|
|
func (q *ExecQueue) Wait(ctx context.Context) error {
|
|
q.registerMetrics()
|
|
|
|
q.mu.Lock()
|
|
waitCh := q.doneWaiter
|
|
if q.inFlight && waitCh == nil {
|
|
waitCh = make(chan struct{})
|
|
q.doneWaiter = waitCh
|
|
}
|
|
q.mu.Unlock()
|
|
|
|
if waitCh == nil {
|
|
return nil
|
|
}
|
|
|
|
select {
|
|
case <-waitCh:
|
|
return nil
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
}
|
|
}
|