packer-cn/packer/rpc/hook.go

93 lines
2.0 KiB
Go
Raw Normal View History

2013-05-11 12:51:49 -04:00
package rpc
import (
"context"
"log"
"net/rpc"
2019-05-10 13:25:54 -04:00
"sync"
2018-01-22 20:21:10 -05:00
"github.com/hashicorp/packer/packer"
2013-05-11 12:51:49 -04:00
)
// An implementation of packer.Hook where the hook is actually executed
// over an RPC connection.
type hook struct {
client *rpc.Client
2014-09-02 17:23:06 -04:00
mux *muxBroker
2013-05-11 12:51:49 -04:00
}
// HookServer wraps a packer.Hook implementation and makes it exportable
// as part of a Golang RPC server.
type HookServer struct {
context context.Context
contextCancel func()
2013-05-11 12:51:49 -04:00
hook packer.Hook
2019-05-10 13:25:54 -04:00
lock sync.Mutex
2014-09-02 17:23:06 -04:00
mux *muxBroker
2013-05-11 12:51:49 -04:00
}
type HookRunArgs struct {
2013-12-10 14:50:30 -05:00
Name string
Data interface{}
StreamId uint32
2013-05-11 12:51:49 -04:00
}
func (h *hook) Run(ctx context.Context, name string, ui packer.Ui, comm packer.Communicator, data interface{}) error {
2013-12-10 14:50:30 -05:00
nextId := h.mux.NextId()
server := newServerWithMux(h.mux, nextId)
2013-12-10 14:50:30 -05:00
server.RegisterCommunicator(comm)
server.RegisterUi(ui)
go server.Serve()
2013-05-11 12:51:49 -04:00
done := make(chan interface{})
defer close(done)
go func() {
select {
case <-ctx.Done():
log.Printf("Cancelling hook after context cancellation %v", ctx.Err())
if err := h.client.Call("Hook.Cancel", new(interface{}), new(interface{})); err != nil {
log.Printf("Error cancelling builder: %s", err)
}
case <-done:
}
}()
2013-12-10 14:50:30 -05:00
args := HookRunArgs{
Name: name,
Data: data,
StreamId: nextId,
}
return h.client.Call("Hook.Run", &args, new(interface{}))
}
func (h *HookServer) Run(args *HookRunArgs, reply *interface{}) error {
client, err := newClientWithMux(h.mux, args.StreamId)
2013-05-11 12:51:49 -04:00
if err != nil {
2013-12-10 14:50:30 -05:00
return NewBasicError(err)
2013-05-11 12:51:49 -04:00
}
2013-12-10 14:50:30 -05:00
defer client.Close()
2013-05-11 12:51:49 -04:00
2019-05-10 13:25:54 -04:00
h.lock.Lock()
if h.context == nil {
h.context, h.contextCancel = context.WithCancel(context.Background())
}
2019-05-10 13:25:54 -04:00
h.lock.Unlock()
if err := h.hook.Run(h.context, args.Name, client.Ui(), client.Communicator(), args.Data); err != nil {
return NewBasicError(err)
}
2013-05-11 12:51:49 -04:00
*reply = nil
return nil
}
func (h *HookServer) Cancel(args *interface{}, reply *interface{}) error {
2019-05-10 13:25:54 -04:00
h.lock.Lock()
if h.contextCancel != nil {
h.contextCancel()
}
2019-05-10 13:25:54 -04:00
h.lock.Unlock()
return nil
}