12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322 |
- // Copyright 2013 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 cookiejar
- import (
- "fmt"
- "net/http"
- "net/url"
- "sort"
- "strings"
- "testing"
- "time"
- )
- // tNow is the synthetic current time used as now during testing.
- var tNow = time.Date(2013, 1, 1, 12, 0, 0, 0, time.UTC)
- // testPSL implements PublicSuffixList with just two rules: "co.uk"
- // and the default rule "*".
- // The implementation has two intentional bugs:
- // PublicSuffix("www.buggy.psl") == "xy"
- // PublicSuffix("www2.buggy.psl") == "com"
- type testPSL struct{}
- func (testPSL) String() string {
- return "testPSL"
- }
- func (testPSL) PublicSuffix(d string) string {
- if d == "co.uk" || strings.HasSuffix(d, ".co.uk") {
- return "co.uk"
- }
- if d == "www.buggy.psl" {
- return "xy"
- }
- if d == "www2.buggy.psl" {
- return "com"
- }
- return d[strings.LastIndex(d, ".")+1:]
- }
- // newTestJar creates an empty Jar with testPSL as the public suffix list.
- func newTestJar() *Jar {
- jar, err := New(&Options{PublicSuffixList: testPSL{}})
- if err != nil {
- panic(err)
- }
- return jar
- }
- var hasDotSuffixTests = [...]struct {
- s, suffix string
- }{
- {"", ""},
- {"", "."},
- {"", "x"},
- {".", ""},
- {".", "."},
- {".", ".."},
- {".", "x"},
- {".", "x."},
- {".", ".x"},
- {".", ".x."},
- {"x", ""},
- {"x", "."},
- {"x", ".."},
- {"x", "x"},
- {"x", "x."},
- {"x", ".x"},
- {"x", ".x."},
- {".x", ""},
- {".x", "."},
- {".x", ".."},
- {".x", "x"},
- {".x", "x."},
- {".x", ".x"},
- {".x", ".x."},
- {"x.", ""},
- {"x.", "."},
- {"x.", ".."},
- {"x.", "x"},
- {"x.", "x."},
- {"x.", ".x"},
- {"x.", ".x."},
- {"com", ""},
- {"com", "m"},
- {"com", "om"},
- {"com", "com"},
- {"com", ".com"},
- {"com", "x.com"},
- {"com", "xcom"},
- {"com", "xorg"},
- {"com", "org"},
- {"com", "rg"},
- {"foo.com", ""},
- {"foo.com", "m"},
- {"foo.com", "om"},
- {"foo.com", "com"},
- {"foo.com", ".com"},
- {"foo.com", "o.com"},
- {"foo.com", "oo.com"},
- {"foo.com", "foo.com"},
- {"foo.com", ".foo.com"},
- {"foo.com", "x.foo.com"},
- {"foo.com", "xfoo.com"},
- {"foo.com", "xfoo.org"},
- {"foo.com", "foo.org"},
- {"foo.com", "oo.org"},
- {"foo.com", "o.org"},
- {"foo.com", ".org"},
- {"foo.com", "org"},
- {"foo.com", "rg"},
- }
- func TestHasDotSuffix(t *testing.T) {
- for _, tc := range hasDotSuffixTests {
- got := hasDotSuffix(tc.s, tc.suffix)
- want := strings.HasSuffix(tc.s, "."+tc.suffix)
- if got != want {
- t.Errorf("s=%q, suffix=%q: got %v, want %v", tc.s, tc.suffix, got, want)
- }
- }
- }
- var canonicalHostTests = map[string]string{
- "www.example.com": "www.example.com",
- "WWW.EXAMPLE.COM": "www.example.com",
- "wWw.eXAmple.CoM": "www.example.com",
- "www.example.com:80": "www.example.com",
- "192.168.0.10": "192.168.0.10",
- "192.168.0.5:8080": "192.168.0.5",
- "2001:4860:0:2001::68": "2001:4860:0:2001::68",
- "[2001:4860:0:::68]:8080": "2001:4860:0:::68",
- "www.bücher.de": "www.xn--bcher-kva.de",
- "www.example.com.": "www.example.com",
- // TODO: Fix canonicalHost so that all of the following malformed
- // domain names trigger an error. (This list is not exhaustive, e.g.
- // malformed internationalized domain names are missing.)
- ".": "",
- "..": ".",
- "...": "..",
- ".net": ".net",
- ".net.": ".net",
- "a..": "a.",
- "b.a..": "b.a.",
- "weird.stuff...": "weird.stuff..",
- "[bad.unmatched.bracket:": "error",
- }
- func TestCanonicalHost(t *testing.T) {
- for h, want := range canonicalHostTests {
- got, err := canonicalHost(h)
- if want == "error" {
- if err == nil {
- t.Errorf("%q: got %q and nil error, want non-nil", h, got)
- }
- continue
- }
- if err != nil {
- t.Errorf("%q: %v", h, err)
- continue
- }
- if got != want {
- t.Errorf("%q: got %q, want %q", h, got, want)
- continue
- }
- }
- }
- var hasPortTests = map[string]bool{
- "www.example.com": false,
- "www.example.com:80": true,
- "127.0.0.1": false,
- "127.0.0.1:8080": true,
- "2001:4860:0:2001::68": false,
- "[2001::0:::68]:80": true,
- }
- func TestHasPort(t *testing.T) {
- for host, want := range hasPortTests {
- if got := hasPort(host); got != want {
- t.Errorf("%q: got %t, want %t", host, got, want)
- }
- }
- }
- var jarKeyTests = map[string]string{
- "foo.www.example.com": "example.com",
- "www.example.com": "example.com",
- "example.com": "example.com",
- "com": "com",
- "foo.www.bbc.co.uk": "bbc.co.uk",
- "www.bbc.co.uk": "bbc.co.uk",
- "bbc.co.uk": "bbc.co.uk",
- "co.uk": "co.uk",
- "uk": "uk",
- "192.168.0.5": "192.168.0.5",
- "www.buggy.psl": "www.buggy.psl",
- "www2.buggy.psl": "buggy.psl",
- // The following are actual outputs of canonicalHost for
- // malformed inputs to canonicalHost (see above).
- "": "",
- ".": ".",
- "..": ".",
- ".net": ".net",
- "a.": "a.",
- "b.a.": "a.",
- "weird.stuff..": ".",
- }
- func TestJarKey(t *testing.T) {
- for host, want := range jarKeyTests {
- if got := jarKey(host, testPSL{}); got != want {
- t.Errorf("%q: got %q, want %q", host, got, want)
- }
- }
- }
- var jarKeyNilPSLTests = map[string]string{
- "foo.www.example.com": "example.com",
- "www.example.com": "example.com",
- "example.com": "example.com",
- "com": "com",
- "foo.www.bbc.co.uk": "co.uk",
- "www.bbc.co.uk": "co.uk",
- "bbc.co.uk": "co.uk",
- "co.uk": "co.uk",
- "uk": "uk",
- "192.168.0.5": "192.168.0.5",
- // The following are actual outputs of canonicalHost for
- // malformed inputs to canonicalHost.
- "": "",
- ".": ".",
- "..": "..",
- ".net": ".net",
- "a.": "a.",
- "b.a.": "a.",
- "weird.stuff..": "stuff..",
- }
- func TestJarKeyNilPSL(t *testing.T) {
- for host, want := range jarKeyNilPSLTests {
- if got := jarKey(host, nil); got != want {
- t.Errorf("%q: got %q, want %q", host, got, want)
- }
- }
- }
- var isIPTests = map[string]bool{
- "127.0.0.1": true,
- "1.2.3.4": true,
- "2001:4860:0:2001::68": true,
- "example.com": false,
- "1.1.1.300": false,
- "www.foo.bar.net": false,
- "123.foo.bar.net": false,
- }
- func TestIsIP(t *testing.T) {
- for host, want := range isIPTests {
- if got := isIP(host); got != want {
- t.Errorf("%q: got %t, want %t", host, got, want)
- }
- }
- }
- var defaultPathTests = map[string]string{
- "/": "/",
- "/abc": "/",
- "/abc/": "/abc",
- "/abc/xyz": "/abc",
- "/abc/xyz/": "/abc/xyz",
- "/a/b/c.html": "/a/b",
- "": "/",
- "strange": "/",
- "//": "/",
- "/a//b": "/a/",
- "/a/./b": "/a/.",
- "/a/../b": "/a/..",
- }
- func TestDefaultPath(t *testing.T) {
- for path, want := range defaultPathTests {
- if got := defaultPath(path); got != want {
- t.Errorf("%q: got %q, want %q", path, got, want)
- }
- }
- }
- var domainAndTypeTests = [...]struct {
- host string // host Set-Cookie header was received from
- domain string // domain attribute in Set-Cookie header
- wantDomain string // expected domain of cookie
- wantHostOnly bool // expected host-cookie flag
- wantErr error // expected error
- }{
- {"www.example.com", "", "www.example.com", true, nil},
- {"127.0.0.1", "", "127.0.0.1", true, nil},
- {"2001:4860:0:2001::68", "", "2001:4860:0:2001::68", true, nil},
- {"www.example.com", "example.com", "example.com", false, nil},
- {"www.example.com", ".example.com", "example.com", false, nil},
- {"www.example.com", "www.example.com", "www.example.com", false, nil},
- {"www.example.com", ".www.example.com", "www.example.com", false, nil},
- {"foo.sso.example.com", "sso.example.com", "sso.example.com", false, nil},
- {"bar.co.uk", "bar.co.uk", "bar.co.uk", false, nil},
- {"foo.bar.co.uk", ".bar.co.uk", "bar.co.uk", false, nil},
- {"127.0.0.1", "127.0.0.1", "", false, errNoHostname},
- {"2001:4860:0:2001::68", "2001:4860:0:2001::68", "2001:4860:0:2001::68", false, errNoHostname},
- {"www.example.com", ".", "", false, errMalformedDomain},
- {"www.example.com", "..", "", false, errMalformedDomain},
- {"www.example.com", "other.com", "", false, errIllegalDomain},
- {"www.example.com", "com", "", false, errIllegalDomain},
- {"www.example.com", ".com", "", false, errIllegalDomain},
- {"foo.bar.co.uk", ".co.uk", "", false, errIllegalDomain},
- {"127.www.0.0.1", "127.0.0.1", "", false, errIllegalDomain},
- {"com", "", "com", true, nil},
- {"com", "com", "com", true, nil},
- {"com", ".com", "com", true, nil},
- {"co.uk", "", "co.uk", true, nil},
- {"co.uk", "co.uk", "co.uk", true, nil},
- {"co.uk", ".co.uk", "co.uk", true, nil},
- }
- func TestDomainAndType(t *testing.T) {
- jar := newTestJar()
- for _, tc := range domainAndTypeTests {
- domain, hostOnly, err := jar.domainAndType(tc.host, tc.domain)
- if err != tc.wantErr {
- t.Errorf("%q/%q: got %q error, want %q",
- tc.host, tc.domain, err, tc.wantErr)
- continue
- }
- if err != nil {
- continue
- }
- if domain != tc.wantDomain || hostOnly != tc.wantHostOnly {
- t.Errorf("%q/%q: got %q/%t want %q/%t",
- tc.host, tc.domain, domain, hostOnly,
- tc.wantDomain, tc.wantHostOnly)
- }
- }
- }
- // expiresIn creates an expires attribute delta seconds from tNow.
- func expiresIn(delta int) string {
- t := tNow.Add(time.Duration(delta) * time.Second)
- return "expires=" + t.Format(time.RFC1123)
- }
- // mustParseURL parses s to an URL and panics on error.
- func mustParseURL(s string) *url.URL {
- u, err := url.Parse(s)
- if err != nil || u.Scheme == "" || u.Host == "" {
- panic(fmt.Sprintf("Unable to parse URL %s.", s))
- }
- return u
- }
- // jarTest encapsulates the following actions on a jar:
- // 1. Perform SetCookies with fromURL and the cookies from setCookies.
- // (Done at time tNow + 0 ms.)
- // 2. Check that the entries in the jar matches content.
- // (Done at time tNow + 1001 ms.)
- // 3. For each query in tests: Check that Cookies with toURL yields the
- // cookies in want.
- // (Query n done at tNow + (n+2)*1001 ms.)
- type jarTest struct {
- description string // The description of what this test is supposed to test
- fromURL string // The full URL of the request from which Set-Cookie headers where received
- setCookies []string // All the cookies received from fromURL
- content string // The whole (non-expired) content of the jar
- queries []query // Queries to test the Jar.Cookies method
- }
- // query contains one test of the cookies returned from Jar.Cookies.
- type query struct {
- toURL string // the URL in the Cookies call
- want string // the expected list of cookies (order matters)
- }
- // run runs the jarTest.
- func (test jarTest) run(t *testing.T, jar *Jar) {
- now := tNow
- // Populate jar with cookies.
- setCookies := make([]*http.Cookie, len(test.setCookies))
- for i, cs := range test.setCookies {
- cookies := (&http.Response{Header: http.Header{"Set-Cookie": {cs}}}).Cookies()
- if len(cookies) != 1 {
- panic(fmt.Sprintf("Wrong cookie line %q: %#v", cs, cookies))
- }
- setCookies[i] = cookies[0]
- }
- jar.setCookies(mustParseURL(test.fromURL), setCookies, now)
- now = now.Add(1001 * time.Millisecond)
- // Serialize non-expired entries in the form "name1=val1 name2=val2".
- var cs []string
- for _, submap := range jar.entries {
- for _, cookie := range submap {
- if !cookie.Expires.After(now) {
- continue
- }
- cs = append(cs, cookie.Name+"="+cookie.Value)
- }
- }
- sort.Strings(cs)
- got := strings.Join(cs, " ")
- // Make sure jar content matches our expectations.
- if got != test.content {
- t.Errorf("Test %q Content\ngot %q\nwant %q",
- test.description, got, test.content)
- }
- // Test different calls to Cookies.
- for i, query := range test.queries {
- now = now.Add(1001 * time.Millisecond)
- var s []string
- for _, c := range jar.cookies(mustParseURL(query.toURL), now) {
- s = append(s, c.Name+"="+c.Value)
- }
- if got := strings.Join(s, " "); got != query.want {
- t.Errorf("Test %q #%d\ngot %q\nwant %q", test.description, i, got, query.want)
- }
- }
- }
- // basicsTests contains fundamental tests. Each jarTest has to be performed on
- // a fresh, empty Jar.
- var basicsTests = [...]jarTest{
- {
- "Retrieval of a plain host cookie.",
- "http://www.host.test/",
- []string{"A=a"},
- "A=a",
- []query{
- {"http://www.host.test", "A=a"},
- {"http://www.host.test/", "A=a"},
- {"http://www.host.test/some/path", "A=a"},
- {"https://www.host.test", "A=a"},
- {"https://www.host.test/", "A=a"},
- {"https://www.host.test/some/path", "A=a"},
- {"ftp://www.host.test", ""},
- {"ftp://www.host.test/", ""},
- {"ftp://www.host.test/some/path", ""},
- {"http://www.other.org", ""},
- {"http://sibling.host.test", ""},
- {"http://deep.www.host.test", ""},
- },
- },
- {
- "Secure cookies are not returned to http.",
- "http://www.host.test/",
- []string{"A=a; secure"},
- "A=a",
- []query{
- {"http://www.host.test", ""},
- {"http://www.host.test/", ""},
- {"http://www.host.test/some/path", ""},
- {"https://www.host.test", "A=a"},
- {"https://www.host.test/", "A=a"},
- {"https://www.host.test/some/path", "A=a"},
- },
- },
- {
- "Explicit path.",
- "http://www.host.test/",
- []string{"A=a; path=/some/path"},
- "A=a",
- []query{
- {"http://www.host.test", ""},
- {"http://www.host.test/", ""},
- {"http://www.host.test/some", ""},
- {"http://www.host.test/some/", ""},
- {"http://www.host.test/some/path", "A=a"},
- {"http://www.host.test/some/paths", ""},
- {"http://www.host.test/some/path/foo", "A=a"},
- {"http://www.host.test/some/path/foo/", "A=a"},
- },
- },
- {
- "Implicit path #1: path is a directory.",
- "http://www.host.test/some/path/",
- []string{"A=a"},
- "A=a",
- []query{
- {"http://www.host.test", ""},
- {"http://www.host.test/", ""},
- {"http://www.host.test/some", ""},
- {"http://www.host.test/some/", ""},
- {"http://www.host.test/some/path", "A=a"},
- {"http://www.host.test/some/paths", ""},
- {"http://www.host.test/some/path/foo", "A=a"},
- {"http://www.host.test/some/path/foo/", "A=a"},
- },
- },
- {
- "Implicit path #2: path is not a directory.",
- "http://www.host.test/some/path/index.html",
- []string{"A=a"},
- "A=a",
- []query{
- {"http://www.host.test", ""},
- {"http://www.host.test/", ""},
- {"http://www.host.test/some", ""},
- {"http://www.host.test/some/", ""},
- {"http://www.host.test/some/path", "A=a"},
- {"http://www.host.test/some/paths", ""},
- {"http://www.host.test/some/path/foo", "A=a"},
- {"http://www.host.test/some/path/foo/", "A=a"},
- },
- },
- {
- "Implicit path #3: no path in URL at all.",
- "http://www.host.test",
- []string{"A=a"},
- "A=a",
- []query{
- {"http://www.host.test", "A=a"},
- {"http://www.host.test/", "A=a"},
- {"http://www.host.test/some/path", "A=a"},
- },
- },
- {
- "Cookies are sorted by path length.",
- "http://www.host.test/",
- []string{
- "A=a; path=/foo/bar",
- "B=b; path=/foo/bar/baz/qux",
- "C=c; path=/foo/bar/baz",
- "D=d; path=/foo"},
- "A=a B=b C=c D=d",
- []query{
- {"http://www.host.test/foo/bar/baz/qux", "B=b C=c A=a D=d"},
- {"http://www.host.test/foo/bar/baz/", "C=c A=a D=d"},
- {"http://www.host.test/foo/bar", "A=a D=d"},
- },
- },
- {
- "Creation time determines sorting on same length paths.",
- "http://www.host.test/",
- []string{
- "A=a; path=/foo/bar",
- "X=x; path=/foo/bar",
- "Y=y; path=/foo/bar/baz/qux",
- "B=b; path=/foo/bar/baz/qux",
- "C=c; path=/foo/bar/baz",
- "W=w; path=/foo/bar/baz",
- "Z=z; path=/foo",
- "D=d; path=/foo"},
- "A=a B=b C=c D=d W=w X=x Y=y Z=z",
- []query{
- {"http://www.host.test/foo/bar/baz/qux", "Y=y B=b C=c W=w A=a X=x Z=z D=d"},
- {"http://www.host.test/foo/bar/baz/", "C=c W=w A=a X=x Z=z D=d"},
- {"http://www.host.test/foo/bar", "A=a X=x Z=z D=d"},
- },
- },
- {
- "Sorting of same-name cookies.",
- "http://www.host.test/",
- []string{
- "A=1; path=/",
- "A=2; path=/path",
- "A=3; path=/quux",
- "A=4; path=/path/foo",
- "A=5; domain=.host.test; path=/path",
- "A=6; domain=.host.test; path=/quux",
- "A=7; domain=.host.test; path=/path/foo",
- },
- "A=1 A=2 A=3 A=4 A=5 A=6 A=7",
- []query{
- {"http://www.host.test/path", "A=2 A=5 A=1"},
- {"http://www.host.test/path/foo", "A=4 A=7 A=2 A=5 A=1"},
- },
- },
- {
- "Disallow domain cookie on public suffix.",
- "http://www.bbc.co.uk",
- []string{
- "a=1",
- "b=2; domain=co.uk",
- },
- "a=1",
- []query{{"http://www.bbc.co.uk", "a=1"}},
- },
- {
- "Host cookie on IP.",
- "http://192.168.0.10",
- []string{"a=1"},
- "a=1",
- []query{{"http://192.168.0.10", "a=1"}},
- },
- {
- "Port is ignored #1.",
- "http://www.host.test/",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://www.host.test", "a=1"},
- {"http://www.host.test:8080/", "a=1"},
- },
- },
- {
- "Port is ignored #2.",
- "http://www.host.test:8080/",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://www.host.test", "a=1"},
- {"http://www.host.test:8080/", "a=1"},
- {"http://www.host.test:1234/", "a=1"},
- },
- },
- }
- func TestBasics(t *testing.T) {
- for _, test := range basicsTests {
- jar := newTestJar()
- test.run(t, jar)
- }
- }
- // updateAndDeleteTests contains jarTests which must be performed on the same
- // Jar.
- var updateAndDeleteTests = [...]jarTest{
- {
- "Set initial cookies.",
- "http://www.host.test",
- []string{
- "a=1",
- "b=2; secure",
- "c=3; httponly",
- "d=4; secure; httponly"},
- "a=1 b=2 c=3 d=4",
- []query{
- {"http://www.host.test", "a=1 c=3"},
- {"https://www.host.test", "a=1 b=2 c=3 d=4"},
- },
- },
- {
- "Update value via http.",
- "http://www.host.test",
- []string{
- "a=w",
- "b=x; secure",
- "c=y; httponly",
- "d=z; secure; httponly"},
- "a=w b=x c=y d=z",
- []query{
- {"http://www.host.test", "a=w c=y"},
- {"https://www.host.test", "a=w b=x c=y d=z"},
- },
- },
- {
- "Clear Secure flag from a http.",
- "http://www.host.test/",
- []string{
- "b=xx",
- "d=zz; httponly"},
- "a=w b=xx c=y d=zz",
- []query{{"http://www.host.test", "a=w b=xx c=y d=zz"}},
- },
- {
- "Delete all.",
- "http://www.host.test/",
- []string{
- "a=1; max-Age=-1", // delete via MaxAge
- "b=2; " + expiresIn(-10), // delete via Expires
- "c=2; max-age=-1; " + expiresIn(-10), // delete via both
- "d=4; max-age=-1; " + expiresIn(10)}, // MaxAge takes precedence
- "",
- []query{{"http://www.host.test", ""}},
- },
- {
- "Refill #1.",
- "http://www.host.test",
- []string{
- "A=1",
- "A=2; path=/foo",
- "A=3; domain=.host.test",
- "A=4; path=/foo; domain=.host.test"},
- "A=1 A=2 A=3 A=4",
- []query{{"http://www.host.test/foo", "A=2 A=4 A=1 A=3"}},
- },
- {
- "Refill #2.",
- "http://www.google.com",
- []string{
- "A=6",
- "A=7; path=/foo",
- "A=8; domain=.google.com",
- "A=9; path=/foo; domain=.google.com"},
- "A=1 A=2 A=3 A=4 A=6 A=7 A=8 A=9",
- []query{
- {"http://www.host.test/foo", "A=2 A=4 A=1 A=3"},
- {"http://www.google.com/foo", "A=7 A=9 A=6 A=8"},
- },
- },
- {
- "Delete A7.",
- "http://www.google.com",
- []string{"A=; path=/foo; max-age=-1"},
- "A=1 A=2 A=3 A=4 A=6 A=8 A=9",
- []query{
- {"http://www.host.test/foo", "A=2 A=4 A=1 A=3"},
- {"http://www.google.com/foo", "A=9 A=6 A=8"},
- },
- },
- {
- "Delete A4.",
- "http://www.host.test",
- []string{"A=; path=/foo; domain=host.test; max-age=-1"},
- "A=1 A=2 A=3 A=6 A=8 A=9",
- []query{
- {"http://www.host.test/foo", "A=2 A=1 A=3"},
- {"http://www.google.com/foo", "A=9 A=6 A=8"},
- },
- },
- {
- "Delete A6.",
- "http://www.google.com",
- []string{"A=; max-age=-1"},
- "A=1 A=2 A=3 A=8 A=9",
- []query{
- {"http://www.host.test/foo", "A=2 A=1 A=3"},
- {"http://www.google.com/foo", "A=9 A=8"},
- },
- },
- {
- "Delete A3.",
- "http://www.host.test",
- []string{"A=; domain=host.test; max-age=-1"},
- "A=1 A=2 A=8 A=9",
- []query{
- {"http://www.host.test/foo", "A=2 A=1"},
- {"http://www.google.com/foo", "A=9 A=8"},
- },
- },
- {
- "No cross-domain delete.",
- "http://www.host.test",
- []string{
- "A=; domain=google.com; max-age=-1",
- "A=; path=/foo; domain=google.com; max-age=-1"},
- "A=1 A=2 A=8 A=9",
- []query{
- {"http://www.host.test/foo", "A=2 A=1"},
- {"http://www.google.com/foo", "A=9 A=8"},
- },
- },
- {
- "Delete A8 and A9.",
- "http://www.google.com",
- []string{
- "A=; domain=google.com; max-age=-1",
- "A=; path=/foo; domain=google.com; max-age=-1"},
- "A=1 A=2",
- []query{
- {"http://www.host.test/foo", "A=2 A=1"},
- {"http://www.google.com/foo", ""},
- },
- },
- }
- func TestUpdateAndDelete(t *testing.T) {
- jar := newTestJar()
- for _, test := range updateAndDeleteTests {
- test.run(t, jar)
- }
- }
- func TestExpiration(t *testing.T) {
- jar := newTestJar()
- jarTest{
- "Expiration.",
- "http://www.host.test",
- []string{
- "a=1",
- "b=2; max-age=3",
- "c=3; " + expiresIn(3),
- "d=4; max-age=5",
- "e=5; " + expiresIn(5),
- "f=6; max-age=100",
- },
- "a=1 b=2 c=3 d=4 e=5 f=6", // executed at t0 + 1001 ms
- []query{
- {"http://www.host.test", "a=1 b=2 c=3 d=4 e=5 f=6"}, // t0 + 2002 ms
- {"http://www.host.test", "a=1 d=4 e=5 f=6"}, // t0 + 3003 ms
- {"http://www.host.test", "a=1 d=4 e=5 f=6"}, // t0 + 4004 ms
- {"http://www.host.test", "a=1 f=6"}, // t0 + 5005 ms
- {"http://www.host.test", "a=1 f=6"}, // t0 + 6006 ms
- },
- }.run(t, jar)
- }
- //
- // Tests derived from Chromium's cookie_store_unittest.h.
- //
- // See http://src.chromium.org/viewvc/chrome/trunk/src/net/cookies/cookie_store_unittest.h?revision=159685&content-type=text/plain
- // Some of the original tests are in a bad condition (e.g.
- // DomainWithTrailingDotTest) or are not RFC 6265 conforming (e.g.
- // TestNonDottedAndTLD #1 and #6) and have not been ported.
- // chromiumBasicsTests contains fundamental tests. Each jarTest has to be
- // performed on a fresh, empty Jar.
- var chromiumBasicsTests = [...]jarTest{
- {
- "DomainWithTrailingDotTest.",
- "http://www.google.com/",
- []string{
- "a=1; domain=.www.google.com.",
- "b=2; domain=.www.google.com.."},
- "",
- []query{
- {"http://www.google.com", ""},
- },
- },
- {
- "ValidSubdomainTest #1.",
- "http://a.b.c.d.com",
- []string{
- "a=1; domain=.a.b.c.d.com",
- "b=2; domain=.b.c.d.com",
- "c=3; domain=.c.d.com",
- "d=4; domain=.d.com"},
- "a=1 b=2 c=3 d=4",
- []query{
- {"http://a.b.c.d.com", "a=1 b=2 c=3 d=4"},
- {"http://b.c.d.com", "b=2 c=3 d=4"},
- {"http://c.d.com", "c=3 d=4"},
- {"http://d.com", "d=4"},
- },
- },
- {
- "ValidSubdomainTest #2.",
- "http://a.b.c.d.com",
- []string{
- "a=1; domain=.a.b.c.d.com",
- "b=2; domain=.b.c.d.com",
- "c=3; domain=.c.d.com",
- "d=4; domain=.d.com",
- "X=bcd; domain=.b.c.d.com",
- "X=cd; domain=.c.d.com"},
- "X=bcd X=cd a=1 b=2 c=3 d=4",
- []query{
- {"http://b.c.d.com", "b=2 c=3 d=4 X=bcd X=cd"},
- {"http://c.d.com", "c=3 d=4 X=cd"},
- },
- },
- {
- "InvalidDomainTest #1.",
- "http://foo.bar.com",
- []string{
- "a=1; domain=.yo.foo.bar.com",
- "b=2; domain=.foo.com",
- "c=3; domain=.bar.foo.com",
- "d=4; domain=.foo.bar.com.net",
- "e=5; domain=ar.com",
- "f=6; domain=.",
- "g=7; domain=/",
- "h=8; domain=http://foo.bar.com",
- "i=9; domain=..foo.bar.com",
- "j=10; domain=..bar.com",
- "k=11; domain=.foo.bar.com?blah",
- "l=12; domain=.foo.bar.com/blah",
- "m=12; domain=.foo.bar.com:80",
- "n=14; domain=.foo.bar.com:",
- "o=15; domain=.foo.bar.com#sup",
- },
- "", // Jar is empty.
- []query{{"http://foo.bar.com", ""}},
- },
- {
- "InvalidDomainTest #2.",
- "http://foo.com.com",
- []string{"a=1; domain=.foo.com.com.com"},
- "",
- []query{{"http://foo.bar.com", ""}},
- },
- {
- "DomainWithoutLeadingDotTest #1.",
- "http://manage.hosted.filefront.com",
- []string{"a=1; domain=filefront.com"},
- "a=1",
- []query{{"http://www.filefront.com", "a=1"}},
- },
- {
- "DomainWithoutLeadingDotTest #2.",
- "http://www.google.com",
- []string{"a=1; domain=www.google.com"},
- "a=1",
- []query{
- {"http://www.google.com", "a=1"},
- {"http://sub.www.google.com", "a=1"},
- {"http://something-else.com", ""},
- },
- },
- {
- "CaseInsensitiveDomainTest.",
- "http://www.google.com",
- []string{
- "a=1; domain=.GOOGLE.COM",
- "b=2; domain=.www.gOOgLE.coM"},
- "a=1 b=2",
- []query{{"http://www.google.com", "a=1 b=2"}},
- },
- {
- "TestIpAddress #1.",
- "http://1.2.3.4/foo",
- []string{"a=1; path=/"},
- "a=1",
- []query{{"http://1.2.3.4/foo", "a=1"}},
- },
- {
- "TestIpAddress #2.",
- "http://1.2.3.4/foo",
- []string{
- "a=1; domain=.1.2.3.4",
- "b=2; domain=.3.4"},
- "",
- []query{{"http://1.2.3.4/foo", ""}},
- },
- {
- "TestIpAddress #3.",
- "http://1.2.3.4/foo",
- []string{"a=1; domain=1.2.3.4"},
- "",
- []query{{"http://1.2.3.4/foo", ""}},
- },
- {
- "TestNonDottedAndTLD #2.",
- "http://com./index.html",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://com./index.html", "a=1"},
- {"http://no-cookies.com./index.html", ""},
- },
- },
- {
- "TestNonDottedAndTLD #3.",
- "http://a.b",
- []string{
- "a=1; domain=.b",
- "b=2; domain=b"},
- "",
- []query{{"http://bar.foo", ""}},
- },
- {
- "TestNonDottedAndTLD #4.",
- "http://google.com",
- []string{
- "a=1; domain=.com",
- "b=2; domain=com"},
- "",
- []query{{"http://google.com", ""}},
- },
- {
- "TestNonDottedAndTLD #5.",
- "http://google.co.uk",
- []string{
- "a=1; domain=.co.uk",
- "b=2; domain=.uk"},
- "",
- []query{
- {"http://google.co.uk", ""},
- {"http://else.co.com", ""},
- {"http://else.uk", ""},
- },
- },
- {
- "TestHostEndsWithDot.",
- "http://www.google.com",
- []string{
- "a=1",
- "b=2; domain=.www.google.com."},
- "a=1",
- []query{{"http://www.google.com", "a=1"}},
- },
- {
- "PathTest",
- "http://www.google.izzle",
- []string{"a=1; path=/wee"},
- "a=1",
- []query{
- {"http://www.google.izzle/wee", "a=1"},
- {"http://www.google.izzle/wee/", "a=1"},
- {"http://www.google.izzle/wee/war", "a=1"},
- {"http://www.google.izzle/wee/war/more/more", "a=1"},
- {"http://www.google.izzle/weehee", ""},
- {"http://www.google.izzle/", ""},
- },
- },
- }
- func TestChromiumBasics(t *testing.T) {
- for _, test := range chromiumBasicsTests {
- jar := newTestJar()
- test.run(t, jar)
- }
- }
- // chromiumDomainTests contains jarTests which must be executed all on the
- // same Jar.
- var chromiumDomainTests = [...]jarTest{
- {
- "Fill #1.",
- "http://www.google.izzle",
- []string{"A=B"},
- "A=B",
- []query{{"http://www.google.izzle", "A=B"}},
- },
- {
- "Fill #2.",
- "http://www.google.izzle",
- []string{"C=D; domain=.google.izzle"},
- "A=B C=D",
- []query{{"http://www.google.izzle", "A=B C=D"}},
- },
- {
- "Verify A is a host cookie and not accessible from subdomain.",
- "http://unused.nil",
- []string{},
- "A=B C=D",
- []query{{"http://foo.www.google.izzle", "C=D"}},
- },
- {
- "Verify domain cookies are found on proper domain.",
- "http://www.google.izzle",
- []string{"E=F; domain=.www.google.izzle"},
- "A=B C=D E=F",
- []query{{"http://www.google.izzle", "A=B C=D E=F"}},
- },
- {
- "Leading dots in domain attributes are optional.",
- "http://www.google.izzle",
- []string{"G=H; domain=www.google.izzle"},
- "A=B C=D E=F G=H",
- []query{{"http://www.google.izzle", "A=B C=D E=F G=H"}},
- },
- {
- "Verify domain enforcement works #1.",
- "http://www.google.izzle",
- []string{"K=L; domain=.bar.www.google.izzle"},
- "A=B C=D E=F G=H",
- []query{{"http://bar.www.google.izzle", "C=D E=F G=H"}},
- },
- {
- "Verify domain enforcement works #2.",
- "http://unused.nil",
- []string{},
- "A=B C=D E=F G=H",
- []query{{"http://www.google.izzle", "A=B C=D E=F G=H"}},
- },
- }
- func TestChromiumDomain(t *testing.T) {
- jar := newTestJar()
- for _, test := range chromiumDomainTests {
- test.run(t, jar)
- }
- }
- // chromiumDeletionTests must be performed all on the same Jar.
- var chromiumDeletionTests = [...]jarTest{
- {
- "Create session cookie a1.",
- "http://www.google.com",
- []string{"a=1"},
- "a=1",
- []query{{"http://www.google.com", "a=1"}},
- },
- {
- "Delete sc a1 via MaxAge.",
- "http://www.google.com",
- []string{"a=1; max-age=-1"},
- "",
- []query{{"http://www.google.com", ""}},
- },
- {
- "Create session cookie b2.",
- "http://www.google.com",
- []string{"b=2"},
- "b=2",
- []query{{"http://www.google.com", "b=2"}},
- },
- {
- "Delete sc b2 via Expires.",
- "http://www.google.com",
- []string{"b=2; " + expiresIn(-10)},
- "",
- []query{{"http://www.google.com", ""}},
- },
- {
- "Create persistent cookie c3.",
- "http://www.google.com",
- []string{"c=3; max-age=3600"},
- "c=3",
- []query{{"http://www.google.com", "c=3"}},
- },
- {
- "Delete pc c3 via MaxAge.",
- "http://www.google.com",
- []string{"c=3; max-age=-1"},
- "",
- []query{{"http://www.google.com", ""}},
- },
- {
- "Create persistent cookie d4.",
- "http://www.google.com",
- []string{"d=4; max-age=3600"},
- "d=4",
- []query{{"http://www.google.com", "d=4"}},
- },
- {
- "Delete pc d4 via Expires.",
- "http://www.google.com",
- []string{"d=4; " + expiresIn(-10)},
- "",
- []query{{"http://www.google.com", ""}},
- },
- }
- func TestChromiumDeletion(t *testing.T) {
- jar := newTestJar()
- for _, test := range chromiumDeletionTests {
- test.run(t, jar)
- }
- }
- // domainHandlingTests tests and documents the rules for domain handling.
- // Each test must be performed on an empty new Jar.
- var domainHandlingTests = [...]jarTest{
- {
- "Host cookie",
- "http://www.host.test",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://www.host.test", "a=1"},
- {"http://host.test", ""},
- {"http://bar.host.test", ""},
- {"http://foo.www.host.test", ""},
- {"http://other.test", ""},
- {"http://test", ""},
- },
- },
- {
- "Domain cookie #1",
- "http://www.host.test",
- []string{"a=1; domain=host.test"},
- "a=1",
- []query{
- {"http://www.host.test", "a=1"},
- {"http://host.test", "a=1"},
- {"http://bar.host.test", "a=1"},
- {"http://foo.www.host.test", "a=1"},
- {"http://other.test", ""},
- {"http://test", ""},
- },
- },
- {
- "Domain cookie #2",
- "http://www.host.test",
- []string{"a=1; domain=.host.test"},
- "a=1",
- []query{
- {"http://www.host.test", "a=1"},
- {"http://host.test", "a=1"},
- {"http://bar.host.test", "a=1"},
- {"http://foo.www.host.test", "a=1"},
- {"http://other.test", ""},
- {"http://test", ""},
- },
- },
- {
- "Host cookie on IDNA domain #1",
- "http://www.bücher.test",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://www.bücher.test", "a=1"},
- {"http://www.xn--bcher-kva.test", "a=1"},
- {"http://bücher.test", ""},
- {"http://xn--bcher-kva.test", ""},
- {"http://bar.bücher.test", ""},
- {"http://bar.xn--bcher-kva.test", ""},
- {"http://foo.www.bücher.test", ""},
- {"http://foo.www.xn--bcher-kva.test", ""},
- {"http://other.test", ""},
- {"http://test", ""},
- },
- },
- {
- "Host cookie on IDNA domain #2",
- "http://www.xn--bcher-kva.test",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://www.bücher.test", "a=1"},
- {"http://www.xn--bcher-kva.test", "a=1"},
- {"http://bücher.test", ""},
- {"http://xn--bcher-kva.test", ""},
- {"http://bar.bücher.test", ""},
- {"http://bar.xn--bcher-kva.test", ""},
- {"http://foo.www.bücher.test", ""},
- {"http://foo.www.xn--bcher-kva.test", ""},
- {"http://other.test", ""},
- {"http://test", ""},
- },
- },
- {
- "Domain cookie on IDNA domain #1",
- "http://www.bücher.test",
- []string{"a=1; domain=xn--bcher-kva.test"},
- "a=1",
- []query{
- {"http://www.bücher.test", "a=1"},
- {"http://www.xn--bcher-kva.test", "a=1"},
- {"http://bücher.test", "a=1"},
- {"http://xn--bcher-kva.test", "a=1"},
- {"http://bar.bücher.test", "a=1"},
- {"http://bar.xn--bcher-kva.test", "a=1"},
- {"http://foo.www.bücher.test", "a=1"},
- {"http://foo.www.xn--bcher-kva.test", "a=1"},
- {"http://other.test", ""},
- {"http://test", ""},
- },
- },
- {
- "Domain cookie on IDNA domain #2",
- "http://www.xn--bcher-kva.test",
- []string{"a=1; domain=xn--bcher-kva.test"},
- "a=1",
- []query{
- {"http://www.bücher.test", "a=1"},
- {"http://www.xn--bcher-kva.test", "a=1"},
- {"http://bücher.test", "a=1"},
- {"http://xn--bcher-kva.test", "a=1"},
- {"http://bar.bücher.test", "a=1"},
- {"http://bar.xn--bcher-kva.test", "a=1"},
- {"http://foo.www.bücher.test", "a=1"},
- {"http://foo.www.xn--bcher-kva.test", "a=1"},
- {"http://other.test", ""},
- {"http://test", ""},
- },
- },
- {
- "Host cookie on TLD.",
- "http://com",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://com", "a=1"},
- {"http://any.com", ""},
- {"http://any.test", ""},
- },
- },
- {
- "Domain cookie on TLD becomes a host cookie.",
- "http://com",
- []string{"a=1; domain=com"},
- "a=1",
- []query{
- {"http://com", "a=1"},
- {"http://any.com", ""},
- {"http://any.test", ""},
- },
- },
- {
- "Host cookie on public suffix.",
- "http://co.uk",
- []string{"a=1"},
- "a=1",
- []query{
- {"http://co.uk", "a=1"},
- {"http://uk", ""},
- {"http://some.co.uk", ""},
- {"http://foo.some.co.uk", ""},
- {"http://any.uk", ""},
- },
- },
- {
- "Domain cookie on public suffix is ignored.",
- "http://some.co.uk",
- []string{"a=1; domain=co.uk"},
- "",
- []query{
- {"http://co.uk", ""},
- {"http://uk", ""},
- {"http://some.co.uk", ""},
- {"http://foo.some.co.uk", ""},
- {"http://any.uk", ""},
- },
- },
- }
- func TestDomainHandling(t *testing.T) {
- for _, test := range domainHandlingTests {
- jar := newTestJar()
- test.run(t, jar)
- }
- }
- func TestIssue19384(t *testing.T) {
- cookies := []*http.Cookie{{Name: "name", Value: "value"}}
- for _, host := range []string{"", ".", "..", "..."} {
- jar, _ := New(nil)
- u := &url.URL{Scheme: "http", Host: host, Path: "/"}
- if got := jar.Cookies(u); len(got) != 0 {
- t.Errorf("host %q, got %v", host, got)
- }
- jar.SetCookies(u, cookies)
- if got := jar.Cookies(u); len(got) != 1 || got[0].Value != "value" {
- t.Errorf("host %q, got %v", host, got)
- }
- }
- }
|