2019-06-03 18:44:43 +01:00
|
|
|
// Package event provides a handler which publishes an event
|
|
|
|
package event
|
|
|
|
|
|
|
|
import (
|
2019-11-19 00:37:45 +08:00
|
|
|
"encoding/json"
|
2019-06-03 18:44:43 +01:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"path"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2019-06-07 13:53:42 +01:00
|
|
|
"github.com/google/uuid"
|
2020-01-30 14:39:00 +03:00
|
|
|
"github.com/micro/go-micro/v2/api/handler"
|
|
|
|
proto "github.com/micro/go-micro/v2/api/proto"
|
|
|
|
"github.com/micro/go-micro/v2/util/ctx"
|
2019-06-03 18:44:43 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type event struct {
|
|
|
|
options handler.Options
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
Handler = "event"
|
|
|
|
versionRe = regexp.MustCompilePOSIX("^v[0-9]+$")
|
|
|
|
)
|
|
|
|
|
|
|
|
func eventName(parts []string) string {
|
|
|
|
return strings.Join(parts, ".")
|
|
|
|
}
|
|
|
|
|
|
|
|
func evRoute(ns, p string) (string, string) {
|
|
|
|
p = path.Clean(p)
|
|
|
|
p = strings.TrimPrefix(p, "/")
|
|
|
|
|
|
|
|
if len(p) == 0 {
|
|
|
|
return ns, "event"
|
|
|
|
}
|
|
|
|
|
|
|
|
parts := strings.Split(p, "/")
|
|
|
|
|
|
|
|
// no path
|
|
|
|
if len(parts) == 0 {
|
|
|
|
// topic: namespace
|
|
|
|
// action: event
|
|
|
|
return strings.Trim(ns, "."), "event"
|
|
|
|
}
|
|
|
|
|
|
|
|
// Treat /v[0-9]+ as versioning
|
|
|
|
// /v1/foo/bar => topic: v1.foo action: bar
|
|
|
|
if len(parts) >= 2 && versionRe.Match([]byte(parts[0])) {
|
|
|
|
topic := ns + "." + strings.Join(parts[:2], ".")
|
|
|
|
action := eventName(parts[1:])
|
|
|
|
return topic, action
|
|
|
|
}
|
|
|
|
|
|
|
|
// /foo => topic: ns.foo action: foo
|
|
|
|
// /foo/bar => topic: ns.foo action: bar
|
|
|
|
topic := ns + "." + strings.Join(parts[:1], ".")
|
|
|
|
action := eventName(parts[1:])
|
|
|
|
|
|
|
|
return topic, action
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *event) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// request to topic:event
|
|
|
|
// create event
|
|
|
|
// publish to topic
|
|
|
|
|
|
|
|
topic, action := evRoute(e.options.Namespace, r.URL.Path)
|
|
|
|
|
|
|
|
// create event
|
|
|
|
ev := &proto.Event{
|
|
|
|
Name: action,
|
|
|
|
// TODO: dedupe event
|
2019-06-07 13:53:42 +01:00
|
|
|
Id: fmt.Sprintf("%s-%s-%s", topic, action, uuid.New().String()),
|
2019-06-03 18:44:43 +01:00
|
|
|
Header: make(map[string]*proto.Pair),
|
|
|
|
Timestamp: time.Now().Unix(),
|
|
|
|
}
|
|
|
|
|
|
|
|
// set headers
|
|
|
|
for key, vals := range r.Header {
|
|
|
|
header, ok := ev.Header[key]
|
|
|
|
if !ok {
|
|
|
|
header = &proto.Pair{
|
|
|
|
Key: key,
|
|
|
|
}
|
|
|
|
ev.Header[key] = header
|
|
|
|
}
|
|
|
|
header.Values = vals
|
|
|
|
}
|
|
|
|
|
|
|
|
// set body
|
2019-11-19 00:37:45 +08:00
|
|
|
if r.Method == "GET" {
|
|
|
|
bytes, _ := json.Marshal(r.URL.Query())
|
|
|
|
ev.Data = string(bytes)
|
|
|
|
} else {
|
|
|
|
b, err := ioutil.ReadAll(r.Body)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(w, err.Error(), 500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ev.Data = string(b)
|
2019-06-03 18:44:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// get client
|
|
|
|
c := e.options.Service.Client()
|
|
|
|
|
|
|
|
// create publication
|
|
|
|
p := c.NewMessage(topic, ev)
|
|
|
|
|
|
|
|
// publish event
|
|
|
|
if err := c.Publish(ctx.FromRequest(r), p); err != nil {
|
|
|
|
http.Error(w, err.Error(), 500)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *event) String() string {
|
|
|
|
return "event"
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewHandler(opts ...handler.Option) handler.Handler {
|
|
|
|
return &event{
|
|
|
|
options: handler.NewOptions(opts...),
|
|
|
|
}
|
|
|
|
}
|