Initialize module and dependencies
This commit is contained in:
155
vendor/github.com/emersion/go-message/encoding.go
generated
vendored
Normal file
155
vendor/github.com/emersion/go-message/encoding.go
generated
vendored
Normal file
@@ -0,0 +1,155 @@
|
||||
package message
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"encoding/base64"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"mime/quotedprintable"
|
||||
"strings"
|
||||
)
|
||||
|
||||
type UnknownEncodingError struct {
|
||||
e error
|
||||
}
|
||||
|
||||
func (u UnknownEncodingError) Unwrap() error { return u.e }
|
||||
|
||||
func (u UnknownEncodingError) Error() string {
|
||||
return "encoding error: " + u.e.Error()
|
||||
}
|
||||
|
||||
// IsUnknownEncoding returns a boolean indicating whether the error is known to
|
||||
// report that the encoding advertised by the entity is unknown.
|
||||
func IsUnknownEncoding(err error) bool {
|
||||
return errors.As(err, new(UnknownEncodingError))
|
||||
}
|
||||
|
||||
func encodingReader(enc string, r io.Reader) (io.Reader, error) {
|
||||
var dec io.Reader
|
||||
switch strings.ToLower(enc) {
|
||||
case "quoted-printable":
|
||||
dec = quotedprintable.NewReader(r)
|
||||
case "base64":
|
||||
wrapped := &whitespaceReplacingReader{wrapped: r}
|
||||
dec = base64.NewDecoder(base64.StdEncoding, wrapped)
|
||||
case "7bit", "8bit", "binary", "":
|
||||
dec = r
|
||||
default:
|
||||
return nil, fmt.Errorf("unhandled encoding %q", enc)
|
||||
}
|
||||
return dec, nil
|
||||
}
|
||||
|
||||
type nopCloser struct {
|
||||
io.Writer
|
||||
}
|
||||
|
||||
func (nopCloser) Close() error {
|
||||
return nil
|
||||
}
|
||||
|
||||
func encodingWriter(enc string, w io.Writer) (io.WriteCloser, error) {
|
||||
var wc io.WriteCloser
|
||||
switch strings.ToLower(enc) {
|
||||
case "quoted-printable":
|
||||
wc = quotedprintable.NewWriter(w)
|
||||
case "base64":
|
||||
wc = base64.NewEncoder(base64.StdEncoding, &lineWrapper{w: w, maxLineLen: 76})
|
||||
case "7bit", "8bit":
|
||||
wc = nopCloser{&lineWrapper{w: w, maxLineLen: 998}}
|
||||
case "binary", "":
|
||||
wc = nopCloser{w}
|
||||
default:
|
||||
return nil, fmt.Errorf("unhandled encoding %q", enc)
|
||||
}
|
||||
return wc, nil
|
||||
}
|
||||
|
||||
// whitespaceReplacingReader replaces space and tab characters with a LF so
|
||||
// base64 bodies with a continuation indent can be decoded by the base64 decoder
|
||||
// even though it is against the spec.
|
||||
type whitespaceReplacingReader struct {
|
||||
wrapped io.Reader
|
||||
}
|
||||
|
||||
func (r *whitespaceReplacingReader) Read(p []byte) (int, error) {
|
||||
n, err := r.wrapped.Read(p)
|
||||
|
||||
for i := 0; i < n; i++ {
|
||||
if p[i] == ' ' || p[i] == '\t' {
|
||||
p[i] = '\n'
|
||||
}
|
||||
}
|
||||
|
||||
return n, err
|
||||
}
|
||||
|
||||
type lineWrapper struct {
|
||||
w io.Writer
|
||||
maxLineLen int
|
||||
|
||||
curLineLen int
|
||||
cr bool
|
||||
}
|
||||
|
||||
func (w *lineWrapper) Write(b []byte) (int, error) {
|
||||
var written int
|
||||
for len(b) > 0 {
|
||||
var l []byte
|
||||
l, b = cutLine(b, w.maxLineLen-w.curLineLen)
|
||||
|
||||
lf := bytes.HasSuffix(l, []byte("\n"))
|
||||
l = bytes.TrimSuffix(l, []byte("\n"))
|
||||
|
||||
n, err := w.w.Write(l)
|
||||
if err != nil {
|
||||
return written, err
|
||||
}
|
||||
written += n
|
||||
|
||||
cr := bytes.HasSuffix(l, []byte("\r"))
|
||||
if len(l) == 0 {
|
||||
cr = w.cr
|
||||
}
|
||||
|
||||
if !lf && len(b) == 0 {
|
||||
w.curLineLen += len(l)
|
||||
w.cr = cr
|
||||
break
|
||||
}
|
||||
w.curLineLen = 0
|
||||
|
||||
ending := []byte("\r\n")
|
||||
if cr {
|
||||
ending = []byte("\n")
|
||||
}
|
||||
_, err = w.w.Write(ending)
|
||||
if err != nil {
|
||||
return written, err
|
||||
}
|
||||
// If the written `\n` was part of the input bytes slice, then account for it.
|
||||
if lf {
|
||||
written++
|
||||
}
|
||||
w.cr = false
|
||||
}
|
||||
|
||||
return written, nil
|
||||
}
|
||||
|
||||
func cutLine(b []byte, max int) ([]byte, []byte) {
|
||||
for i := 0; i < len(b); i++ {
|
||||
if b[i] == '\r' && i == max {
|
||||
continue
|
||||
}
|
||||
if b[i] == '\n' {
|
||||
return b[:i+1], b[i+1:]
|
||||
}
|
||||
if i >= max {
|
||||
return b[:i], b[i:]
|
||||
}
|
||||
}
|
||||
return b, nil
|
||||
}
|
||||
Reference in New Issue
Block a user