2018-08-08 16:02:24 -05:00
|
|
|
package plugin
|
|
|
|
|
|
|
|
import (
|
|
|
|
"log"
|
2019-11-06 13:27:28 -06:00
|
|
|
"strings"
|
|
|
|
"unicode/utf8"
|
2018-08-08 16:02:24 -05:00
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/helper/schema"
|
2018-11-19 11:39:16 -06:00
|
|
|
proto "github.com/hashicorp/terraform/internal/tfplugin5"
|
2018-08-15 11:03:39 -05:00
|
|
|
"github.com/hashicorp/terraform/plugin/convert"
|
2018-08-08 16:02:24 -05:00
|
|
|
"github.com/hashicorp/terraform/terraform"
|
|
|
|
"github.com/zclconf/go-cty/cty"
|
2018-08-15 11:03:39 -05:00
|
|
|
ctyconvert "github.com/zclconf/go-cty/cty/convert"
|
2018-08-08 16:02:24 -05:00
|
|
|
"github.com/zclconf/go-cty/cty/msgpack"
|
|
|
|
context "golang.org/x/net/context"
|
|
|
|
)
|
|
|
|
|
2018-08-14 15:39:30 -05:00
|
|
|
// NewGRPCProvisionerServerShim wraps a terraform.ResourceProvisioner in a
|
|
|
|
// proto.ProvisionerServer implementation. If the provided provisioner is not a
|
|
|
|
// *schema.Provisioner, this will return nil,
|
|
|
|
func NewGRPCProvisionerServerShim(p terraform.ResourceProvisioner) *GRPCProvisionerServer {
|
|
|
|
sp, ok := p.(*schema.Provisioner)
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return &GRPCProvisionerServer{
|
|
|
|
provisioner: sp,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-08 16:02:24 -05:00
|
|
|
type GRPCProvisionerServer struct {
|
|
|
|
provisioner *schema.Provisioner
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *GRPCProvisionerServer) GetSchema(_ context.Context, req *proto.GetProvisionerSchema_Request) (*proto.GetProvisionerSchema_Response, error) {
|
|
|
|
resp := &proto.GetProvisionerSchema_Response{}
|
|
|
|
|
|
|
|
resp.Provisioner = &proto.Schema{
|
2018-08-15 11:03:39 -05:00
|
|
|
Block: convert.ConfigSchemaToProto(schema.InternalMap(s.provisioner.Schema).CoreConfigSchema()),
|
2018-08-08 16:02:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *GRPCProvisionerServer) ValidateProvisionerConfig(_ context.Context, req *proto.ValidateProvisionerConfig_Request) (*proto.ValidateProvisionerConfig_Response, error) {
|
|
|
|
resp := &proto.ValidateProvisionerConfig_Response{}
|
|
|
|
|
|
|
|
cfgSchema := schema.InternalMap(s.provisioner.Schema).CoreConfigSchema()
|
|
|
|
|
|
|
|
configVal, err := msgpack.Unmarshal(req.Config.Msgpack, cfgSchema.ImpliedType())
|
|
|
|
if err != nil {
|
2018-08-15 11:03:39 -05:00
|
|
|
resp.Diagnostics = convert.AppendProtoDiag(resp.Diagnostics, err)
|
2018-08-08 16:02:24 -05:00
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
config := terraform.NewResourceConfigShimmed(configVal, cfgSchema)
|
|
|
|
|
|
|
|
warns, errs := s.provisioner.Validate(config)
|
2018-08-15 11:03:39 -05:00
|
|
|
resp.Diagnostics = convert.AppendProtoDiag(resp.Diagnostics, convert.WarnsAndErrsToProto(warns, errs))
|
2018-08-08 16:02:24 -05:00
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// stringMapFromValue converts a cty.Value to a map[stirng]string.
|
|
|
|
// This will panic if the val is not a cty.Map(cty.String).
|
|
|
|
func stringMapFromValue(val cty.Value) map[string]string {
|
|
|
|
m := map[string]string{}
|
|
|
|
if val.IsNull() || !val.IsKnown() {
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
|
|
|
for it := val.ElementIterator(); it.Next(); {
|
|
|
|
ak, av := it.Element()
|
|
|
|
name := ak.AsString()
|
|
|
|
|
|
|
|
if !av.IsKnown() || av.IsNull() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-08-15 11:03:39 -05:00
|
|
|
av, _ = ctyconvert.Convert(av, cty.String)
|
2018-08-08 16:02:24 -05:00
|
|
|
m[name] = av.AsString()
|
|
|
|
}
|
|
|
|
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
|
|
|
// uiOutput implements the terraform.UIOutput interface to adapt the grpc
|
|
|
|
// stream to the legacy Provisioner.Apply method.
|
|
|
|
type uiOutput struct {
|
|
|
|
srv proto.Provisioner_ProvisionResourceServer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o uiOutput) Output(s string) {
|
|
|
|
err := o.srv.Send(&proto.ProvisionResource_Response{
|
2019-11-06 13:27:28 -06:00
|
|
|
Output: toValidUTF8(s, string(utf8.RuneError)),
|
2018-08-08 16:02:24 -05:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("[ERROR] %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *GRPCProvisionerServer) ProvisionResource(req *proto.ProvisionResource_Request, srv proto.Provisioner_ProvisionResourceServer) error {
|
|
|
|
// We send back a diagnostics over the stream if there was a
|
|
|
|
// provisioner-side problem.
|
|
|
|
srvResp := &proto.ProvisionResource_Response{}
|
|
|
|
|
|
|
|
cfgSchema := schema.InternalMap(s.provisioner.Schema).CoreConfigSchema()
|
|
|
|
cfgVal, err := msgpack.Unmarshal(req.Config.Msgpack, cfgSchema.ImpliedType())
|
|
|
|
if err != nil {
|
2018-08-15 11:03:39 -05:00
|
|
|
srvResp.Diagnostics = convert.AppendProtoDiag(srvResp.Diagnostics, err)
|
2018-08-08 16:02:24 -05:00
|
|
|
srv.Send(srvResp)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
resourceConfig := terraform.NewResourceConfigShimmed(cfgVal, cfgSchema)
|
|
|
|
|
|
|
|
connVal, err := msgpack.Unmarshal(req.Connection.Msgpack, cty.Map(cty.String))
|
|
|
|
if err != nil {
|
2018-08-15 11:03:39 -05:00
|
|
|
srvResp.Diagnostics = convert.AppendProtoDiag(srvResp.Diagnostics, err)
|
2018-08-08 16:02:24 -05:00
|
|
|
srv.Send(srvResp)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
conn := stringMapFromValue(connVal)
|
|
|
|
|
|
|
|
instanceState := &terraform.InstanceState{
|
|
|
|
Ephemeral: terraform.EphemeralState{
|
|
|
|
ConnInfo: conn,
|
|
|
|
},
|
2018-10-08 11:25:16 -05:00
|
|
|
Meta: make(map[string]interface{}),
|
2018-08-08 16:02:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
err = s.provisioner.Apply(uiOutput{srv}, instanceState, resourceConfig)
|
|
|
|
if err != nil {
|
2018-08-15 11:03:39 -05:00
|
|
|
srvResp.Diagnostics = convert.AppendProtoDiag(srvResp.Diagnostics, err)
|
2018-08-08 16:02:24 -05:00
|
|
|
srv.Send(srvResp)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *GRPCProvisionerServer) Stop(_ context.Context, req *proto.Stop_Request) (*proto.Stop_Response, error) {
|
|
|
|
resp := &proto.Stop_Response{}
|
|
|
|
|
|
|
|
err := s.provisioner.Stop()
|
|
|
|
if err != nil {
|
|
|
|
resp.Error = err.Error()
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
2019-11-06 13:27:28 -06:00
|
|
|
|
|
|
|
// FIXME: backported from go1.13 strings package, remove once terraform is
|
|
|
|
// using go >= 1.13
|
|
|
|
// ToValidUTF8 returns a copy of the string s with each run of invalid UTF-8 byte sequences
|
|
|
|
// replaced by the replacement string, which may be empty.
|
|
|
|
func toValidUTF8(s, replacement string) string {
|
|
|
|
var b strings.Builder
|
|
|
|
|
|
|
|
for i, c := range s {
|
|
|
|
if c != utf8.RuneError {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
_, wid := utf8.DecodeRuneInString(s[i:])
|
|
|
|
if wid == 1 {
|
|
|
|
b.Grow(len(s) + len(replacement))
|
|
|
|
b.WriteString(s[:i])
|
|
|
|
s = s[i:]
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fast path for unchanged input
|
|
|
|
if b.Cap() == 0 { // didn't call b.Grow above
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
invalid := false // previous byte was from an invalid UTF-8 sequence
|
|
|
|
for i := 0; i < len(s); {
|
|
|
|
c := s[i]
|
|
|
|
if c < utf8.RuneSelf {
|
|
|
|
i++
|
|
|
|
invalid = false
|
|
|
|
b.WriteByte(c)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
_, wid := utf8.DecodeRuneInString(s[i:])
|
|
|
|
if wid == 1 {
|
|
|
|
i++
|
|
|
|
if !invalid {
|
|
|
|
invalid = true
|
|
|
|
b.WriteString(replacement)
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
invalid = false
|
|
|
|
b.WriteString(s[i : i+wid])
|
|
|
|
i += wid
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.String()
|
|
|
|
}
|