Plan 9 from Bell Labs’s /usr/web/sources/contrib/stallion/root/386/go/src/net/mail/message_test.go

Copyright © 2021 Plan 9 Foundation.
Distributed under the MIT License.
Download the Plan 9 distribution.


// Copyright 2011 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

package mail

import (
	"bytes"
	"io"
	"io/ioutil"
	"mime"
	"reflect"
	"strings"
	"testing"
	"time"
)

var parseTests = []struct {
	in     string
	header Header
	body   string
}{
	{
		// RFC 5322, Appendix A.1.1
		in: `From: John Doe <[email protected]>
To: Mary Smith <[email protected]>
Subject: Saying Hello
Date: Fri, 21 Nov 1997 09:55:06 -0600
Message-ID: <[email protected]>

This is a message just to say hello.
So, "Hello".
`,
		header: Header{
			"From":       []string{"John Doe <[email protected]>"},
			"To":         []string{"Mary Smith <[email protected]>"},
			"Subject":    []string{"Saying Hello"},
			"Date":       []string{"Fri, 21 Nov 1997 09:55:06 -0600"},
			"Message-Id": []string{"<[email protected]>"},
		},
		body: "This is a message just to say hello.\nSo, \"Hello\".\n",
	},
}

func TestParsing(t *testing.T) {
	for i, test := range parseTests {
		msg, err := ReadMessage(bytes.NewBuffer([]byte(test.in)))
		if err != nil {
			t.Errorf("test #%d: Failed parsing message: %v", i, err)
			continue
		}
		if !headerEq(msg.Header, test.header) {
			t.Errorf("test #%d: Incorrectly parsed message header.\nGot:\n%+v\nWant:\n%+v",
				i, msg.Header, test.header)
		}
		body, err := ioutil.ReadAll(msg.Body)
		if err != nil {
			t.Errorf("test #%d: Failed reading body: %v", i, err)
			continue
		}
		bodyStr := string(body)
		if bodyStr != test.body {
			t.Errorf("test #%d: Incorrectly parsed message body.\nGot:\n%+v\nWant:\n%+v",
				i, bodyStr, test.body)
		}
	}
}

func headerEq(a, b Header) bool {
	if len(a) != len(b) {
		return false
	}
	for k, as := range a {
		bs, ok := b[k]
		if !ok {
			return false
		}
		if !reflect.DeepEqual(as, bs) {
			return false
		}
	}
	return true
}

func TestDateParsing(t *testing.T) {
	tests := []struct {
		dateStr string
		exp     time.Time
	}{
		// RFC 5322, Appendix A.1.1
		{
			"Fri, 21 Nov 1997 09:55:06 -0600",
			time.Date(1997, 11, 21, 9, 55, 6, 0, time.FixedZone("", -6*60*60)),
		},
		// RFC 5322, Appendix A.6.2
		// Obsolete date.
		{
			"21 Nov 97 09:55:06 GMT",
			time.Date(1997, 11, 21, 9, 55, 6, 0, time.FixedZone("GMT", 0)),
		},
		// Commonly found format not specified by RFC 5322.
		{
			"Fri, 21 Nov 1997 09:55:06 -0600 (MDT)",
			time.Date(1997, 11, 21, 9, 55, 6, 0, time.FixedZone("", -6*60*60)),
		},
	}
	for _, test := range tests {
		hdr := Header{
			"Date": []string{test.dateStr},
		}
		date, err := hdr.Date()
		if err != nil {
			t.Errorf("Header(Date: %s).Date(): %v", test.dateStr, err)
		} else if !date.Equal(test.exp) {
			t.Errorf("Header(Date: %s).Date() = %+v, want %+v", test.dateStr, date, test.exp)
		}

		date, err = ParseDate(test.dateStr)
		if err != nil {
			t.Errorf("ParseDate(%s): %v", test.dateStr, err)
		} else if !date.Equal(test.exp) {
			t.Errorf("ParseDate(%s) = %+v, want %+v", test.dateStr, date, test.exp)
		}
	}
}

func TestAddressParsingError(t *testing.T) {
	mustErrTestCases := [...]struct {
		text        string
		wantErrText string
	}{
		0:  {"=?iso-8859-2?Q?Bogl=E1rka_Tak=E1cs?= <[email protected]>", "charset not supported"},
		1:  {"[email protected] [email protected]", "expected single address"},
		2:  {string([]byte{0xed, 0xa0, 0x80}) + " <[email protected]>", "invalid utf-8 in address"},
		3:  {"\"" + string([]byte{0xed, 0xa0, 0x80}) + "\" <[email protected]>", "invalid utf-8 in quoted-string"},
		4:  {"\"\\" + string([]byte{0x80}) + "\" <[email protected]>", "invalid utf-8 in quoted-string"},
		5:  {"\"\x00\" <[email protected]>", "bad character in quoted-string"},
		6:  {"\"\\\x00\" <[email protected]>", "bad character in quoted-string"},
		7:  {"John Doe", "no angle-addr"},
		8:  {`<jdoe#machine.example>`, "missing @ in addr-spec"},
		9:  {`John <middle> Doe <[email protected]>`, "missing @ in addr-spec"},
		10: {"[email protected] (", "misformatted parenthetical comment"},
		11: {"empty group: ;", "empty group"},
		12: {"root group: embed group: [email protected];", "no angle-addr"},
		13: {"group not closed: [email protected]", "expected comma"},
		14: {"group: [email protected], [email protected];", "group with multiple addresses"},
		15: {"john.doe", "missing '@' or angle-addr"},
		16: {"john.doe@", "no angle-addr"},
		17: {"John [email protected]", "no angle-addr"},
	}

	for i, tc := range mustErrTestCases {
		_, err := ParseAddress(tc.text)
		if err == nil || !strings.Contains(err.Error(), tc.wantErrText) {
			t.Errorf(`mail.ParseAddress(%q) #%d want %q, got %v`, tc.text, i, tc.wantErrText, err)
		}
	}
}

func TestAddressParsing(t *testing.T) {
	tests := []struct {
		addrsStr string
		exp      []*Address
	}{
		// Bare address
		{
			`[email protected]`,
			[]*Address{{
				Address: "[email protected]",
			}},
		},
		// RFC 5322, Appendix A.1.1
		{
			`John Doe <[email protected]>`,
			[]*Address{{
				Name:    "John Doe",
				Address: "[email protected]",
			}},
		},
		// RFC 5322, Appendix A.1.2
		{
			`"Joe Q. Public" <[email protected]>`,
			[]*Address{{
				Name:    "Joe Q. Public",
				Address: "[email protected]",
			}},
		},
		{
			`"John (middle) Doe" <[email protected]>`,
			[]*Address{{
				Name:    "John (middle) Doe",
				Address: "[email protected]",
			}},
		},
		{
			`John (middle) Doe <[email protected]>`,
			[]*Address{{
				Name:    "John (middle) Doe",
				Address: "[email protected]",
			}},
		},
		{
			`John !@M@! Doe <[email protected]>`,
			[]*Address{{
				Name:    "John !@M@! Doe",
				Address: "[email protected]",
			}},
		},
		{
			`"John <middle> Doe" <[email protected]>`,
			[]*Address{{
				Name:    "John <middle> Doe",
				Address: "[email protected]",
			}},
		},
		{
			`Mary Smith <[email protected]>, [email protected], Who? <[email protected]>`,
			[]*Address{
				{
					Name:    "Mary Smith",
					Address: "[email protected]",
				},
				{
					Address: "[email protected]",
				},
				{
					Name:    "Who?",
					Address: "[email protected]",
				},
			},
		},
		{
			`<[email protected]>, "Giant; \"Big\" Box" <[email protected]>`,
			[]*Address{
				{
					Address: "[email protected]",
				},
				{
					Name:    `Giant; "Big" Box`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 5322, Appendix A.6.1
		{
			`Joe Q. Public <[email protected]>`,
			[]*Address{{
				Name:    "Joe Q. Public",
				Address: "[email protected]",
			}},
		},
		// RFC 5322, Appendix A.1.3
		{
			`group1: [email protected];`,
			[]*Address{
				{
					Name:    "",
					Address: "[email protected]",
				},
			},
		},
		{
			`empty group: ;`,
			[]*Address(nil),
		},
		{
			`A Group:Ed Jones <[email protected]>,[email protected],John <[email protected]>;`,
			[]*Address{
				{
					Name:    "Ed Jones",
					Address: "[email protected]",
				},
				{
					Name:    "",
					Address: "[email protected]",
				},
				{
					Name:    "John",
					Address: "[email protected]",
				},
			},
		},
		{
			`Group1: <[email protected]>;, Group 2: [email protected];, John <[email protected]>`,
			[]*Address{
				{
					Name:    "",
					Address: "[email protected]",
				},
				{
					Name:    "",
					Address: "[email protected]",
				},
				{
					Name:    "John",
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "Q"-encoded ISO-8859-1 address.
		{
			`=?iso-8859-1?q?J=F6rg_Doe?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg Doe`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "Q"-encoded US-ASCII address. Dumb but legal.
		{
			`=?us-ascii?q?J=6Frg_Doe?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jorg Doe`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "Q"-encoded UTF-8 address.
		{
			`=?utf-8?q?J=C3=B6rg_Doe?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg Doe`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "Q"-encoded UTF-8 address with multiple encoded-words.
		{
			`=?utf-8?q?J=C3=B6rg?=  =?utf-8?q?Doe?= <[email protected]>`,
			[]*Address{
				{
					Name:    `JörgDoe`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047, Section 8.
		{
			`=?ISO-8859-1?Q?Andr=E9?= Pirard <[email protected]>`,
			[]*Address{
				{
					Name:    `André Pirard`,
					Address: "[email protected]",
				},
			},
		},
		// Custom example of RFC 2047 "B"-encoded ISO-8859-1 address.
		{
			`=?ISO-8859-1?B?SvZyZw==?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg`,
					Address: "[email protected]",
				},
			},
		},
		// Custom example of RFC 2047 "B"-encoded UTF-8 address.
		{
			`=?UTF-8?B?SsO2cmc=?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg`,
					Address: "[email protected]",
				},
			},
		},
		// Custom example with "." in name. For issue 4938
		{
			`Asem H. <[email protected]>`,
			[]*Address{
				{
					Name:    `Asem H.`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 6532 3.2.3, qtext /= UTF8-non-ascii
		{
			`"Gø Pher" <[email protected]>`,
			[]*Address{
				{
					Name:    `Gø Pher`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 6532 3.2, atext /= UTF8-non-ascii
		{
			`µ <[email protected]>`,
			[]*Address{
				{
					Name:    `µ`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 6532 3.2.2, local address parts allow UTF-8
		{
			`Micro <µ@example.com>`,
			[]*Address{
				{
					Name:    `Micro`,
					Address: "µ@example.com",
				},
			},
		},
		// RFC 6532 3.2.4, domains parts allow UTF-8
		{
			`Micro <micro@µ.example.com>`,
			[]*Address{
				{
					Name:    `Micro`,
					Address: "micro@µ.example.com",
				},
			},
		},
		// Issue 14866
		{
			`"" <[email protected]>`,
			[]*Address{
				{
					Name:    "",
					Address: "[email protected]",
				},
			},
		},
		// CFWS
		{
			`<[email protected]> (CFWS (cfws))  (another comment)`,
			[]*Address{
				{
					Name:    "",
					Address: "[email protected]",
				},
			},
		},
		{
			`<[email protected]> ()  (another comment), <[email protected]> (another)`,
			[]*Address{
				{
					Name:    "",
					Address: "[email protected]",
				},
				{
					Name:    "",
					Address: "[email protected]",
				},
			},
		},
		// Comment as display name
		{
			`[email protected] (John Doe)`,
			[]*Address{
				{
					Name:    "John Doe",
					Address: "[email protected]",
				},
			},
		},
		// Comment and display name
		{
			`John Doe <[email protected]> (Joey)`,
			[]*Address{
				{
					Name:    "John Doe",
					Address: "[email protected]",
				},
			},
		},
		// Comment as display name, no space
		{
			`[email protected](John Doe)`,
			[]*Address{
				{
					Name:    "John Doe",
					Address: "[email protected]",
				},
			},
		},
		// Comment as display name, Q-encoded
		{
			`[email protected] (Adam =?utf-8?Q?Sj=C3=B8gren?=)`,
			[]*Address{
				{
					Name:    "Adam Sjøgren",
					Address: "[email protected]",
				},
			},
		},
		// Comment as display name, Q-encoded and tab-separated
		{
			`[email protected] (Adam	=?utf-8?Q?Sj=C3=B8gren?=)`,
			[]*Address{
				{
					Name:    "Adam Sjøgren",
					Address: "[email protected]",
				},
			},
		},
		// Nested comment as display name, Q-encoded
		{
			`[email protected] (Adam =?utf-8?Q?Sj=C3=B8gren?= (Debian))`,
			[]*Address{
				{
					Name:    "Adam Sjøgren (Debian)",
					Address: "[email protected]",
				},
			},
		},
	}
	for _, test := range tests {
		if len(test.exp) == 1 {
			addr, err := ParseAddress(test.addrsStr)
			if err != nil {
				t.Errorf("Failed parsing (single) %q: %v", test.addrsStr, err)
				continue
			}
			if !reflect.DeepEqual([]*Address{addr}, test.exp) {
				t.Errorf("Parse (single) of %q: got %+v, want %+v", test.addrsStr, addr, test.exp)
			}
		}

		addrs, err := ParseAddressList(test.addrsStr)
		if err != nil {
			t.Errorf("Failed parsing (list) %q: %v", test.addrsStr, err)
			continue
		}
		if !reflect.DeepEqual(addrs, test.exp) {
			t.Errorf("Parse (list) of %q: got %+v, want %+v", test.addrsStr, addrs, test.exp)
		}
	}
}

func TestAddressParser(t *testing.T) {
	tests := []struct {
		addrsStr string
		exp      []*Address
	}{
		// Bare address
		{
			`[email protected]`,
			[]*Address{{
				Address: "[email protected]",
			}},
		},
		// RFC 5322, Appendix A.1.1
		{
			`John Doe <[email protected]>`,
			[]*Address{{
				Name:    "John Doe",
				Address: "[email protected]",
			}},
		},
		// RFC 5322, Appendix A.1.2
		{
			`"Joe Q. Public" <[email protected]>`,
			[]*Address{{
				Name:    "Joe Q. Public",
				Address: "[email protected]",
			}},
		},
		{
			`Mary Smith <[email protected]>, [email protected], Who? <[email protected]>`,
			[]*Address{
				{
					Name:    "Mary Smith",
					Address: "[email protected]",
				},
				{
					Address: "[email protected]",
				},
				{
					Name:    "Who?",
					Address: "[email protected]",
				},
			},
		},
		{
			`<[email protected]>, "Giant; \"Big\" Box" <[email protected]>`,
			[]*Address{
				{
					Address: "[email protected]",
				},
				{
					Name:    `Giant; "Big" Box`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "Q"-encoded ISO-8859-1 address.
		{
			`=?iso-8859-1?q?J=F6rg_Doe?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg Doe`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "Q"-encoded US-ASCII address. Dumb but legal.
		{
			`=?us-ascii?q?J=6Frg_Doe?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jorg Doe`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "Q"-encoded ISO-8859-15 address.
		{
			`=?ISO-8859-15?Q?J=F6rg_Doe?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg Doe`,
					Address: "[email protected]",
				},
			},
		},
		// RFC 2047 "B"-encoded windows-1252 address.
		{
			`=?windows-1252?q?Andr=E9?= Pirard <[email protected]>`,
			[]*Address{
				{
					Name:    `André Pirard`,
					Address: "[email protected]",
				},
			},
		},
		// Custom example of RFC 2047 "B"-encoded ISO-8859-15 address.
		{
			`=?ISO-8859-15?B?SvZyZw==?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg`,
					Address: "[email protected]",
				},
			},
		},
		// Custom example of RFC 2047 "B"-encoded UTF-8 address.
		{
			`=?UTF-8?B?SsO2cmc=?= <[email protected]>`,
			[]*Address{
				{
					Name:    `Jörg`,
					Address: "[email protected]",
				},
			},
		},
		// Custom example with "." in name. For issue 4938
		{
			`Asem H. <[email protected]>`,
			[]*Address{
				{
					Name:    `Asem H.`,
					Address: "[email protected]",
				},
			},
		},
	}

	ap := AddressParser{WordDecoder: &mime.WordDecoder{
		CharsetReader: func(charset string, input io.Reader) (io.Reader, error) {
			in, err := ioutil.ReadAll(input)
			if err != nil {
				return nil, err
			}

			switch charset {
			case "iso-8859-15":
				in = bytes.ReplaceAll(in, []byte("\xf6"), []byte("ö"))
			case "windows-1252":
				in = bytes.ReplaceAll(in, []byte("\xe9"), []byte("é"))
			}

			return bytes.NewReader(in), nil
		},
	}}

	for _, test := range tests {
		if len(test.exp) == 1 {
			addr, err := ap.Parse(test.addrsStr)
			if err != nil {
				t.Errorf("Failed parsing (single) %q: %v", test.addrsStr, err)
				continue
			}
			if !reflect.DeepEqual([]*Address{addr}, test.exp) {
				t.Errorf("Parse (single) of %q: got %+v, want %+v", test.addrsStr, addr, test.exp)
			}
		}

		addrs, err := ap.ParseList(test.addrsStr)
		if err != nil {
			t.Errorf("Failed parsing (list) %q: %v", test.addrsStr, err)
			continue
		}
		if !reflect.DeepEqual(addrs, test.exp) {
			t.Errorf("Parse (list) of %q: got %+v, want %+v", test.addrsStr, addrs, test.exp)
		}
	}
}

func TestAddressString(t *testing.T) {
	tests := []struct {
		addr *Address
		exp  string
	}{
		{
			&Address{Address: "[email protected]"},
			"<[email protected]>",
		},
		{ // quoted local parts: RFC 5322, 3.4.1. and 3.2.4.
			&Address{Address: `my@idiot@[email protected]`},
			`<"my@idiot@address"@example.com>`,
		},
		{ // quoted local parts
			&Address{Address: ` @example.com`},
			`<" "@example.com>`,
		},
		{
			&Address{Name: "Bob", Address: "[email protected]"},
			`"Bob" <[email protected]>`,
		},
		{
			// note the ö (o with an umlaut)
			&Address{Name: "Böb", Address: "[email protected]"},
			`=?utf-8?q?B=C3=B6b?= <[email protected]>`,
		},
		{
			&Address{Name: "Bob Jane", Address: "[email protected]"},
			`"Bob Jane" <[email protected]>`,
		},
		{
			&Address{Name: "Böb Jacöb", Address: "[email protected]"},
			`=?utf-8?q?B=C3=B6b_Jac=C3=B6b?= <[email protected]>`,
		},
		{ // https://golang.org/issue/12098
			&Address{Name: "Rob", Address: ""},
			`"Rob" <@>`,
		},
		{ // https://golang.org/issue/12098
			&Address{Name: "Rob", Address: "@"},
			`"Rob" <@>`,
		},
		{
			&Address{Name: "Böb, Jacöb", Address: "[email protected]"},
			`=?utf-8?b?QsO2YiwgSmFjw7Zi?= <[email protected]>`,
		},
		{
			&Address{Name: "=??Q?x?=", Address: "[email protected]"},
			`"=??Q?x?=" <[email protected]>`,
		},
		{
			&Address{Name: "=?hello", Address: "[email protected]"},
			`"=?hello" <[email protected]>`,
		},
		{
			&Address{Name: "world?=", Address: "[email protected]"},
			`"world?=" <[email protected]>`,
		},
		{
			// should q-encode even for invalid utf-8.
			&Address{Name: string([]byte{0xed, 0xa0, 0x80}), Address: "[email protected]"},
			"=?utf-8?q?=ED=A0=80?= <[email protected]>",
		},
	}
	for _, test := range tests {
		s := test.addr.String()
		if s != test.exp {
			t.Errorf("Address%+v.String() = %v, want %v", *test.addr, s, test.exp)
			continue
		}

		// Check round-trip.
		if test.addr.Address != "" && test.addr.Address != "@" {
			a, err := ParseAddress(test.exp)
			if err != nil {
				t.Errorf("ParseAddress(%#q): %v", test.exp, err)
				continue
			}
			if a.Name != test.addr.Name || a.Address != test.addr.Address {
				t.Errorf("ParseAddress(%#q) = %#v, want %#v", test.exp, a, test.addr)
			}
		}
	}
}

// Check if all valid addresses can be parsed, formatted and parsed again
func TestAddressParsingAndFormatting(t *testing.T) {

	// Should pass
	tests := []string{
		`<[email protected]>`,
		`<[email protected]>`,
		`<".bob"@example.com>`,
		`<" "@example.com>`,
		`<[email protected]>`,
		`<"dot.and space"@example.com>`,
		`<"[email protected]"@example.com>`,
		`<admin@mailserver1>`,
		`<postmaster@localhost>`,
		"<#!$%&'*+-/=?^_`{}|[email protected]>",
		`<"very.(),:;<>[]\".VERY.\"very@\\ \"very\".unusual"@strange.example.com>`, // escaped quotes
		`<"()<>[]:,;@\\\"!#$%&'*+-/=?^_{}| ~.a"@example.org>`,                      // escaped backslashes
		`<"Abc\\@def"@example.com>`,
		`<"Joe\\Blow"@example.com>`,
		`<test1/[email protected]>`,
		`<def!xyz%[email protected]>`,
		`<[email protected]>`,
		`<joe@uk>`,
		`<[email protected]>`,
		`<"..."@test.com>`,
		`<"john..doe"@example.com>`,
		`<"john.doe."@example.com>`,
		`<".john.doe"@example.com>`,
		`<"."@example.com>`,
		`<".."@example.com>`,
		`<"0:"@0>`,
	}

	for _, test := range tests {
		addr, err := ParseAddress(test)
		if err != nil {
			t.Errorf("Couldn't parse address %s: %s", test, err.Error())
			continue
		}
		str := addr.String()
		addr, err = ParseAddress(str)
		if err != nil {
			t.Errorf("ParseAddr(%q) error: %v", test, err)
			continue
		}

		if addr.String() != test {
			t.Errorf("String() round-trip = %q; want %q", addr, test)
			continue
		}

	}

	// Should fail
	badTests := []string{
		`<Abc.example.com>`,
		`<A@b@[email protected]>`,
		`<a"b(c)d,e:f;g<h>i[j\k][email protected]>`,
		`<just"not"[email protected]>`,
		`<this is"not\[email protected]>`,
		`<this\ still\"not\\[email protected]>`,
		`<[email protected]>`,
		`<[email protected]>`,
		`<[email protected]>`,
		`<[email protected]>`,
		`<[email protected]>`,
		`<[email protected]>`,
		`<@example.com>`,
		`<[email protected]>`,
		`<test@.>`,
		`< @example.com>`,
		`<""test""blah""@example.com>`,
		`<""@0>`,
	}

	for _, test := range badTests {
		_, err := ParseAddress(test)
		if err == nil {
			t.Errorf("Should have failed to parse address: %s", test)
			continue
		}

	}

}

func TestAddressFormattingAndParsing(t *testing.T) {
	tests := []*Address{
		{Name: "@lïce", Address: "[email protected]"},
		{Name: "Böb O'Connor", Address: "[email protected]"},
		{Name: "???", Address: "[email protected]"},
		{Name: "Böb ???", Address: "[email protected]"},
		{Name: "Böb (Jacöb)", Address: "[email protected]"},
		{Name: "à#$%&'(),.:;<>@[]^`{|}~'", Address: "[email protected]"},
		// https://golang.org/issue/11292
		{Name: "\"\\\x1f,\"", Address: "0@0"},
		// https://golang.org/issue/12782
		{Name: "naé, mée", Address: "[email protected]"},
	}

	for i, test := range tests {
		parsed, err := ParseAddress(test.String())
		if err != nil {
			t.Errorf("test #%d: ParseAddr(%q) error: %v", i, test.String(), err)
			continue
		}
		if parsed.Name != test.Name {
			t.Errorf("test #%d: Parsed name = %q; want %q", i, parsed.Name, test.Name)
		}
		if parsed.Address != test.Address {
			t.Errorf("test #%d: Parsed address = %q; want %q", i, parsed.Address, test.Address)
		}
	}
}

Bell Labs OSI certified Powered by Plan 9

(Return to Plan 9 Home Page)

Copyright © 2021 Plan 9 Foundation. All Rights Reserved.
Comments to [email protected].