2019-10-15 19:14:25 +00:00
|
|
|
package nomad
|
|
|
|
|
|
|
|
import (
|
2019-11-01 14:33:28 +00:00
|
|
|
"bytes"
|
2019-10-15 19:14:25 +00:00
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net"
|
2019-11-01 14:33:28 +00:00
|
|
|
"time"
|
2019-10-15 19:14:25 +00:00
|
|
|
|
|
|
|
log "github.com/hashicorp/go-hclog"
|
2019-11-01 14:33:28 +00:00
|
|
|
sframer "github.com/hashicorp/nomad/client/lib/streamframer"
|
2019-10-15 19:14:25 +00:00
|
|
|
cstructs "github.com/hashicorp/nomad/client/structs"
|
|
|
|
"github.com/hashicorp/nomad/command/agent/monitor"
|
|
|
|
"github.com/hashicorp/nomad/helper"
|
|
|
|
"github.com/hashicorp/nomad/nomad/structs"
|
|
|
|
|
|
|
|
"github.com/ugorji/go/codec"
|
|
|
|
)
|
|
|
|
|
2019-10-30 13:28:24 +00:00
|
|
|
type Agent struct {
|
2019-10-15 19:14:25 +00:00
|
|
|
srv *Server
|
|
|
|
}
|
|
|
|
|
2019-10-30 13:28:24 +00:00
|
|
|
func (m *Agent) register() {
|
2019-10-15 19:14:25 +00:00
|
|
|
m.srv.streamingRpcs.Register("Agent.Monitor", m.monitor)
|
|
|
|
}
|
|
|
|
|
2019-10-30 13:28:24 +00:00
|
|
|
func (m *Agent) monitor(conn io.ReadWriteCloser) {
|
2019-10-15 19:14:25 +00:00
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
// Decode args
|
|
|
|
var args cstructs.MonitorRequest
|
|
|
|
decoder := codec.NewDecoder(conn, structs.MsgpackHandle)
|
|
|
|
encoder := codec.NewEncoder(conn, structs.MsgpackHandle)
|
|
|
|
|
|
|
|
if err := decoder.Decode(&args); err != nil {
|
|
|
|
handleStreamResultError(err, helper.Int64ToPtr(500), encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-11-04 14:04:47 +00:00
|
|
|
// Check agent read permissions
|
2019-10-15 19:14:25 +00:00
|
|
|
if aclObj, err := m.srv.ResolveToken(args.AuthToken); err != nil {
|
|
|
|
handleStreamResultError(err, nil, encoder)
|
|
|
|
return
|
2019-10-25 18:25:19 +00:00
|
|
|
} else if aclObj != nil && !aclObj.AllowAgentRead() {
|
|
|
|
handleStreamResultError(structs.ErrPermissionDenied, helper.Int64ToPtr(403), encoder)
|
2019-10-15 19:14:25 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-30 13:36:39 +00:00
|
|
|
logLevel := log.LevelFromString(args.LogLevel)
|
2019-10-15 19:14:25 +00:00
|
|
|
if args.LogLevel == "" {
|
|
|
|
logLevel = log.LevelFromString("INFO")
|
|
|
|
}
|
|
|
|
|
|
|
|
if logLevel == log.NoLevel {
|
|
|
|
handleStreamResultError(errors.New("Unknown log level"), helper.Int64ToPtr(400), encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-11-07 13:33:53 +00:00
|
|
|
// Targeting a node, forward request to node
|
2019-10-15 19:14:25 +00:00
|
|
|
if args.NodeID != "" {
|
2019-11-04 14:25:19 +00:00
|
|
|
m.forwardMonitor(conn, args, encoder, decoder)
|
2019-11-07 13:33:53 +00:00
|
|
|
// forwarded request has ended, return
|
|
|
|
return
|
2019-10-15 19:14:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NodeID was empty, so monitor this current server
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
monitor := monitor.New(512, m.srv.logger, &log.LoggerOptions{
|
|
|
|
Level: logLevel,
|
2019-10-24 16:47:46 +00:00
|
|
|
JSONFormat: args.LogJSON,
|
2019-10-15 19:14:25 +00:00
|
|
|
})
|
|
|
|
|
2019-11-01 14:33:28 +00:00
|
|
|
frames := make(chan *sframer.StreamFrame, 32)
|
|
|
|
errCh := make(chan error)
|
|
|
|
var buf bytes.Buffer
|
|
|
|
frameCodec := codec.NewEncoder(&buf, structs.JsonHandle)
|
|
|
|
|
|
|
|
framer := sframer.NewStreamFramer(frames, 1*time.Second, 200*time.Millisecond, 1024)
|
|
|
|
framer.Run()
|
|
|
|
defer framer.Destroy()
|
|
|
|
|
|
|
|
// goroutine to detect remote side closing
|
2019-10-15 19:14:25 +00:00
|
|
|
go func() {
|
|
|
|
if _, err := conn.Read(nil); err != nil {
|
2019-11-01 14:33:28 +00:00
|
|
|
// One end of the pipe explicitly closed, exit
|
2019-10-15 19:14:25 +00:00
|
|
|
cancel()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2019-11-05 14:16:51 +00:00
|
|
|
logCh := monitor.Start()
|
|
|
|
defer monitor.Stop()
|
2019-11-01 14:33:28 +00:00
|
|
|
initialOffset := int64(0)
|
|
|
|
|
|
|
|
// receive logs and build frames
|
|
|
|
go func() {
|
|
|
|
defer framer.Destroy()
|
|
|
|
LOOP:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case log := <-logCh:
|
|
|
|
if err := framer.Send("", "log", log, initialOffset); err != nil {
|
|
|
|
select {
|
|
|
|
case errCh <- err:
|
|
|
|
case <-ctx.Done():
|
|
|
|
}
|
|
|
|
break LOOP
|
|
|
|
}
|
|
|
|
case <-ctx.Done():
|
|
|
|
break LOOP
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
2019-10-15 19:14:25 +00:00
|
|
|
|
|
|
|
var streamErr error
|
|
|
|
OUTER:
|
|
|
|
for {
|
|
|
|
select {
|
2019-11-01 14:33:28 +00:00
|
|
|
case frame, ok := <-frames:
|
|
|
|
if !ok {
|
|
|
|
// frame may have been closed when an error
|
|
|
|
// occurred. Check once more for an error.
|
|
|
|
select {
|
|
|
|
case streamErr = <-errCh:
|
|
|
|
// There was a pending error!
|
|
|
|
default:
|
|
|
|
// No error, continue on
|
|
|
|
}
|
|
|
|
|
|
|
|
break OUTER
|
|
|
|
}
|
|
|
|
|
2019-10-15 19:14:25 +00:00
|
|
|
var resp cstructs.StreamErrWrapper
|
2019-11-01 14:33:28 +00:00
|
|
|
if args.PlainText {
|
|
|
|
resp.Payload = frame.Data
|
|
|
|
} else {
|
|
|
|
if err := frameCodec.Encode(frame); err != nil {
|
|
|
|
streamErr = err
|
|
|
|
break OUTER
|
|
|
|
}
|
|
|
|
|
|
|
|
resp.Payload = buf.Bytes()
|
|
|
|
buf.Reset()
|
|
|
|
}
|
|
|
|
|
2019-10-15 19:14:25 +00:00
|
|
|
if err := encoder.Encode(resp); err != nil {
|
|
|
|
streamErr = err
|
|
|
|
break OUTER
|
|
|
|
}
|
|
|
|
encoder.Reset(conn)
|
|
|
|
case <-ctx.Done():
|
|
|
|
break OUTER
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if streamErr != nil {
|
2019-11-04 19:32:53 +00:00
|
|
|
handleStreamResultError(streamErr, helper.Int64ToPtr(500), encoder)
|
|
|
|
return
|
2019-10-15 19:14:25 +00:00
|
|
|
}
|
|
|
|
}
|
2019-11-04 14:25:19 +00:00
|
|
|
|
|
|
|
func (m *Agent) forwardMonitor(conn io.ReadWriteCloser, args cstructs.MonitorRequest, encoder *codec.Encoder, decoder *codec.Decoder) {
|
|
|
|
nodeID := args.NodeID
|
|
|
|
|
|
|
|
snap, err := m.srv.State().Snapshot()
|
|
|
|
if err != nil {
|
|
|
|
handleStreamResultError(err, nil, encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
node, err := snap.NodeByID(nil, nodeID)
|
|
|
|
if err != nil {
|
|
|
|
handleStreamResultError(err, helper.Int64ToPtr(500), encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if node == nil {
|
|
|
|
err := fmt.Errorf("Unknown node %q", nodeID)
|
|
|
|
handleStreamResultError(err, helper.Int64ToPtr(400), encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := nodeSupportsRpc(node); err != nil {
|
|
|
|
handleStreamResultError(err, helper.Int64ToPtr(400), encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the Connection to the client either by fowarding to another server
|
|
|
|
// or creating direct stream
|
|
|
|
var clientConn net.Conn
|
|
|
|
state, ok := m.srv.getNodeConn(nodeID)
|
|
|
|
if !ok {
|
|
|
|
// Determine the server that has a connection to the node
|
|
|
|
srv, err := m.srv.serverWithNodeConn(nodeID, m.srv.Region())
|
|
|
|
if err != nil {
|
|
|
|
var code *int64
|
|
|
|
if structs.IsErrNoNodeConn(err) {
|
|
|
|
code = helper.Int64ToPtr(404)
|
|
|
|
}
|
|
|
|
handleStreamResultError(err, code, encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
conn, err := m.srv.streamingRpc(srv, "Agent.Monitor")
|
|
|
|
if err != nil {
|
|
|
|
handleStreamResultError(err, nil, encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
clientConn = conn
|
|
|
|
} else {
|
|
|
|
stream, err := NodeStreamingRpc(state.Session, "Agent.Monitor")
|
|
|
|
if err != nil {
|
|
|
|
handleStreamResultError(err, nil, encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
clientConn = stream
|
|
|
|
}
|
|
|
|
defer clientConn.Close()
|
|
|
|
|
|
|
|
// Send the Request
|
|
|
|
outEncoder := codec.NewEncoder(clientConn, structs.MsgpackHandle)
|
|
|
|
if err := outEncoder.Encode(args); err != nil {
|
|
|
|
handleStreamResultError(err, nil, encoder)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
structs.Bridge(conn, clientConn)
|
|
|
|
return
|
|
|
|
}
|