 c553882adc
			
		
	
	c553882adc
	
	
	
		
			
			This changes the copyIO in the shim to support logging to files from client requests in addition to fifos. Signed-off-by: Michael Crosby <crosbymichael@gmail.com>
		
			
				
	
	
		
			147 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			147 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // +build !windows
 | |
| 
 | |
| /*
 | |
|    Copyright The containerd Authors.
 | |
| 
 | |
|    Licensed under the Apache License, Version 2.0 (the "License");
 | |
|    you may not use this file except in compliance with the License.
 | |
|    You may obtain a copy of the License at
 | |
| 
 | |
|        http://www.apache.org/licenses/LICENSE-2.0
 | |
| 
 | |
|    Unless required by applicable law or agreed to in writing, software
 | |
|    distributed under the License is distributed on an "AS IS" BASIS,
 | |
|    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|    See the License for the specific language governing permissions and
 | |
|    limitations under the License.
 | |
| */
 | |
| 
 | |
| package proc
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"os"
 | |
| 	"sync"
 | |
| 	"syscall"
 | |
| 
 | |
| 	"github.com/containerd/fifo"
 | |
| 	runc "github.com/containerd/go-runc"
 | |
| )
 | |
| 
 | |
| var bufPool = sync.Pool{
 | |
| 	New: func() interface{} {
 | |
| 		buffer := make([]byte, 32<<10)
 | |
| 		return &buffer
 | |
| 	},
 | |
| }
 | |
| 
 | |
| func copyPipes(ctx context.Context, rio runc.IO, stdin, stdout, stderr string, wg, cwg *sync.WaitGroup) error {
 | |
| 	var sameFile io.WriteCloser
 | |
| 	for _, i := range []struct {
 | |
| 		name string
 | |
| 		dest func(wc io.WriteCloser, rc io.Closer)
 | |
| 	}{
 | |
| 		{
 | |
| 			name: stdout,
 | |
| 			dest: func(wc io.WriteCloser, rc io.Closer) {
 | |
| 				wg.Add(1)
 | |
| 				cwg.Add(1)
 | |
| 				go func() {
 | |
| 					cwg.Done()
 | |
| 					p := bufPool.Get().(*[]byte)
 | |
| 					defer bufPool.Put(p)
 | |
| 					io.CopyBuffer(wc, rio.Stdout(), *p)
 | |
| 					wg.Done()
 | |
| 					wc.Close()
 | |
| 					if rc != nil {
 | |
| 						rc.Close()
 | |
| 					}
 | |
| 				}()
 | |
| 			},
 | |
| 		}, {
 | |
| 			name: stderr,
 | |
| 			dest: func(wc io.WriteCloser, rc io.Closer) {
 | |
| 				wg.Add(1)
 | |
| 				cwg.Add(1)
 | |
| 				go func() {
 | |
| 					cwg.Done()
 | |
| 					p := bufPool.Get().(*[]byte)
 | |
| 					defer bufPool.Put(p)
 | |
| 					io.CopyBuffer(wc, rio.Stderr(), *p)
 | |
| 					wg.Done()
 | |
| 					wc.Close()
 | |
| 					if rc != nil {
 | |
| 						rc.Close()
 | |
| 					}
 | |
| 				}()
 | |
| 			},
 | |
| 		},
 | |
| 	} {
 | |
| 		ok, err := isFifo(i.name)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		var (
 | |
| 			fw io.WriteCloser
 | |
| 			fr io.Closer
 | |
| 		)
 | |
| 		if ok {
 | |
| 			if fw, err = fifo.OpenFifo(ctx, i.name, syscall.O_WRONLY, 0); err != nil {
 | |
| 				return fmt.Errorf("containerd-shim: opening %s failed: %s", i.name, err)
 | |
| 			}
 | |
| 			if fr, err = fifo.OpenFifo(ctx, i.name, syscall.O_RDONLY, 0); err != nil {
 | |
| 				return fmt.Errorf("containerd-shim: opening %s failed: %s", i.name, err)
 | |
| 			}
 | |
| 		} else {
 | |
| 			if sameFile != nil {
 | |
| 				i.dest(sameFile, nil)
 | |
| 				continue
 | |
| 			}
 | |
| 			if fw, err = os.OpenFile(i.name, syscall.O_WRONLY|syscall.O_APPEND, 0); err != nil {
 | |
| 				return fmt.Errorf("containerd-shim: opening %s failed: %s", i.name, err)
 | |
| 			}
 | |
| 			if stdout == stderr {
 | |
| 				sameFile = fw
 | |
| 			}
 | |
| 		}
 | |
| 		i.dest(fw, fr)
 | |
| 	}
 | |
| 	if stdin == "" {
 | |
| 		rio.Stdin().Close()
 | |
| 		return nil
 | |
| 	}
 | |
| 	f, err := fifo.OpenFifo(ctx, stdin, syscall.O_RDONLY, 0)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("containerd-shim: opening %s failed: %s", stdin, err)
 | |
| 	}
 | |
| 	cwg.Add(1)
 | |
| 	go func() {
 | |
| 		cwg.Done()
 | |
| 		p := bufPool.Get().(*[]byte)
 | |
| 		defer bufPool.Put(p)
 | |
| 
 | |
| 		io.CopyBuffer(rio.Stdin(), f, *p)
 | |
| 		rio.Stdin().Close()
 | |
| 		f.Close()
 | |
| 	}()
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // isFifo checks if a file is a fifo
 | |
| // if the file does not exist then it returns false
 | |
| func isFifo(path string) (bool, error) {
 | |
| 	stat, err := os.Stat(path)
 | |
| 	if err != nil {
 | |
| 		if os.IsNotExist(err) {
 | |
| 			return false, nil
 | |
| 		}
 | |
| 		return false, err
 | |
| 	}
 | |
| 	if stat.Mode()&os.ModeNamedPipe == os.ModeNamedPipe {
 | |
| 		return true, nil
 | |
| 	}
 | |
| 	return false, nil
 | |
| }
 |