-
Notifications
You must be signed in to change notification settings - Fork 2k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
clientv2: base driver plugin (#4671)
Driver plugin framework to facilitate development of driver plugins. Implementing plugins only need to implement the DriverPlugin interface. The framework proxies this interface to the go-plugin GRPC interface generated from the driver.proto spec. A testing harness is provided to allow implementing drivers to test the full lifecycle of the driver plugin. An example use: func TestMyDriver(t *testing.T) { harness := NewDriverHarness(t, &MyDiverPlugin{}) // The harness implements the DriverPlugin interface and can be used as such taskHandle, err := harness.StartTask(...) }
- Loading branch information
1 parent
092a9fa
commit a73555f
Showing
17 changed files
with
1,989 additions
and
468 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,302 @@ | ||
package drivers | ||
|
||
import ( | ||
"errors" | ||
"fmt" | ||
"io" | ||
"time" | ||
|
||
"github.com/golang/protobuf/ptypes" | ||
hclog "github.com/hashicorp/go-hclog" | ||
"github.com/hashicorp/nomad/plugins/base" | ||
"github.com/hashicorp/nomad/plugins/drivers/proto" | ||
"github.com/hashicorp/nomad/plugins/shared/hclspec" | ||
"golang.org/x/net/context" | ||
) | ||
|
||
var _ DriverPlugin = &driverPluginClient{} | ||
|
||
type driverPluginClient struct { | ||
base.BasePluginClient | ||
|
||
client proto.DriverClient | ||
logger hclog.Logger | ||
} | ||
|
||
func (d *driverPluginClient) TaskConfigSchema() (*hclspec.Spec, error) { | ||
req := &proto.TaskConfigSchemaRequest{} | ||
|
||
resp, err := d.client.TaskConfigSchema(context.Background(), req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
return resp.Spec, nil | ||
} | ||
|
||
func (d *driverPluginClient) Capabilities() (*Capabilities, error) { | ||
req := &proto.CapabilitiesRequest{} | ||
|
||
resp, err := d.client.Capabilities(context.Background(), req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
caps := &Capabilities{} | ||
if resp.Capabilities != nil { | ||
caps.SendSignals = resp.Capabilities.SendSignals | ||
caps.Exec = resp.Capabilities.Exec | ||
|
||
switch resp.Capabilities.FsIsolation { | ||
case proto.DriverCapabilities_NONE: | ||
caps.FSIsolation = FSIsolationNone | ||
case proto.DriverCapabilities_CHROOT: | ||
caps.FSIsolation = FSIsolationChroot | ||
case proto.DriverCapabilities_IMAGE: | ||
caps.FSIsolation = FSIsolationImage | ||
default: | ||
caps.FSIsolation = FSIsolationNone | ||
} | ||
} | ||
|
||
return caps, nil | ||
} | ||
|
||
// Fingerprint the driver, return a chan that will be pushed to periodically and on changes to health | ||
func (d *driverPluginClient) Fingerprint(ctx context.Context) (<-chan *Fingerprint, error) { | ||
req := &proto.FingerprintRequest{} | ||
|
||
stream, err := d.client.Fingerprint(context.Background(), req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
ch := make(chan *Fingerprint) | ||
go d.handleFingerprint(ch, stream) | ||
|
||
return ch, nil | ||
} | ||
|
||
func (d *driverPluginClient) handleFingerprint(ch chan *Fingerprint, stream proto.Driver_FingerprintClient) { | ||
defer close(ch) | ||
for { | ||
pb, err := stream.Recv() | ||
if err == io.EOF { | ||
break | ||
} | ||
if err != nil { | ||
d.logger.Error("error receiving stream from Fingerprint driver RPC", "error", err) | ||
ch <- &Fingerprint{Err: fmt.Errorf("error from RPC stream: %v", err)} | ||
break | ||
} | ||
f := &Fingerprint{ | ||
Attributes: pb.Attributes, | ||
Health: healthStateFromProto(pb.Health), | ||
HealthDescription: pb.HealthDescription, | ||
} | ||
ch <- f | ||
} | ||
|
||
} | ||
|
||
// RecoverTask does internal state recovery to be able to control the task of | ||
// the given TaskHandle | ||
func (d *driverPluginClient) RecoverTask(h *TaskHandle) error { | ||
req := &proto.RecoverTaskRequest{Handle: taskHandleToProto(h)} | ||
|
||
_, err := d.client.RecoverTask(context.Background(), req) | ||
return err | ||
} | ||
|
||
// StartTask starts execution of a task with the given TaskConfig. A TaskHandle | ||
// is returned to the caller that can be used to recover state of the task, | ||
// should the driver crash or exit prematurely. | ||
func (d *driverPluginClient) StartTask(c *TaskConfig) (*TaskHandle, error) { | ||
req := &proto.StartTaskRequest{ | ||
Task: taskConfigToProto(c), | ||
} | ||
|
||
resp, err := d.client.StartTask(context.Background(), req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
return taskHandleFromProto(resp.Handle), nil | ||
} | ||
|
||
// WaitTask returns a channel that will have an ExitResult pushed to it once when the task | ||
// exits on its own or is killed. If WaitTask is called after the task has exited, the channel | ||
// will immedialy return the ExitResult. WaitTask can be called multiple times for | ||
// the same task without issue. | ||
func (d *driverPluginClient) WaitTask(ctx context.Context, id string) (<-chan *ExitResult, error) { | ||
ch := make(chan *ExitResult) | ||
go d.handleWaitTask(ctx, id, ch) | ||
return ch, nil | ||
} | ||
|
||
func (d *driverPluginClient) handleWaitTask(ctx context.Context, id string, ch chan *ExitResult) { | ||
defer close(ch) | ||
var result ExitResult | ||
req := &proto.WaitTaskRequest{ | ||
TaskId: id, | ||
} | ||
|
||
resp, err := d.client.WaitTask(ctx, req) | ||
if err != nil { | ||
result.Err = err | ||
} else { | ||
result.ExitCode = int(resp.Result.ExitCode) | ||
result.Signal = int(resp.Result.Signal) | ||
result.OOMKilled = resp.Result.OomKilled | ||
if len(resp.Err) > 0 { | ||
result.Err = errors.New(resp.Err) | ||
} | ||
} | ||
ch <- &result | ||
} | ||
|
||
// StopTask stops the task with the given taskID. A timeout and signal can be | ||
// given to control a graceful termination of the task. The driver will send the | ||
// given signal to the task and wait for the given timeout for it to exit. If the | ||
// task does not exit within the timeout it will be forcefully killed. | ||
func (d *driverPluginClient) StopTask(taskID string, timeout time.Duration, signal string) error { | ||
req := &proto.StopTaskRequest{ | ||
TaskId: taskID, | ||
Timeout: ptypes.DurationProto(timeout), | ||
Signal: signal, | ||
} | ||
|
||
_, err := d.client.StopTask(context.Background(), req) | ||
return err | ||
} | ||
|
||
// DestroyTask removes the task from the driver's in memory state. The task | ||
// cannot be running unless force is set to true. If force is set to true the | ||
// driver will forcefully terminate the task before removing it. | ||
func (d *driverPluginClient) DestroyTask(taskID string, force bool) error { | ||
req := &proto.DestroyTaskRequest{ | ||
TaskId: taskID, | ||
Force: force, | ||
} | ||
|
||
_, err := d.client.DestroyTask(context.Background(), req) | ||
return err | ||
} | ||
|
||
// InspectTask returns status information for a task | ||
func (d *driverPluginClient) InspectTask(taskID string) (*TaskStatus, error) { | ||
req := &proto.InspectTaskRequest{TaskId: taskID} | ||
|
||
resp, err := d.client.InspectTask(context.Background(), req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
status, err := taskStatusFromProto(resp.Task) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
if resp.Driver != nil { | ||
status.DriverAttributes = resp.Driver.Attributes | ||
} | ||
if resp.NetworkOverride != nil { | ||
status.NetworkOverride = &NetworkOverride{ | ||
PortMap: resp.NetworkOverride.PortMap, | ||
Addr: resp.NetworkOverride.Addr, | ||
AutoAdvertise: resp.NetworkOverride.AutoAdvertise, | ||
} | ||
} | ||
|
||
return status, nil | ||
} | ||
|
||
// TaskStats returns resource usage statistics for the task | ||
func (d *driverPluginClient) TaskStats(taskID string) (*TaskStats, error) { | ||
req := &proto.TaskStatsRequest{TaskId: taskID} | ||
|
||
resp, err := d.client.TaskStats(context.Background(), req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
stats, err := taskStatsFromProto(resp.Stats) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
return stats, nil | ||
} | ||
|
||
// TaskEvents returns a channel that will receive events from the driver about all | ||
// tasks such as lifecycle events, terminal errors, etc. | ||
func (d *driverPluginClient) TaskEvents(ctx context.Context) (<-chan *TaskEvent, error) { | ||
req := &proto.TaskEventsRequest{} | ||
stream, err := d.client.TaskEvents(ctx, req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
ch := make(chan *TaskEvent) | ||
go d.handleTaskEvents(ch, stream) | ||
return ch, nil | ||
} | ||
|
||
func (d *driverPluginClient) handleTaskEvents(ch chan *TaskEvent, stream proto.Driver_TaskEventsClient) { | ||
defer close(ch) | ||
for { | ||
ev, err := stream.Recv() | ||
if err == io.EOF { | ||
break | ||
} | ||
if err != nil { | ||
d.logger.Error("error receiving stream from TaskEvents driver RPC", "error", err) | ||
ch <- &TaskEvent{Err: err} | ||
break | ||
} | ||
timestamp, _ := ptypes.Timestamp(ev.Timestamp) | ||
event := &TaskEvent{ | ||
TaskID: ev.TaskId, | ||
Annotations: ev.Annotations, | ||
Message: ev.Message, | ||
Timestamp: timestamp, | ||
} | ||
ch <- event | ||
} | ||
} | ||
|
||
// SignalTask will send the given signal to the specified task | ||
func (d *driverPluginClient) SignalTask(taskID string, signal string) error { | ||
req := &proto.SignalTaskRequest{ | ||
TaskId: taskID, | ||
Signal: signal, | ||
} | ||
_, err := d.client.SignalTask(context.Background(), req) | ||
return err | ||
} | ||
|
||
// ExecTask will run the given command within the execution context of the task. | ||
// The driver will wait for the given timeout for the command to complete before | ||
// terminating it. The stdout and stderr of the command will be return to the caller, | ||
// along with other exit information such as exit code. | ||
func (d *driverPluginClient) ExecTask(taskID string, cmd []string, timeout time.Duration) (*ExecTaskResult, error) { | ||
req := &proto.ExecTaskRequest{ | ||
TaskId: taskID, | ||
Command: cmd, | ||
Timeout: ptypes.DurationProto(timeout), | ||
} | ||
|
||
resp, err := d.client.ExecTask(context.Background(), req) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
result := &ExecTaskResult{ | ||
Stdout: resp.Stdout, | ||
Stderr: resp.Stderr, | ||
ExitResult: exitResultFromProto(resp.Result), | ||
} | ||
|
||
return result, nil | ||
|
||
} |
Oops, something went wrong.