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 = "" 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 }