185 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			185 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// +build !windows
 | 
						|
 | 
						|
package cio
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"io"
 | 
						|
	"io/ioutil"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"sync"
 | 
						|
	"syscall"
 | 
						|
 | 
						|
	"github.com/containerd/fifo"
 | 
						|
)
 | 
						|
 | 
						|
// NewFifos returns a new set of fifos for the task
 | 
						|
func NewFifos(id string) (*FIFOSet, error) {
 | 
						|
	root := "/run/containerd/fifo"
 | 
						|
	if err := os.MkdirAll(root, 0700); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	dir, err := ioutil.TempDir(root, "")
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return &FIFOSet{
 | 
						|
		Dir: dir,
 | 
						|
		In:  filepath.Join(dir, id+"-stdin"),
 | 
						|
		Out: filepath.Join(dir, id+"-stdout"),
 | 
						|
		Err: filepath.Join(dir, id+"-stderr"),
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
func copyIO(fifos *FIFOSet, ioset *ioSet, tty bool) (_ *wgCloser, err error) {
 | 
						|
	var (
 | 
						|
		f           io.ReadWriteCloser
 | 
						|
		set         []io.Closer
 | 
						|
		ctx, cancel = context.WithCancel(context.Background())
 | 
						|
		wg          = &sync.WaitGroup{}
 | 
						|
	)
 | 
						|
	defer func() {
 | 
						|
		if err != nil {
 | 
						|
			for _, f := range set {
 | 
						|
				f.Close()
 | 
						|
			}
 | 
						|
			cancel()
 | 
						|
		}
 | 
						|
	}()
 | 
						|
 | 
						|
	if f, err = fifo.OpenFifo(ctx, fifos.In, syscall.O_WRONLY|syscall.O_CREAT|syscall.O_NONBLOCK, 0700); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	set = append(set, f)
 | 
						|
	go func(w io.WriteCloser) {
 | 
						|
		io.Copy(w, ioset.in)
 | 
						|
		w.Close()
 | 
						|
	}(f)
 | 
						|
 | 
						|
	if f, err = fifo.OpenFifo(ctx, fifos.Out, syscall.O_RDONLY|syscall.O_CREAT|syscall.O_NONBLOCK, 0700); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	set = append(set, f)
 | 
						|
	wg.Add(1)
 | 
						|
	go func(r io.ReadCloser) {
 | 
						|
		io.Copy(ioset.out, r)
 | 
						|
		r.Close()
 | 
						|
		wg.Done()
 | 
						|
	}(f)
 | 
						|
 | 
						|
	if f, err = fifo.OpenFifo(ctx, fifos.Err, syscall.O_RDONLY|syscall.O_CREAT|syscall.O_NONBLOCK, 0700); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	set = append(set, f)
 | 
						|
 | 
						|
	if !tty {
 | 
						|
		wg.Add(1)
 | 
						|
		go func(r io.ReadCloser) {
 | 
						|
			io.Copy(ioset.err, r)
 | 
						|
			r.Close()
 | 
						|
			wg.Done()
 | 
						|
		}(f)
 | 
						|
	}
 | 
						|
	return &wgCloser{
 | 
						|
		wg:     wg,
 | 
						|
		dir:    fifos.Dir,
 | 
						|
		set:    set,
 | 
						|
		cancel: cancel,
 | 
						|
	}, nil
 | 
						|
}
 | 
						|
 | 
						|
// NewDirectIO returns an IO implementation that exposes the pipes directly
 | 
						|
func NewDirectIO(ctx context.Context, terminal bool) (*DirectIO, error) {
 | 
						|
	set, err := NewFifos("")
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	f := &DirectIO{
 | 
						|
		set:      set,
 | 
						|
		terminal: terminal,
 | 
						|
	}
 | 
						|
	defer func() {
 | 
						|
		if err != nil {
 | 
						|
			f.Delete()
 | 
						|
		}
 | 
						|
	}()
 | 
						|
	if f.Stdin, err = fifo.OpenFifo(ctx, set.In, syscall.O_WRONLY|syscall.O_CREAT|syscall.O_NONBLOCK, 0700); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if f.Stdout, err = fifo.OpenFifo(ctx, set.Out, syscall.O_RDONLY|syscall.O_CREAT|syscall.O_NONBLOCK, 0700); err != nil {
 | 
						|
		f.Stdin.Close()
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if f.Stderr, err = fifo.OpenFifo(ctx, set.Err, syscall.O_RDONLY|syscall.O_CREAT|syscall.O_NONBLOCK, 0700); err != nil {
 | 
						|
		f.Stdin.Close()
 | 
						|
		f.Stdout.Close()
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return f, nil
 | 
						|
}
 | 
						|
 | 
						|
// DirectIO allows task IO to be handled externally by the caller
 | 
						|
type DirectIO struct {
 | 
						|
	Stdin  io.WriteCloser
 | 
						|
	Stdout io.ReadCloser
 | 
						|
	Stderr io.ReadCloser
 | 
						|
 | 
						|
	set      *FIFOSet
 | 
						|
	terminal bool
 | 
						|
}
 | 
						|
 | 
						|
// IOCreate returns IO avaliable for use with task creation
 | 
						|
func (f *DirectIO) IOCreate(id string) (IO, error) {
 | 
						|
	return f, nil
 | 
						|
}
 | 
						|
 | 
						|
// IOAttach returns IO avaliable for use with task attachment
 | 
						|
func (f *DirectIO) IOAttach(set *FIFOSet) (IO, error) {
 | 
						|
	return f, nil
 | 
						|
}
 | 
						|
 | 
						|
// Config returns the Config
 | 
						|
func (f *DirectIO) Config() Config {
 | 
						|
	return Config{
 | 
						|
		Terminal: f.terminal,
 | 
						|
		Stdin:    f.set.In,
 | 
						|
		Stdout:   f.set.Out,
 | 
						|
		Stderr:   f.set.Err,
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Cancel stops any IO copy operations
 | 
						|
//
 | 
						|
// Not applicable for DirectIO
 | 
						|
func (f *DirectIO) Cancel() {
 | 
						|
	// nothing to cancel as all operations are handled externally
 | 
						|
}
 | 
						|
 | 
						|
// Wait on any IO copy operations
 | 
						|
//
 | 
						|
// Not applicable for DirectIO
 | 
						|
func (f *DirectIO) Wait() {
 | 
						|
	// nothing to wait on as all operations are handled externally
 | 
						|
}
 | 
						|
 | 
						|
// Close closes all open fds
 | 
						|
func (f *DirectIO) Close() error {
 | 
						|
	err := f.Stdin.Close()
 | 
						|
	if err2 := f.Stdout.Close(); err == nil {
 | 
						|
		err = err2
 | 
						|
	}
 | 
						|
	if err2 := f.Stderr.Close(); err == nil {
 | 
						|
		err = err2
 | 
						|
	}
 | 
						|
	return err
 | 
						|
}
 | 
						|
 | 
						|
// Delete removes the underlying directory containing fifos
 | 
						|
func (f *DirectIO) Delete() error {
 | 
						|
	if f.set.Dir == "" {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
	return os.RemoveAll(f.set.Dir)
 | 
						|
}
 |