Merge "Convert soong_javac_filter to a wrapper"
This commit is contained in:
@@ -13,11 +13,11 @@
|
|||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
blueprint_go_binary {
|
blueprint_go_binary {
|
||||||
name: "soong_javac_filter",
|
name: "soong_javac_wrapper",
|
||||||
srcs: [
|
srcs: [
|
||||||
"javac_filter.go",
|
"javac_wrapper.go",
|
||||||
],
|
],
|
||||||
testSrcs: [
|
testSrcs: [
|
||||||
"javac_filter_test.go",
|
"javac_wrapper_test.go",
|
||||||
],
|
],
|
||||||
}
|
}
|
@@ -12,8 +12,8 @@
|
|||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
// soong_javac_filter expects the output of javac on stdin, and produces
|
// soong_javac_wrapper expects a javac command line and argments, executes
|
||||||
// an ANSI colorized version of the output on stdout.
|
// it, and produces an ANSI colorized version of the output on stdout.
|
||||||
//
|
//
|
||||||
// It also hides the unhelpful and unhideable "warning there is a warning"
|
// It also hides the unhelpful and unhideable "warning there is a warning"
|
||||||
// messages.
|
// messages.
|
||||||
@@ -24,17 +24,19 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
|
"os/exec"
|
||||||
"regexp"
|
"regexp"
|
||||||
|
"syscall"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Regular expressions are based on
|
// Regular expressions are based on
|
||||||
// https://chromium.googlesource.com/chromium/src/+/master/build/android/gyp/javac.py
|
// https://chromium.googlesource.com/chromium/src/+/master/build/android/gyp/javac.py
|
||||||
// Colors are based on clang's output
|
// Colors are based on clang's output
|
||||||
var (
|
var (
|
||||||
filelinePrefix = `^[-.\w/\\]+.java:[0-9]+:`
|
filelinePrefix = `^([-.\w/\\]+.java:[0-9]+: )`
|
||||||
warningRe = regexp.MustCompile(filelinePrefix + ` (warning:) .*$`)
|
warningRe = regexp.MustCompile(filelinePrefix + `?(warning:) .*$`)
|
||||||
errorRe = regexp.MustCompile(filelinePrefix + ` (.*?:) .*$`)
|
errorRe = regexp.MustCompile(filelinePrefix + `(.*?:) .*$`)
|
||||||
markerRe = regexp.MustCompile(`\s*(\^)\s*$`)
|
markerRe = regexp.MustCompile(`()\s*(\^)\s*$`)
|
||||||
|
|
||||||
escape = "\x1b"
|
escape = "\x1b"
|
||||||
reset = escape + "[0m"
|
reset = escape + "[0m"
|
||||||
@@ -45,11 +47,69 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
err := process(bufio.NewReader(os.Stdin), os.Stdout)
|
exitCode, err := Main(os.Args[0], os.Args[1:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Fprintln(os.Stderr, "reading standard input:", err)
|
fmt.Fprintln(os.Stderr, err.Error())
|
||||||
os.Exit(-1)
|
|
||||||
}
|
}
|
||||||
|
os.Exit(exitCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Main(name string, args []string) (int, error) {
|
||||||
|
if len(args) < 1 {
|
||||||
|
return 1, fmt.Errorf("usage: %s javac ...", name)
|
||||||
|
}
|
||||||
|
|
||||||
|
pr, pw, err := os.Pipe()
|
||||||
|
if err != nil {
|
||||||
|
return 1, fmt.Errorf("creating output pipe: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd := exec.Command(args[0], args[1:]...)
|
||||||
|
cmd.Stdin = os.Stdin
|
||||||
|
cmd.Stdout = pw
|
||||||
|
cmd.Stderr = pw
|
||||||
|
err = cmd.Start()
|
||||||
|
if err != nil {
|
||||||
|
return 1, fmt.Errorf("starting subprocess: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
pw.Close()
|
||||||
|
|
||||||
|
// Process subprocess stdout asynchronously
|
||||||
|
errCh := make(chan error)
|
||||||
|
go func() {
|
||||||
|
errCh <- process(pr, os.Stdout)
|
||||||
|
}()
|
||||||
|
|
||||||
|
// Wait for subprocess to finish
|
||||||
|
cmdErr := cmd.Wait()
|
||||||
|
|
||||||
|
// Wait for asynchronous stdout processing to finish
|
||||||
|
err = <-errCh
|
||||||
|
|
||||||
|
// Check for subprocess exit code
|
||||||
|
if cmdErr != nil {
|
||||||
|
if exitErr, ok := cmdErr.(*exec.ExitError); ok {
|
||||||
|
if status, ok := exitErr.Sys().(syscall.WaitStatus); ok {
|
||||||
|
if status.Exited() {
|
||||||
|
return status.ExitStatus(), nil
|
||||||
|
} else if status.Signaled() {
|
||||||
|
exitCode := 128 + int(status.Signal())
|
||||||
|
return exitCode, nil
|
||||||
|
} else {
|
||||||
|
return 1, exitErr
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return 1, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return 1, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func process(r io.Reader, w io.Writer) error {
|
func process(r io.Reader, w io.Writer) error {
|
||||||
@@ -61,7 +121,11 @@ func process(r io.Reader, w io.Writer) error {
|
|||||||
for scanner.Scan() {
|
for scanner.Scan() {
|
||||||
processLine(w, scanner.Text())
|
processLine(w, scanner.Text())
|
||||||
}
|
}
|
||||||
return scanner.Err()
|
err := scanner.Err()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("scanning input: %s", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func processLine(w io.Writer, line string) {
|
func processLine(w io.Writer, line string) {
|
||||||
@@ -83,7 +147,7 @@ func processLine(w io.Writer, line string) {
|
|||||||
// Returns line, modified if it matched, and true if it matched.
|
// Returns line, modified if it matched, and true if it matched.
|
||||||
func applyColor(line, color string, re *regexp.Regexp) (string, bool) {
|
func applyColor(line, color string, re *regexp.Regexp) (string, bool) {
|
||||||
if m := re.FindStringSubmatchIndex(line); m != nil {
|
if m := re.FindStringSubmatchIndex(line); m != nil {
|
||||||
tagStart, tagEnd := m[2], m[3]
|
tagStart, tagEnd := m[4], m[5]
|
||||||
line = bold + line[:tagStart] +
|
line = bold + line[:tagStart] +
|
||||||
color + line[tagStart:tagEnd] + reset + bold +
|
color + line[tagStart:tagEnd] + reset + bold +
|
||||||
line[tagEnd:] + reset
|
line[tagEnd:] + reset
|
@@ -16,6 +16,7 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"strconv"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -38,6 +39,10 @@ var testCases = []struct {
|
|||||||
in: "File.java:398: warning: [RectIntersectReturnValueIgnored] Return value of com.blah.function() must be checked\n",
|
in: "File.java:398: warning: [RectIntersectReturnValueIgnored] Return value of com.blah.function() must be checked\n",
|
||||||
out: "\x1b[1mFile.java:398: \x1b[35mwarning:\x1b[0m\x1b[1m [RectIntersectReturnValueIgnored] Return value of com.blah.function() must be checked\x1b[0m\n",
|
out: "\x1b[1mFile.java:398: \x1b[35mwarning:\x1b[0m\x1b[1m [RectIntersectReturnValueIgnored] Return value of com.blah.function() must be checked\x1b[0m\n",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
in: "warning: [options] bootstrap class path not set in conjunction with -source 1.7\n",
|
||||||
|
out: "\x1b[1m\x1b[35mwarning:\x1b[0m\x1b[1m [options] bootstrap class path not set in conjunction with -source 1.7\x1b[0m\n",
|
||||||
|
},
|
||||||
{
|
{
|
||||||
in: " (see http://go/errorprone/bugpattern/RectIntersectReturnValueIgnored.md)\n",
|
in: " (see http://go/errorprone/bugpattern/RectIntersectReturnValueIgnored.md)\n",
|
||||||
out: " (see http://go/errorprone/bugpattern/RectIntersectReturnValueIgnored.md)\n",
|
out: " (see http://go/errorprone/bugpattern/RectIntersectReturnValueIgnored.md)\n",
|
||||||
@@ -60,15 +65,50 @@ Note: dir/file.java uses unchecked or unsafe operations.
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestJavacColorize(t *testing.T) {
|
func TestJavacColorize(t *testing.T) {
|
||||||
for _, test := range testCases {
|
for i, test := range testCases {
|
||||||
buf := new(bytes.Buffer)
|
t.Run(strconv.Itoa(i), func(t *testing.T) {
|
||||||
err := process(bytes.NewReader([]byte(test.in)), buf)
|
buf := new(bytes.Buffer)
|
||||||
if err != nil {
|
err := process(bytes.NewReader([]byte(test.in)), buf)
|
||||||
t.Errorf("error: %q", err)
|
if err != nil {
|
||||||
}
|
t.Errorf("error: %q", err)
|
||||||
got := string(buf.Bytes())
|
}
|
||||||
if got != test.out {
|
got := string(buf.Bytes())
|
||||||
t.Errorf("expected %q got %q", test.out, got)
|
if got != test.out {
|
||||||
}
|
t.Errorf("expected %q got %q", test.out, got)
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestSubprocess(t *testing.T) {
|
||||||
|
t.Run("failure", func(t *testing.T) {
|
||||||
|
exitCode, err := Main("test", []string{"sh", "-c", "exit 9"})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
if exitCode != 9 {
|
||||||
|
t.Fatal("expected exit code 9, got", exitCode)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("signal", func(t *testing.T) {
|
||||||
|
exitCode, err := Main("test", []string{"sh", "-c", "kill -9 $$"})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
if exitCode != 137 {
|
||||||
|
t.Fatal("expected exit code 137, got", exitCode)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("success", func(t *testing.T) {
|
||||||
|
exitCode, err := Main("test", []string{"echo"})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
if exitCode != 0 {
|
||||||
|
t.Fatal("expected exit code 0, got", exitCode)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
Reference in New Issue
Block a user