forked from Shiloh/githaven
167 lines
3.0 KiB
Go
167 lines
3.0 KiB
Go
|
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'
|
||
|
}
|