2015-01-10 04:02:55 +00:00
|
|
|
package influxdb
|
|
|
|
|
|
|
|
import (
|
2015-04-08 18:25:04 +00:00
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2015-01-10 04:02:55 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/influxdb/influxdb/messaging"
|
|
|
|
)
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
const (
|
|
|
|
// DefaultContinuousQueryCheckTime is how frequently the broker will ask a data node
|
|
|
|
// in the cluster to run any continuous queries that should be run.
|
|
|
|
DefaultContinuousQueryCheckTime = 1 * time.Second
|
|
|
|
|
|
|
|
// DefaultDataNodeTimeout is how long the broker will wait before timing out on a data node
|
|
|
|
// that it has requested process continuous queries.
|
|
|
|
DefaultDataNodeTimeout = 1 * time.Second
|
|
|
|
|
|
|
|
// DefaultFailureSleep is how long the broker will sleep before trying the next data node in
|
|
|
|
// the cluster if the current data node failed to respond
|
|
|
|
DefaultFailureSleep = 100 * time.Millisecond
|
|
|
|
)
|
|
|
|
|
2015-01-10 04:02:55 +00:00
|
|
|
// Broker represents an InfluxDB specific messaging broker.
|
|
|
|
type Broker struct {
|
|
|
|
*messaging.Broker
|
|
|
|
|
2015-02-12 21:38:33 +00:00
|
|
|
done chan struct{}
|
2015-01-24 20:00:51 +00:00
|
|
|
|
|
|
|
// send CQ processing requests to the same data node
|
2015-04-08 18:25:04 +00:00
|
|
|
currentCQProcessingNode *url.URL
|
2015-01-24 20:00:51 +00:00
|
|
|
|
|
|
|
// variables to control when to trigger processing and when to timeout
|
|
|
|
TriggerInterval time.Duration
|
|
|
|
TriggerTimeout time.Duration
|
|
|
|
TriggerFailurePause time.Duration
|
2015-01-10 04:02:55 +00:00
|
|
|
}
|
|
|
|
|
2015-01-20 02:44:47 +00:00
|
|
|
// NewBroker returns a new instance of a Broker with default values.
|
|
|
|
func NewBroker() *Broker {
|
2015-03-09 21:47:41 +00:00
|
|
|
return &Broker{
|
|
|
|
Broker: messaging.NewBroker(),
|
2015-01-25 21:41:39 +00:00
|
|
|
TriggerInterval: 5 * time.Second,
|
|
|
|
TriggerTimeout: 20 * time.Second,
|
|
|
|
TriggerFailurePause: 1 * time.Second,
|
2015-01-24 20:00:51 +00:00
|
|
|
}
|
2015-01-20 02:44:47 +00:00
|
|
|
}
|
2015-01-10 04:02:55 +00:00
|
|
|
|
2015-02-16 20:30:58 +00:00
|
|
|
// RunContinuousQueryLoop starts running continuous queries on a background goroutine.
|
2015-02-12 21:38:33 +00:00
|
|
|
func (b *Broker) RunContinuousQueryLoop() {
|
2015-04-08 18:25:04 +00:00
|
|
|
b.done = make(chan struct{})
|
|
|
|
go b.continuousQueryLoop(b.done)
|
2015-01-10 04:02:55 +00:00
|
|
|
}
|
|
|
|
|
2015-02-16 20:30:58 +00:00
|
|
|
// Close closes the broker.
|
2015-01-10 04:02:55 +00:00
|
|
|
func (b *Broker) Close() error {
|
|
|
|
if b.done != nil {
|
|
|
|
close(b.done)
|
|
|
|
b.done = nil
|
|
|
|
}
|
|
|
|
return b.Broker.Close()
|
|
|
|
}
|
|
|
|
|
2015-02-12 00:57:50 +00:00
|
|
|
func (b *Broker) continuousQueryLoop(done chan struct{}) {
|
2015-01-10 04:02:55 +00:00
|
|
|
for {
|
|
|
|
// Check if broker is currently leader.
|
|
|
|
if b.Broker.IsLeader() {
|
2015-01-24 20:00:51 +00:00
|
|
|
b.runContinuousQueries()
|
2015-01-10 04:02:55 +00:00
|
|
|
}
|
|
|
|
|
2015-02-12 00:57:50 +00:00
|
|
|
// Sleep until either the broker is closed or we need to run continuous queries again
|
2015-01-10 04:02:55 +00:00
|
|
|
select {
|
2015-02-12 00:57:50 +00:00
|
|
|
case <-done:
|
2015-01-10 04:02:55 +00:00
|
|
|
return
|
2015-02-12 00:57:50 +00:00
|
|
|
case <-time.After(DefaultContinuousQueryCheckTime):
|
2015-01-10 04:02:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-20 02:44:47 +00:00
|
|
|
|
|
|
|
func (b *Broker) runContinuousQueries() {
|
2015-01-24 20:00:51 +00:00
|
|
|
next := 0
|
|
|
|
for {
|
|
|
|
// if the current node hasn't been set it's our first time or we're reset. move to the next one
|
|
|
|
if b.currentCQProcessingNode == nil {
|
2015-04-08 18:25:04 +00:00
|
|
|
topic := b.Broker.Topic(BroadcastTopicID)
|
|
|
|
if topic == nil {
|
|
|
|
log.Println("broker cq: no topics currently available.")
|
2015-04-08 20:00:18 +00:00
|
|
|
return // don't have any topics to get data urls from, give it up
|
2015-01-24 20:00:51 +00:00
|
|
|
}
|
2015-04-08 18:25:04 +00:00
|
|
|
dataURLs := topic.DataURLs()
|
|
|
|
if len(dataURLs) == 0 {
|
|
|
|
log.Println("broker cq: no data nodes currently available.")
|
2015-04-08 20:00:18 +00:00
|
|
|
return // don't have any data urls to try, give it up
|
2015-04-08 18:25:04 +00:00
|
|
|
}
|
|
|
|
next = next % len(dataURLs)
|
|
|
|
u := dataURLs[next]
|
|
|
|
b.currentCQProcessingNode = &u
|
2015-01-24 20:00:51 +00:00
|
|
|
next++
|
|
|
|
}
|
|
|
|
|
|
|
|
// if no error, we're all good
|
|
|
|
err := b.requestContinuousQueryProcessing()
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
2015-04-08 18:25:04 +00:00
|
|
|
log.Printf("broker cq: error hitting data node: %s: %s\n", b.currentCQProcessingNode, err.Error())
|
2015-01-24 20:00:51 +00:00
|
|
|
|
|
|
|
// reset and let the loop try the next data node in the cluster
|
|
|
|
b.currentCQProcessingNode = nil
|
2015-02-12 00:57:50 +00:00
|
|
|
<-time.After(DefaultFailureSleep)
|
2015-01-24 20:00:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Broker) requestContinuousQueryProcessing() error {
|
|
|
|
// Send request.
|
2015-04-08 18:25:04 +00:00
|
|
|
cqURL := copyURL(b.currentCQProcessingNode)
|
2015-02-12 00:57:50 +00:00
|
|
|
cqURL.Path = "/process_continuous_queries"
|
|
|
|
cqURL.Scheme = "http"
|
2015-01-24 20:00:51 +00:00
|
|
|
client := &http.Client{
|
2015-02-12 00:57:50 +00:00
|
|
|
Timeout: DefaultDataNodeTimeout,
|
2015-01-24 20:00:51 +00:00
|
|
|
}
|
2015-02-12 00:57:50 +00:00
|
|
|
resp, err := client.Post(cqURL.String(), "application/octet-stream", nil)
|
2015-01-24 20:00:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
// Check if created.
|
|
|
|
if resp.StatusCode != http.StatusAccepted {
|
|
|
|
return fmt.Errorf("request returned status %s", resp.Status)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2015-01-20 02:44:47 +00:00
|
|
|
}
|