1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
|
package handlers
import (
"bytes"
"context"
"crypto/rand"
"encoding/hex"
"errors"
"fmt"
"log"
"net/http"
"net/url"
"path"
"strconv"
"time"
"github.com/coreos/discovery.etcd.io/handlers/httperror"
"github.com/coreos/etcd/client"
"github.com/prometheus/client_golang/prometheus"
)
var newCounter *prometheus.CounterVec
func init() {
newCounter = prometheus.NewCounterVec(
prometheus.CounterOpts{
Name: "endpoint_new_requests_total",
Help: "How many /new requests processed, partitioned by status code and HTTP method.",
},
[]string{"code", "method"},
)
prometheus.MustRegister(newCounter)
}
func generateCluster() string {
b := make([]byte, 16)
_, err := rand.Read(b)
if err != nil {
return ""
}
return hex.EncodeToString(b)
}
func Setup(etcdCURL, disc string) *State {
u, _ := url.Parse(etcdCURL)
return &State{
etcdHost: etcdCURL,
etcdCURL: u,
currentLeader: u.Host,
discHost: disc,
}
}
func (st *State) setupToken(size int) (string, error) {
token := generateCluster()
if token == "" {
return "", errors.New("Couldn't generate a token")
}
c, _ := client.New(client.Config{
Endpoints: []string{st.endpoint()},
Transport: client.DefaultTransport,
// set timeout per request to fail fast when the target endpoint is unavailable
HeaderTimeoutPerRequest: time.Second,
})
kapi := client.NewKeysAPI(c)
key := path.Join("_etcd", "registry", token)
resp, err := kapi.Create(context.Background(), path.Join(key, "_config", "size"), strconv.Itoa(size))
if err != nil {
return "", fmt.Errorf("Couldn't setup state %v %v", resp, err)
}
return token, nil
}
func (st *State) deleteToken(token string) error {
c, _ := client.New(client.Config{
Endpoints: []string{st.endpoint()},
Transport: client.DefaultTransport,
// set timeout per request to fail fast when the target endpoint is unavailable
HeaderTimeoutPerRequest: time.Second,
})
kapi := client.NewKeysAPI(c)
if token == "" {
return errors.New("No token given")
}
_, err := kapi.Delete(
context.Background(),
path.Join("_etcd", "registry", token),
&client.DeleteOptions{Recursive: true},
)
return err
}
func NewTokenHandler(ctx context.Context, w http.ResponseWriter, r *http.Request) {
st := ctx.Value(stateKey).(*State)
var err error
size := 3
s := r.FormValue("size")
if s != "" {
size, err = strconv.Atoi(s)
if err != nil {
httperror.Error(w, r, err.Error(), http.StatusBadRequest, newCounter)
return
}
}
token, err := st.setupToken(size)
if err != nil {
log.Printf("setupToken returned: %v", err)
httperror.Error(w, r, "Unable to generate token", 400, newCounter)
return
}
log.Println("New cluster created", token)
fmt.Fprintf(w, "%s/%s", bytes.TrimRight([]byte(st.discHost), "/"), token)
newCounter.WithLabelValues("200", r.Method).Add(1)
}
|