mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
cmd/fix, cmd/go, cmd/gofmt: refactor common code into new internal diff package
Change-Id: Idac8473d1752059bf2f617fd7a781000ee2c3af4
GitHub-Last-Rev: 02a3aa1a32
GitHub-Pull-Request: golang/go#35141
Reviewed-on: https://go-review.googlesource.com/c/go/+/203218
Run-TryBot: Ian Lance Taylor <iant@golang.org>
Reviewed-by: Ian Lance Taylor <iant@golang.org>
This commit is contained in:
parent
449b6abbac
commit
fd1e60f6e3
6 changed files with 75 additions and 119 deletions
|
|
@ -15,11 +15,11 @@ import (
|
||||||
"go/token"
|
"go/token"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"cmd/internal/diff"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
|
@ -186,7 +186,7 @@ func processFile(filename string, useStdin bool) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
if *doDiff {
|
if *doDiff {
|
||||||
data, err := diff(src, newSrc)
|
data, err := diff.Diff("go-fix", src, newSrc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("computing diff: %s", err)
|
return fmt.Errorf("computing diff: %s", err)
|
||||||
}
|
}
|
||||||
|
|
@ -237,46 +237,3 @@ func isGoFile(f os.FileInfo) bool {
|
||||||
name := f.Name()
|
name := f.Name()
|
||||||
return !f.IsDir() && !strings.HasPrefix(name, ".") && strings.HasSuffix(name, ".go")
|
return !f.IsDir() && !strings.HasPrefix(name, ".") && strings.HasSuffix(name, ".go")
|
||||||
}
|
}
|
||||||
|
|
||||||
func writeTempFile(dir, prefix string, data []byte) (string, error) {
|
|
||||||
file, err := ioutil.TempFile(dir, prefix)
|
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
_, err = file.Write(data)
|
|
||||||
if err1 := file.Close(); err == nil {
|
|
||||||
err = err1
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
os.Remove(file.Name())
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return file.Name(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func diff(b1, b2 []byte) (data []byte, err error) {
|
|
||||||
f1, err := writeTempFile("", "go-fix", b1)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.Remove(f1)
|
|
||||||
|
|
||||||
f2, err := writeTempFile("", "go-fix", b2)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.Remove(f2)
|
|
||||||
|
|
||||||
cmd := "diff"
|
|
||||||
if runtime.GOOS == "plan9" {
|
|
||||||
cmd = "/bin/ape/diff"
|
|
||||||
}
|
|
||||||
|
|
||||||
data, err = exec.Command(cmd, "-u", f1, f2).CombinedOutput()
|
|
||||||
if len(data) > 0 {
|
|
||||||
// diff exits with a non-zero status when the files don't match.
|
|
||||||
// Ignore that failure as long as we get output.
|
|
||||||
err = nil
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -9,6 +9,8 @@ import (
|
||||||
"go/parser"
|
"go/parser"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"cmd/internal/diff"
|
||||||
)
|
)
|
||||||
|
|
||||||
type testCase struct {
|
type testCase struct {
|
||||||
|
|
@ -123,7 +125,7 @@ func TestRewrite(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func tdiff(t *testing.T, a, b string) {
|
func tdiff(t *testing.T, a, b string) {
|
||||||
data, err := diff([]byte(a), []byte(b))
|
data, err := diff.Diff("go-fix-test", []byte(a), []byte(b))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
|
|
|
||||||
|
|
@ -9,11 +9,12 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"reflect"
|
"reflect"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"cmd/internal/diff"
|
||||||
)
|
)
|
||||||
|
|
||||||
// exists reports whether the named file exists.
|
// exists reports whether the named file exists.
|
||||||
|
|
@ -282,37 +283,9 @@ func (eq *eqchecker) checkValue(v, w reflect.Value) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// diff returns the output of running diff on b1 and b2.
|
|
||||||
func diff(b1, b2 []byte) (data []byte, err error) {
|
|
||||||
f1, err := ioutil.TempFile("", "testdiff")
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
defer os.Remove(f1.Name())
|
|
||||||
defer f1.Close()
|
|
||||||
|
|
||||||
f2, err := ioutil.TempFile("", "testdiff")
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
defer os.Remove(f2.Name())
|
|
||||||
defer f2.Close()
|
|
||||||
|
|
||||||
f1.Write(b1)
|
|
||||||
f2.Write(b2)
|
|
||||||
|
|
||||||
data, err = exec.Command("diff", "-u", f1.Name(), f2.Name()).CombinedOutput()
|
|
||||||
if len(data) > 0 {
|
|
||||||
// diff exits with a non-zero status when the files don't match.
|
|
||||||
// Ignore that failure as long as we get output.
|
|
||||||
err = nil
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// tdiff logs the diff output to t.Error.
|
// tdiff logs the diff output to t.Error.
|
||||||
func tdiff(t *testing.T, a, b string) {
|
func tdiff(t *testing.T, a, b string) {
|
||||||
data, err := diff([]byte(a), []byte(b))
|
data, err := diff.Diff("modfile-test", []byte(a), []byte(b))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
return
|
return
|
||||||
|
|
|
||||||
|
|
@ -16,11 +16,12 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
"runtime/pprof"
|
"runtime/pprof"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"cmd/internal/diff"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
|
@ -141,7 +142,7 @@ func processFile(filename string, in io.Reader, out io.Writer, stdin bool) error
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if *doDiff {
|
if *doDiff {
|
||||||
data, err := diff(src, res, filename)
|
data, err := diffWithReplaceTempFile(src, res, filename)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("computing diff: %s", err)
|
return fmt.Errorf("computing diff: %s", err)
|
||||||
}
|
}
|
||||||
|
|
@ -227,47 +228,12 @@ func gofmtMain() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func writeTempFile(dir, prefix string, data []byte) (string, error) {
|
func diffWithReplaceTempFile(b1, b2 []byte, filename string) ([]byte, error) {
|
||||||
file, err := ioutil.TempFile(dir, prefix)
|
data, err := diff.Diff("gofmt", b1, b2)
|
||||||
if err != nil {
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
_, err = file.Write(data)
|
|
||||||
if err1 := file.Close(); err == nil {
|
|
||||||
err = err1
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
os.Remove(file.Name())
|
|
||||||
return "", err
|
|
||||||
}
|
|
||||||
return file.Name(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func diff(b1, b2 []byte, filename string) (data []byte, err error) {
|
|
||||||
f1, err := writeTempFile("", "gofmt", b1)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.Remove(f1)
|
|
||||||
|
|
||||||
f2, err := writeTempFile("", "gofmt", b2)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer os.Remove(f2)
|
|
||||||
|
|
||||||
cmd := "diff"
|
|
||||||
if runtime.GOOS == "plan9" {
|
|
||||||
cmd = "/bin/ape/diff"
|
|
||||||
}
|
|
||||||
|
|
||||||
data, err = exec.Command(cmd, "-u", f1, f2).CombinedOutput()
|
|
||||||
if len(data) > 0 {
|
if len(data) > 0 {
|
||||||
// diff exits with a non-zero status when the files don't match.
|
|
||||||
// Ignore that failure as long as we get output.
|
|
||||||
return replaceTempFilename(data, filename)
|
return replaceTempFilename(data, filename)
|
||||||
}
|
}
|
||||||
return
|
return data, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// replaceTempFilename replaces temporary filenames in diff with actual one.
|
// replaceTempFilename replaces temporary filenames in diff with actual one.
|
||||||
|
|
|
||||||
|
|
@ -112,7 +112,7 @@ func runTest(t *testing.T, in, out string) {
|
||||||
}
|
}
|
||||||
|
|
||||||
t.Errorf("(gofmt %s) != %s (see %s.gofmt)", in, out, in)
|
t.Errorf("(gofmt %s) != %s (see %s.gofmt)", in, out, in)
|
||||||
d, err := diff(expected, got, in)
|
d, err := diffWithReplaceTempFile(expected, got, in)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Errorf("%s", d)
|
t.Errorf("%s", d)
|
||||||
}
|
}
|
||||||
|
|
@ -194,7 +194,7 @@ func TestDiff(t *testing.T) {
|
||||||
in := []byte("first\nsecond\n")
|
in := []byte("first\nsecond\n")
|
||||||
out := []byte("first\nthird\n")
|
out := []byte("first\nthird\n")
|
||||||
filename := "difftest.txt"
|
filename := "difftest.txt"
|
||||||
b, err := diff(in, out, filename)
|
b, err := diffWithReplaceTempFile(in, out, filename)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
58
src/cmd/internal/diff/diff.go
Normal file
58
src/cmd/internal/diff/diff.go
Normal file
|
|
@ -0,0 +1,58 @@
|
||||||
|
// Copyright 2019 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 diff implements a Diff function that compare two inputs
|
||||||
|
// using the 'diff' tool.
|
||||||
|
package diff
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"runtime"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Returns diff of two arrays of bytes in diff tool format.
|
||||||
|
func Diff(prefix string, b1, b2 []byte) ([]byte, error) {
|
||||||
|
f1, err := writeTempFile(prefix, b1)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer os.Remove(f1)
|
||||||
|
|
||||||
|
f2, err := writeTempFile(prefix, b2)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer os.Remove(f2)
|
||||||
|
|
||||||
|
cmd := "diff"
|
||||||
|
if runtime.GOOS == "plan9" {
|
||||||
|
cmd = "/bin/ape/diff"
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := exec.Command(cmd, "-u", f1, f2).CombinedOutput()
|
||||||
|
if len(data) > 0 {
|
||||||
|
// diff exits with a non-zero status when the files don't match.
|
||||||
|
// Ignore that failure as long as we get output.
|
||||||
|
err = nil
|
||||||
|
}
|
||||||
|
return data, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func writeTempFile(prefix string, data []byte) (string, error) {
|
||||||
|
file, err := ioutil.TempFile("", prefix)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
_, err = file.Write(data)
|
||||||
|
if err1 := file.Close(); err == nil {
|
||||||
|
err = err1
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
os.Remove(file.Name())
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return file.Name(), nil
|
||||||
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue