Update go_binary and go_test rules to copy files in the "data"
attribute to runfiles. Add shell-based unit test for runfile
support by these rules.

--
Change-Id: Ia97278059d104b9728d53aa75c0987fc3c47b640
Reviewed-on: https://bazel-review.git.corp.google.com/#/c/2265/
MOS_MIGRATED_REVID=108139011
diff --git a/src/test/shell/bazel/bazel_go_example_test.sh b/src/test/shell/bazel/bazel_go_example_test.sh
index fd64246..0d01147 100755
--- a/src/test/shell/bazel/bazel_go_example_test.sh
+++ b/src/test/shell/bazel/bazel_go_example_test.sh
@@ -82,4 +82,76 @@
   grep "F 42" out || fail "binary output suspect"
 }
 
+function test_runfiles() {
+  mkdir -p ex/
+
+# Note this binary is also a test (for the correct handling of runfiles by
+# Bazel's go_binary rule).
+  cat <<EOF > ex/rf.go
+package main
+import (
+  "fmt"
+  "log"
+  "io/ioutil"
+)
+
+func main() {
+  rfcontent, err := ioutil.ReadFile("ex/runfile")
+  if err != nil {
+    log.Fatalf("Runfiles test binary: Error reading from runfile: %v", err)
+  }
+
+  fmt.Printf("Runfile: %s\n", rfcontent)
+}
+
+EOF
+
+  cat <<EOF > ex/rf_test.go
+package main
+import (
+  "fmt"
+  "io/ioutil"
+  "testing"
+)
+
+func TestRunfiles(t *testing.T) {
+  rfcontent, err := ioutil.ReadFile("ex/runfile")
+  if err != nil {
+    t.Errorf("TestRunfiles: Error reading from runfile: %v", err)
+  }
+
+  if string(rfcontent) != "12345\n" {
+    t.Errorf("TestRunfiles: Read incorrect value from runfile: %s", rfcontent)
+  }
+
+  fmt.Printf("Runfile: %s\n", rfcontent)
+}
+EOF
+
+  cat <<EOF > ex/runfile
+12345
+EOF
+
+  cat <<EOF > ex/BUILD
+load("/tools/build_rules/go/def", "go_binary", "go_test")
+go_binary(name = "runfiles_bin",
+  srcs = [ "rf.go" ],
+  data = [ "runfile" ])
+go_test(name = "runfiles_test",
+  srcs = [ "rf_test.go" ],
+  data = [ "runfile" ])
+EOF
+
+  assert_build //ex:runfiles_bin
+  test -x ./bazel-bin/ex/runfiles_bin || fail "binary not found"
+  (./bazel-bin/ex/runfiles_bin > out) || fail "binary does not execute"
+  grep "Runfile: 12345" out || fail "binary output suspect"
+
+  assert_build //ex:runfiles_test
+  test -x ./bazel-bin/ex/runfiles_test || fail "binary not found"
+  (./bazel-bin/ex/runfiles_test > out) || fail "binary does not execute"
+  grep "Runfile: 12345" out || fail "binary output suspect"
+
+}
+
 run_suite "go_examples"
diff --git a/tools/build_rules/go/def.bzl b/tools/build_rules/go/def.bzl
index a6c22c6..194b996 100644
--- a/tools/build_rules/go/def.bzl
+++ b/tools/build_rules/go/def.bzl
@@ -232,7 +232,11 @@
 
   emit_go_link_action(
     ctx, lib_result.transitive_go_library_object, lib_out, executable)
-  return struct(files = set([executable]) + lib_result.files)
+
+  runfiles = ctx.runfiles(collect_data = True,
+                          files = ctx.files.data)
+  return struct(files = set([executable]) + lib_result.files,
+                runfiles = runfiles)
 
 
 def go_test_impl(ctx):
@@ -268,7 +272,8 @@
   # TODO(bazel-team): the Go tests should do a chdir to the directory
   # holding the data files, so open-source go tests continue to work
   # without code changes.
-  runfiles = ctx.runfiles(collect_data = True, files = [ctx.outputs.executable])
+  runfiles = ctx.runfiles(collect_data = True,
+                          files = ctx.files.data + [ctx.outputs.executable])
   return struct(runfiles=runfiles)
 
 go_library_attrs = {