You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							166 lines
						
					
					
						
							3.0 KiB
						
					
					
				
			
		
		
	
	
							166 lines
						
					
					
						
							3.0 KiB
						
					
					
				| package quotedprintable
 | |
| 
 | |
| import "io"
 | |
| 
 | |
| const lineMaxLen = 76
 | |
| 
 | |
| // A Writer is a quoted-printable writer that implements io.WriteCloser.
 | |
| type Writer struct {
 | |
| 	// Binary mode treats the writer's input as pure binary and processes end of
 | |
| 	// line bytes as binary data.
 | |
| 	Binary bool
 | |
| 
 | |
| 	w    io.Writer
 | |
| 	i    int
 | |
| 	line [78]byte
 | |
| 	cr   bool
 | |
| }
 | |
| 
 | |
| // NewWriter returns a new Writer that writes to w.
 | |
| func NewWriter(w io.Writer) *Writer {
 | |
| 	return &Writer{w: w}
 | |
| }
 | |
| 
 | |
| // Write encodes p using quoted-printable encoding and writes it to the
 | |
| // underlying io.Writer. It limits line length to 76 characters. The encoded
 | |
| // bytes are not necessarily flushed until the Writer is closed.
 | |
| func (w *Writer) Write(p []byte) (n int, err error) {
 | |
| 	for i, b := range p {
 | |
| 		switch {
 | |
| 		// Simple writes are done in batch.
 | |
| 		case b >= '!' && b <= '~' && b != '=':
 | |
| 			continue
 | |
| 		case isWhitespace(b) || !w.Binary && (b == '\n' || b == '\r'):
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		if i > n {
 | |
| 			if err := w.write(p[n:i]); err != nil {
 | |
| 				return n, err
 | |
| 			}
 | |
| 			n = i
 | |
| 		}
 | |
| 
 | |
| 		if err := w.encode(b); err != nil {
 | |
| 			return n, err
 | |
| 		}
 | |
| 		n++
 | |
| 	}
 | |
| 
 | |
| 	if n == len(p) {
 | |
| 		return n, nil
 | |
| 	}
 | |
| 
 | |
| 	if err := w.write(p[n:]); err != nil {
 | |
| 		return n, err
 | |
| 	}
 | |
| 
 | |
| 	return len(p), nil
 | |
| }
 | |
| 
 | |
| // Close closes the Writer, flushing any unwritten data to the underlying
 | |
| // io.Writer, but does not close the underlying io.Writer.
 | |
| func (w *Writer) Close() error {
 | |
| 	if err := w.checkLastByte(); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	return w.flush()
 | |
| }
 | |
| 
 | |
| // write limits text encoded in quoted-printable to 76 characters per line.
 | |
| func (w *Writer) write(p []byte) error {
 | |
| 	for _, b := range p {
 | |
| 		if b == '\n' || b == '\r' {
 | |
| 			// If the previous byte was \r, the CRLF has already been inserted.
 | |
| 			if w.cr && b == '\n' {
 | |
| 				w.cr = false
 | |
| 				continue
 | |
| 			}
 | |
| 
 | |
| 			if b == '\r' {
 | |
| 				w.cr = true
 | |
| 			}
 | |
| 
 | |
| 			if err := w.checkLastByte(); err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			if err := w.insertCRLF(); err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		if w.i == lineMaxLen-1 {
 | |
| 			if err := w.insertSoftLineBreak(); err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		w.line[w.i] = b
 | |
| 		w.i++
 | |
| 		w.cr = false
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (w *Writer) encode(b byte) error {
 | |
| 	if lineMaxLen-1-w.i < 3 {
 | |
| 		if err := w.insertSoftLineBreak(); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	w.line[w.i] = '='
 | |
| 	w.line[w.i+1] = upperhex[b>>4]
 | |
| 	w.line[w.i+2] = upperhex[b&0x0f]
 | |
| 	w.i += 3
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // checkLastByte encodes the last buffered byte if it is a space or a tab.
 | |
| func (w *Writer) checkLastByte() error {
 | |
| 	if w.i == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	b := w.line[w.i-1]
 | |
| 	if isWhitespace(b) {
 | |
| 		w.i--
 | |
| 		if err := w.encode(b); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (w *Writer) insertSoftLineBreak() error {
 | |
| 	w.line[w.i] = '='
 | |
| 	w.i++
 | |
| 
 | |
| 	return w.insertCRLF()
 | |
| }
 | |
| 
 | |
| func (w *Writer) insertCRLF() error {
 | |
| 	w.line[w.i] = '\r'
 | |
| 	w.line[w.i+1] = '\n'
 | |
| 	w.i += 2
 | |
| 
 | |
| 	return w.flush()
 | |
| }
 | |
| 
 | |
| func (w *Writer) flush() error {
 | |
| 	if _, err := w.w.Write(w.line[:w.i]); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	w.i = 0
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func isWhitespace(b byte) bool {
 | |
| 	return b == ' ' || b == '\t'
 | |
| }
 | |
| 
 |