cloudfns: send email updates when the wiki changes

This change introduces two cloud functions: wikiwebhook and
sendwikidiff. The former handles GitHub Wiki change events
sent over HTTP and enqueues them on a pubsub topic for the
latter to pick up. sendwikidiff then checks out the wiki
repo and sends an email with the diff of the change to
golang-wikichanges@.

The reason it is split into two functions is due to GitHub’s
timeout limit on webhook handlers (ten seconds). In testing,
a cold boot of a function that does everything described
above would sometimes hit that limit.

Updates golang/go#27313

Change-Id: I1974e1434c7003482df724d6ea3b537e22231c36
Reviewed-on: https://go-review.googlesource.com/c/158642
Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
diff --git a/cloudfns/wikiwebhook/README.md b/cloudfns/wikiwebhook/README.md
new file mode 100644
index 0000000..c3e9dcb
--- /dev/null
+++ b/cloudfns/wikiwebhook/README.md
@@ -0,0 +1,9 @@
+# wikiwebhook Cloud Function
+
+```sh
+gcloud functions deploy GitHubWikiChangeWebHook \
+  --project=symbolic-datum-552 \
+  --runtime go111 \
+  --trigger-http \
+  --set-env-vars=PUBSUB_TOPIC=github.webhooks.golang.go.wiki,GITHUB_WEBHOOK_SECRET=<github webhook secret>
+```
diff --git a/cloudfns/wikiwebhook/go.mod b/cloudfns/wikiwebhook/go.mod
new file mode 100644
index 0000000..72c81c2
--- /dev/null
+++ b/cloudfns/wikiwebhook/go.mod
@@ -0,0 +1,10 @@
+module golang.org/x/build/cloudfns/wikiwebhook
+
+require (
+	cloud.google.com/go v0.34.0
+	github.com/googleapis/gax-go v2.0.2+incompatible // indirect
+	golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4 // indirect
+	google.golang.org/api v0.1.0 // indirect
+	google.golang.org/genproto v0.0.0-20190111180523-db91494dd46c // indirect
+	google.golang.org/grpc v1.18.0 // indirect
+)
diff --git a/cloudfns/wikiwebhook/go.sum b/cloudfns/wikiwebhook/go.sum
new file mode 100644
index 0000000..9de6b63
--- /dev/null
+++ b/cloudfns/wikiwebhook/go.sum
@@ -0,0 +1,64 @@
+cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
+cloud.google.com/go v0.34.0 h1:eOI3/cP2VTU6uZLDYAoic+eyzzB9YyGmJ7eIjl8rOPg=
+cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
+git.apache.org/thrift.git v0.0.0-20180902110319-2566ecd5d999/go.mod h1:fPE2ZNJGynbRyZ4dJvy6G277gSllfV2HJqblrnkyeyg=
+github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
+github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
+github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
+github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
+github.com/golang/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:tluoj9z5200jBnyusfRPU2LqT6J+DAorxEvtC7LHB+E=
+github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
+github.com/golang/protobuf v1.2.0 h1:P3YflyNX/ehuJFLhxviNdFxQPkGK5cDcApsge1SqnvM=
+github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
+github.com/googleapis/gax-go v2.0.2+incompatible h1:silFMLAnr330+NRuag/VjIGF7TLp/LBrV2CJKFLWEww=
+github.com/googleapis/gax-go v2.0.2+incompatible/go.mod h1:SFVmujtThgffbyetf+mdk2eWhX2bMyUtNHzFKcPA9HY=
+github.com/grpc-ecosystem/grpc-gateway v1.5.0/go.mod h1:RSKVYQBd5MCa4OVpNdGskqpgL2+G+NZTnrVHpWWfpdw=
+github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
+github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
+github.com/openzipkin/zipkin-go v0.1.1/go.mod h1:NtoC/o8u3JlF1lSlyPNswIbeQH9bJTmOf0Erfk+hxe8=
+github.com/prometheus/client_golang v0.8.0/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
+github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
+github.com/prometheus/common v0.0.0-20180801064454-c7de2306084e/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro=
+github.com/prometheus/procfs v0.0.0-20180725123919-05ee40e3a273/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
+go.opencensus.io v0.18.0 h1:Mk5rgZcggtbvtAun5aJzAtjKKN/t0R3jJPlWILlv938=
+go.opencensus.io v0.18.0/go.mod h1:vKdFvxhtzZ9onBp9VKHK8z/sRpBMnKAsufL7wlDrCOA=
+golang.org/x/lint v0.0.0-20180702182130-06c8688daad7/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
+golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
+golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/net v0.0.0-20181106065722-10aee1819953 h1:LuZIitY8waaxUfNIdtajyE/YzA/zyf0YxXG27VpLrkg=
+golang.org/x/net v0.0.0-20181106065722-10aee1819953/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
+golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
+golang.org/x/oauth2 v0.0.0-20181203162652-d668ce993890 h1:uESlIz09WIHT2I+pasSXcpLYqYK8wHcdCetU3VuMBJE=
+golang.org/x/oauth2 v0.0.0-20181203162652-d668ce993890/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
+golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4 h1:YUO/7uOKsKeq9UokNS62b8FYywz3ker1l1vDZRCRefw=
+golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
+golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e h1:o3PsSEY8E4eXWkXrIP9YJALUkVZqzHJT5DOasTyn8Vs=
+golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg=
+golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
+golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
+google.golang.org/api v0.0.0-20180910000450-7ca32eb868bf/go.mod h1:4mhQ8q/RsB7i+udVvVy5NUi08OU8ZlA0gRVgrF7VFY0=
+google.golang.org/api v0.1.0 h1:K6z2u68e86TPdSdefXdzvXgR1zEMa+459vBSfWYAZkI=
+google.golang.org/api v0.1.0/go.mod h1:UGEZY7KEX120AnNLIHFMKIo4obdJhkp2tPbaPlQx13Y=
+google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
+google.golang.org/appengine v1.3.0 h1:FBSsiFRMz3LBeXIomRnVzrQwSDj4ibvcRexLG0LZGQk=
+google.golang.org/appengine v1.3.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
+google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
+google.golang.org/genproto v0.0.0-20180831171423-11092d34479b/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
+google.golang.org/genproto v0.0.0-20181202183823-bd91e49a0898/go.mod h1:7Ep/1NZk928CDR8SjdVbjWNpdIf6nzjE3BTgJDr2Atg=
+google.golang.org/genproto v0.0.0-20190111180523-db91494dd46c h1:LZllHYjdJnynBfmwysp+s4yhMzfc+3BzhdqzAMvwjoc=
+google.golang.org/genproto v0.0.0-20190111180523-db91494dd46c/go.mod h1:7Ep/1NZk928CDR8SjdVbjWNpdIf6nzjE3BTgJDr2Atg=
+google.golang.org/grpc v1.14.0/go.mod h1:yo6s7OP7yaDglbqo1J04qKzAhqBH6lvTonzMVmEdcZw=
+google.golang.org/grpc v1.16.0/go.mod h1:0JHn/cJsOMiMfNA9+DeHDlAU7KAAB5GDlYFpa9MZMio=
+google.golang.org/grpc v1.17.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs=
+google.golang.org/grpc v1.18.0 h1:IZl7mfBGfbhYx2p2rKRtYgDFw6SBz+kclmxYrCksPPA=
+google.golang.org/grpc v1.18.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs=
+gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
+gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
+honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
diff --git a/cloudfns/wikiwebhook/wikiwebhook.go b/cloudfns/wikiwebhook/wikiwebhook.go
new file mode 100644
index 0000000..9daca95
--- /dev/null
+++ b/cloudfns/wikiwebhook/wikiwebhook.go
@@ -0,0 +1,99 @@
+// 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 wikiwebhook implements an Google Cloud Function HTTP handler that
+// expects GitHub webhook change events. Specifically, it reacts to wiki change
+// events and posts the payload to a pubsub topic.
+package wikiwebhook
+
+import (
+	"context"
+	"crypto/hmac"
+	"crypto/sha1"
+	"encoding/hex"
+	"fmt"
+	"io/ioutil"
+	"net/http"
+	"os"
+
+	"cloud.google.com/go/pubsub"
+)
+
+var (
+	githubSecret = os.Getenv("GITHUB_WEBHOOK_SECRET")
+	projectID    = os.Getenv("GCP_PROJECT")
+	pubsubTopic  = os.Getenv("PUBSUB_TOPIC")
+)
+
+func GitHubWikiChangeWebHook(w http.ResponseWriter, r *http.Request) {
+	defer r.Body.Close()
+	body, err := ioutil.ReadAll(r.Body)
+	if err != nil {
+		fmt.Fprintf(os.Stderr, "Could not read request body: %v", err)
+		http.Error(w, err.Error(), http.StatusInternalServerError)
+		return
+	}
+
+	if !validSignature(body, []byte(githubSecret), r.Header.Get("X-Hub-Signature")) {
+		http.Error(w, "signature mismatch", http.StatusUnauthorized)
+		return
+	}
+
+	evt := r.Header.Get("X-GitHub-Event")
+	// Ping event is sent upon initial setup of the webhook.
+	if evt == "ping" {
+		fmt.Fprintf(w, "pong")
+		return
+	}
+	// See https://developer.github.com/v3/activity/events/types/#gollumevent.
+	if evt != "gollum" {
+		http.Error(w, fmt.Sprintf("incorrect event type %q", evt), http.StatusBadRequest)
+		return
+	}
+
+	id, err := publishToTopic(pubsubTopic, body)
+	if err != nil {
+		fmt.Fprintf(os.Stderr, "Unable to publish to topic: %v", err)
+		http.Error(w, err.Error(), http.StatusInternalServerError)
+		return
+	}
+	fmt.Fprintf(w, "Message ID: %s\n", id)
+}
+
+// publishToTopic publishes body to the given topic.
+// It returns the ID of the message published.
+var publishToTopic = func(topic string, body []byte) (string, error) {
+	ctx := context.Background()
+	client, err := pubsub.NewClient(ctx, projectID)
+	if err != nil {
+		return "", fmt.Errorf("pubsub.NewClient: %v", err)
+	}
+
+	t := client.Topic(topic)
+	resp := t.Publish(ctx, &pubsub.Message{Data: body})
+	id, err := resp.Get(ctx)
+	if err != nil {
+		return "", fmt.Errorf("topic.Publish: %v", err)
+	}
+	return id, nil
+}
+
+// validSignature reports whether the HMAC-SHA1 of body with key matches sig,
+// which is in the form "sha1=<HMAC-SHA1 in hex>".
+func validSignature(body, key []byte, sig string) bool {
+	const prefix = "sha1="
+	if len(sig) < len(prefix) {
+		return false
+	}
+	sig = sig[len(prefix):]
+	mac := hmac.New(sha1.New, key)
+	mac.Write(body)
+	b, err := hex.DecodeString(sig)
+	if err != nil {
+		return false
+	}
+
+	// Use hmac.Equal to avoid timing attacks.
+	return hmac.Equal(mac.Sum(nil), b)
+}
diff --git a/cloudfns/wikiwebhook/wikiwebhook_test.go b/cloudfns/wikiwebhook/wikiwebhook_test.go
new file mode 100644
index 0000000..857995c
--- /dev/null
+++ b/cloudfns/wikiwebhook/wikiwebhook_test.go
@@ -0,0 +1,119 @@
+// 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 wikiwebhook
+
+import (
+	"bytes"
+	"errors"
+	"io/ioutil"
+	"net/http/httptest"
+	"testing"
+)
+
+func TestValidSignature(t *testing.T) {
+	testCases := []struct {
+		body, key []byte
+		sig       string
+		matches   bool
+	}{
+		{[]byte("body"), []byte("key"), "sha1=70bbf6819d1037aa94ca7e7f537cbea25fe49283", true},
+		{[]byte("body"), []byte("key"), "sha1=70bbf6819d1037aa94ca7e7f537cbea25fe49284", false},
+		{[]byte{}, []byte{}, "", false},
+		{[]byte{}, []byte{}, "sha1=not a valid hex string", false},
+	}
+	for _, tc := range testCases {
+		if matches := validSignature(tc.body, tc.key, tc.sig); matches != tc.matches {
+			t.Errorf("expected match = %v; got match = %v\nbody: %q, key: %q, sig: %q", tc.matches, matches, tc.body, tc.key, tc.sig)
+		}
+	}
+}
+
+func TestWebHook(t *testing.T) {
+	testCases := []struct {
+		desc       string
+		body       []byte
+		headers    map[string]string
+		publishFn  func(string, []byte) (string, error)
+		statusCode int
+		respBody   []byte
+	}{
+		{
+			"invalid signature",
+			nil,
+			map[string]string{
+				"X-Hub-Signature": "sha1=invalid",
+			},
+			nil,
+			401,
+			[]byte("signature mismatch\n"),
+		},
+		{
+			"ping event",
+			nil,
+			map[string]string{
+				"X-Hub-Signature": "sha1=fbdb1d1b18aa6c08324b7d64b71fb76370690e1d",
+				"X-GitHub-Event":  "ping",
+			},
+			nil,
+			200,
+			[]byte("pong"),
+		},
+		{
+			"wiki change event",
+			[]byte("body"),
+			map[string]string{
+				"X-Hub-Signature": "sha1=cc5e6b2b046bc7401d071a3d9be9a1cf1869376d",
+				"X-GitHub-Event":  "gollum",
+			},
+			func(topic string, body []byte) (string, error) {
+				if got, want := body, []byte("body"); !bytes.Equal(got, want) {
+					t.Errorf("unexpected body: got %q; expected %q", got, want)
+				}
+				return "42", nil
+			},
+			200,
+			[]byte("Message ID: 42\n"),
+		},
+		{
+			"error publishing topic",
+			nil,
+			map[string]string{
+				"X-Hub-Signature": "sha1=fbdb1d1b18aa6c08324b7d64b71fb76370690e1d",
+				"X-GitHub-Event":  "gollum",
+			},
+			func(topic string, body []byte) (string, error) {
+				return "", errors.New("publishToTopic error")
+			},
+			500,
+			[]byte("publishToTopic error\n"),
+		},
+	}
+	for _, tc := range testCases {
+		t.Run(tc.desc, func(t *testing.T) {
+			oldFn := publishToTopic
+			defer func() { publishToTopic = oldFn }()
+			publishToTopic = tc.publishFn
+
+			req := httptest.NewRequest("GET", "http://cloudfunctionz.com/func", bytes.NewReader(tc.body))
+			for k, v := range tc.headers {
+				req.Header.Set(k, v)
+			}
+			w := httptest.NewRecorder()
+			GitHubWikiChangeWebHook(w, req)
+
+			resp := w.Result()
+			body, err := ioutil.ReadAll(resp.Body)
+			if err != nil {
+				t.Errorf("ioutil.ReadAll: %v", err)
+			}
+			if got, want := resp.StatusCode, tc.statusCode; got != want {
+				t.Errorf("Unexpected status code: got %d; want %d", got, want)
+			}
+			if !bytes.Equal(body, tc.respBody) {
+				t.Errorf("Unexpected body: got %q; want %q", body, tc.respBody)
+			}
+		})
+	}
+}