| // Copyright 2015 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. |
| |
| // No testdata on Android. |
| |
| // +build !android |
| |
| package main |
| |
| import ( |
| "bytes" |
| "os" |
| "path/filepath" |
| "runtime" |
| "strings" |
| "testing" |
| ) |
| |
| // TODO(adonovan): |
| // - test introduction of renaming imports. |
| // - test induced failures of rewriteFile. |
| |
| // Guide to the test packages: |
| // |
| // new.com/one -- canonical name for old.com/one |
| // old.com/one -- non-canonical; has import comment "new.com/one" |
| // old.com/bad -- has a parse error |
| // fruit.io/orange \ |
| // fruit.io/banana } orange -> pear -> banana -> titanic.biz/bar |
| // fruit.io/pear / |
| // titanic.biz/bar -- domain is sinking; package has jumped ship to new.com/bar |
| // titanic.biz/foo -- domain is sinking but package has no import comment yet |
| |
| func TestFixImports(t *testing.T) { |
| gopath := filepath.Join(cwd, "testdata") |
| if err := os.Setenv("GOPATH", gopath); err != nil { |
| t.Fatalf("os.Setenv: %v", err) |
| } |
| defer func() { |
| stderr = os.Stderr |
| *badDomains = "code.google.com" |
| *replaceFlag = "" |
| }() |
| |
| for i, test := range []struct { |
| packages []string // packages to rewrite, "go list" syntax |
| badDomains string // -baddomains flag |
| replaceFlag string // -replace flag |
| wantOK bool |
| wantStderr string |
| wantRewrite map[string]string |
| }{ |
| // #0. No errors. |
| { |
| packages: []string{"all"}, |
| badDomains: "code.google.com", |
| wantOK: true, |
| wantStderr: ` |
| testdata/src/old.com/bad/bad.go:2:43: expected 'package', found 'EOF' |
| fruit.io/banana |
| fixed: old.com/one -> new.com/one |
| fixed: titanic.biz/bar -> new.com/bar |
| `, |
| wantRewrite: map[string]string{ |
| "$GOPATH/src/fruit.io/banana/banana.go": `package banana |
| |
| import ( |
| _ "new.com/bar" |
| _ "new.com/one" |
| _ "titanic.biz/foo" |
| )`, |
| }, |
| }, |
| // #1. No packages needed rewriting. |
| { |
| packages: []string{"titanic.biz/...", "old.com/...", "new.com/..."}, |
| badDomains: "code.google.com", |
| wantOK: true, |
| wantStderr: ` |
| testdata/src/old.com/bad/bad.go:2:43: expected 'package', found 'EOF' |
| `, |
| }, |
| // #2. Some packages without import comments matched bad domains. |
| { |
| packages: []string{"all"}, |
| badDomains: "titanic.biz", |
| wantOK: false, |
| wantStderr: ` |
| testdata/src/old.com/bad/bad.go:2:43: expected 'package', found 'EOF' |
| fruit.io/banana |
| testdata/src/fruit.io/banana/banana.go:6: import "titanic.biz/foo" |
| fixed: old.com/one -> new.com/one |
| fixed: titanic.biz/bar -> new.com/bar |
| ERROR: titanic.biz/foo has no import comment |
| imported directly by: |
| fruit.io/pear |
| imported indirectly by: |
| fruit.io/orange |
| `, |
| wantRewrite: map[string]string{ |
| "$GOPATH/src/fruit.io/banana/banana.go": `package banana |
| |
| import ( |
| _ "new.com/bar" |
| _ "new.com/one" |
| _ "titanic.biz/foo" |
| )`, |
| }, |
| }, |
| // #3. The -replace flag lets user supply missing import comments. |
| { |
| packages: []string{"all"}, |
| replaceFlag: "titanic.biz/foo=new.com/foo", |
| wantOK: true, |
| wantStderr: ` |
| testdata/src/old.com/bad/bad.go:2:43: expected 'package', found 'EOF' |
| fruit.io/banana |
| fixed: old.com/one -> new.com/one |
| fixed: titanic.biz/bar -> new.com/bar |
| fixed: titanic.biz/foo -> new.com/foo |
| `, |
| wantRewrite: map[string]string{ |
| "$GOPATH/src/fruit.io/banana/banana.go": `package banana |
| |
| import ( |
| _ "new.com/bar" |
| _ "new.com/foo" |
| _ "new.com/one" |
| )`, |
| }, |
| }, |
| // #4. The -replace flag supports wildcards. |
| // An explicit import comment takes precedence. |
| { |
| packages: []string{"all"}, |
| replaceFlag: "titanic.biz/...=new.com/...", |
| wantOK: true, |
| wantStderr: ` |
| testdata/src/old.com/bad/bad.go:2:43: expected 'package', found 'EOF' |
| fruit.io/banana |
| fixed: old.com/one -> new.com/one |
| fixed: titanic.biz/bar -> new.com/bar |
| fixed: titanic.biz/foo -> new.com/foo |
| `, |
| wantRewrite: map[string]string{ |
| "$GOPATH/src/fruit.io/banana/banana.go": `package banana |
| |
| import ( |
| _ "new.com/bar" |
| _ "new.com/foo" |
| _ "new.com/one" |
| )`, |
| }, |
| }, |
| // #5. The -replace flag trumps -baddomains. |
| { |
| packages: []string{"all"}, |
| badDomains: "titanic.biz", |
| replaceFlag: "titanic.biz/foo=new.com/foo", |
| wantOK: true, |
| wantStderr: ` |
| testdata/src/old.com/bad/bad.go:2:43: expected 'package', found 'EOF' |
| fruit.io/banana |
| fixed: old.com/one -> new.com/one |
| fixed: titanic.biz/bar -> new.com/bar |
| fixed: titanic.biz/foo -> new.com/foo |
| `, |
| wantRewrite: map[string]string{ |
| "$GOPATH/src/fruit.io/banana/banana.go": `package banana |
| |
| import ( |
| _ "new.com/bar" |
| _ "new.com/foo" |
| _ "new.com/one" |
| )`, |
| }, |
| }, |
| } { |
| *badDomains = test.badDomains |
| *replaceFlag = test.replaceFlag |
| |
| stderr = new(bytes.Buffer) |
| gotRewrite := make(map[string]string) |
| writeFile = func(filename string, content []byte, mode os.FileMode) error { |
| filename = strings.Replace(filename, gopath, "$GOPATH", 1) |
| filename = filepath.ToSlash(filename) |
| gotRewrite[filename] = string(bytes.TrimSpace(content)) |
| return nil |
| } |
| |
| if runtime.GOOS == "windows" { |
| test.wantStderr = strings.Replace(test.wantStderr, `testdata/src/old.com/bad/bad.go`, `testdata\src\old.com\bad\bad.go`, -1) |
| test.wantStderr = strings.Replace(test.wantStderr, `testdata/src/fruit.io/banana/banana.go`, `testdata\src\fruit.io\banana\banana.go`, -1) |
| } |
| |
| // Check status code. |
| if fiximports(test.packages...) != test.wantOK { |
| t.Errorf("#%d. fiximports() = %t", i, !test.wantOK) |
| } |
| |
| // Compare stderr output. |
| if got := stderr.(*bytes.Buffer).String(); got != test.wantStderr { |
| if strings.Contains(got, "vendor/golang_org/x/text/unicode/norm") { |
| t.Skip("skipping known-broken test; see golang.org/issue/17417") |
| } |
| t.Errorf("#%d. stderr: got <<%s>>, want <<%s>>", |
| i, stderr, test.wantStderr) |
| } |
| |
| // Compare rewrites. |
| for k, v := range gotRewrite { |
| if test.wantRewrite[k] != v { |
| t.Errorf("#%d. rewrite[%s] = <<%s>>, want <<%s>>", |
| i, k, v, test.wantRewrite[k]) |
| } |
| delete(test.wantRewrite, k) |
| } |
| for k, v := range test.wantRewrite { |
| t.Errorf("#%d. rewrite[%s] missing, want <<%s>>", i, k, v) |
| } |
| } |
| } |
| |
| // TestDryRun tests that the -n flag suppresses calls to writeFile. |
| func TestDryRun(t *testing.T) { |
| gopath := filepath.Join(cwd, "testdata") |
| if err := os.Setenv("GOPATH", gopath); err != nil { |
| t.Fatalf("os.Setenv: %v", err) |
| } |
| |
| *dryrun = true |
| defer func() { *dryrun = false }() // restore |
| stderr = new(bytes.Buffer) |
| writeFile = func(filename string, content []byte, mode os.FileMode) error { |
| t.Fatalf("writeFile(%s) called in dryrun mode", filename) |
| return nil |
| } |
| |
| if !fiximports("all") { |
| t.Fatalf("fiximports failed: %s", stderr) |
| } |
| } |