mirror of https://github.com/containers/podman.git
				
				
				
			
		
			
				
	
	
		
			63 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			63 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			Go
		
	
	
	
| package channel
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"sync"
 | |
| 
 | |
| 	"github.com/pkg/errors"
 | |
| )
 | |
| 
 | |
| // WriteCloser is an io.WriteCloser that that proxies Write() calls to a channel
 | |
| // The []byte buffer of the Write() is queued on the channel as one message.
 | |
| type WriteCloser interface {
 | |
| 	io.WriteCloser
 | |
| 	Chan() <-chan []byte
 | |
| }
 | |
| 
 | |
| type writeCloser struct {
 | |
| 	ch  chan []byte
 | |
| 	mux sync.Mutex
 | |
| }
 | |
| 
 | |
| // NewWriter initializes a new channel writer
 | |
| func NewWriter(c chan []byte) WriteCloser {
 | |
| 	return &writeCloser{
 | |
| 		ch: c,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Chan returns the R/O channel behind WriteCloser
 | |
| func (w *writeCloser) Chan() <-chan []byte {
 | |
| 	return w.ch
 | |
| }
 | |
| 
 | |
| // Write method for WriteCloser
 | |
| func (w *writeCloser) Write(b []byte) (bLen int, err error) {
 | |
| 	// https://github.com/containers/podman/issues/7896
 | |
| 	// when podman-remote pull image, if it was killed, the server will panic: send on closed channel
 | |
| 	// so handle it
 | |
| 	defer func() {
 | |
| 		if rErr := recover(); rErr != nil {
 | |
| 			err = fmt.Errorf("%s", rErr)
 | |
| 		}
 | |
| 	}()
 | |
| 	if w == nil || w.ch == nil {
 | |
| 		return 0, errors.New("use channel.NewWriter() to initialize a WriteCloser")
 | |
| 	}
 | |
| 
 | |
| 	w.mux.Lock()
 | |
| 	defer w.mux.Unlock()
 | |
| 	buf := make([]byte, len(b))
 | |
| 	copy(buf, b)
 | |
| 	w.ch <- buf
 | |
| 
 | |
| 	return len(b), nil
 | |
| }
 | |
| 
 | |
| // Close method for WriteCloser
 | |
| func (w *writeCloser) Close() error {
 | |
| 	close(w.ch)
 | |
| 	return nil
 | |
| }
 |