c2a237f019
This introduces a superficial change to the Beam API: * `beam.SendPipe` is renamed to the more accurate `beam.SendRPipe` * `beam.SendWPipe` is introduced as a mirror to `SendRPipe` There is no other change in the beam API. Docker-DCO-1.1-Signed-off-by: Solomon Hykes <solomon@docker.com> (github: shykes)
166 lines
3.1 KiB
Go
166 lines
3.1 KiB
Go
package beam
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
)
|
|
|
|
type Sender interface {
|
|
Send([]byte, *os.File) error
|
|
}
|
|
|
|
type Receiver interface {
|
|
Receive() ([]byte, *os.File, error)
|
|
}
|
|
|
|
type ReceiveCloser interface {
|
|
Receiver
|
|
Close() error
|
|
}
|
|
|
|
type SendCloser interface {
|
|
Sender
|
|
Close() error
|
|
}
|
|
|
|
type ReceiveSender interface {
|
|
Receiver
|
|
Sender
|
|
}
|
|
|
|
const (
|
|
R int = 1 << (32 - 1 - iota)
|
|
W
|
|
)
|
|
|
|
func sendPipe(dst Sender, data []byte, mode int) (*os.File, error) {
|
|
r, w, err := os.Pipe()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var (
|
|
remote *os.File
|
|
local *os.File
|
|
)
|
|
if mode == R {
|
|
remote = r
|
|
local = w
|
|
} else if mode == W {
|
|
remote = w
|
|
local = r
|
|
}
|
|
if err := dst.Send(data, remote); err != nil {
|
|
local.Close()
|
|
remote.Close()
|
|
return nil, err
|
|
}
|
|
return local, nil
|
|
|
|
}
|
|
|
|
// SendRPipe create a pipe and sends its *read* end attached in a beam message
|
|
// to `dst`, with `data` as the message payload.
|
|
// It returns the *write* end of the pipe, or an error.
|
|
func SendRPipe(dst Sender, data []byte) (*os.File, error) {
|
|
return sendPipe(dst, data, R)
|
|
}
|
|
|
|
// SendWPipe create a pipe and sends its *read* end attached in a beam message
|
|
// to `dst`, with `data` as the message payload.
|
|
// It returns the *write* end of the pipe, or an error.
|
|
func SendWPipe(dst Sender, data []byte) (*os.File, error) {
|
|
return sendPipe(dst, data, W)
|
|
}
|
|
|
|
func SendConn(dst Sender, data []byte) (conn *UnixConn, err error) {
|
|
local, remote, err := SocketPair()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer func() {
|
|
if err != nil {
|
|
local.Close()
|
|
remote.Close()
|
|
}
|
|
}()
|
|
conn, err = FileConn(local)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
local.Close()
|
|
if err := dst.Send(data, remote); err != nil {
|
|
return nil, err
|
|
}
|
|
return conn, nil
|
|
}
|
|
|
|
func ReceiveConn(src Receiver) ([]byte, *UnixConn, error) {
|
|
for {
|
|
data, f, err := src.Receive()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
if f == nil {
|
|
// Skip empty attachments
|
|
continue
|
|
}
|
|
conn, err := FileConn(f)
|
|
if err != nil {
|
|
// Skip beam attachments which are not connections
|
|
// (for example might be a regular file, directory etc)
|
|
continue
|
|
}
|
|
return data, conn, nil
|
|
}
|
|
panic("impossibru!")
|
|
return nil, nil, nil
|
|
}
|
|
|
|
func Copy(dst Sender, src Receiver) (int, error) {
|
|
var n int
|
|
for {
|
|
payload, attachment, err := src.Receive()
|
|
if err == io.EOF {
|
|
return n, nil
|
|
} else if err != nil {
|
|
return n, err
|
|
}
|
|
if err := dst.Send(payload, attachment); err != nil {
|
|
if attachment != nil {
|
|
attachment.Close()
|
|
}
|
|
return n, err
|
|
}
|
|
n++
|
|
}
|
|
panic("impossibru!")
|
|
return n, nil
|
|
}
|
|
|
|
// MsgDesc returns a human readable description of a beam message, usually
|
|
// for debugging purposes.
|
|
func MsgDesc(payload []byte, attachment *os.File) string {
|
|
var filedesc string = "<nil>"
|
|
if attachment != nil {
|
|
filedesc = fmt.Sprintf("%d", attachment.Fd())
|
|
}
|
|
return fmt.Sprintf("'%s'[%s]", payload, filedesc)
|
|
}
|
|
|
|
type devnull struct{}
|
|
|
|
func Devnull() ReceiveSender {
|
|
return devnull{}
|
|
}
|
|
|
|
func (d devnull) Send(p []byte, a *os.File) error {
|
|
if a != nil {
|
|
a.Close()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d devnull) Receive() ([]byte, *os.File, error) {
|
|
return nil, nil, io.EOF
|
|
}
|