| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 1 | package main |
| 2 | |
| 3 | import ( |
| giolekva | 88d6e35 | 2020-04-30 13:32:38 +0400 | [diff] [blame] | 4 | "errors" |
| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 5 | "flag" |
| 6 | "fmt" |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 7 | "io" |
| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 8 | "io/ioutil" |
| 9 | "log" |
| 10 | "net/http" |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 11 | |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 12 | "k8s.io/client-go/kubernetes" |
| 13 | corev1 "k8s.io/client-go/kubernetes/typed/core/v1" |
| 14 | "k8s.io/client-go/rest" |
| 15 | "k8s.io/client-go/tools/clientcmd" |
| giolekva | c76b21b | 2020-04-18 19:28:43 +0400 | [diff] [blame] | 16 | |
| 17 | "github.com/giolekva/pcloud/controller/schema" |
| 18 | |
| 19 | "github.com/golang/glog" |
| 20 | "github.com/itaysk/regogo" |
| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 21 | ) |
| 22 | |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 23 | var kubeconfig = flag.String("kubeconfig", "", "Absolute path to the kubeconfig file.") |
| 24 | |
| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 25 | var port = flag.Int("port", 123, "Port to listen on.") |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 26 | var dgraphGqlAddress = flag.String("graphql_address", "", "GraphQL server address.") |
| 27 | var dgraphSchemaAddress = flag.String("dgraph_admin_address", "", "Dgraph server admin address.") |
| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 28 | |
| giolekva | c76b21b | 2020-04-18 19:28:43 +0400 | [diff] [blame] | 29 | const imgJson = `{ objectPath: \"%s\"}` |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 30 | const insertQuery = `mutation { add%s(input: [%s]) { %s { id } } }` |
| giolekva | c76b21b | 2020-04-18 19:28:43 +0400 | [diff] [blame] | 31 | const getQuery = `{ "query": "{ get%s(id: \"%s\") { id objectPath } } " }` |
| 32 | |
| giolekva | c76b21b | 2020-04-18 19:28:43 +0400 | [diff] [blame] | 33 | type MinioWebhook struct { |
| giolekva | 8fe32df | 2020-05-12 13:31:46 +0400 | [diff] [blame^] | 34 | gql schema.GraphQLClient |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 35 | } |
| 36 | |
| giolekva | 88d6e35 | 2020-04-30 13:32:38 +0400 | [diff] [blame] | 37 | type query struct { |
| 38 | query string |
| 39 | operation string |
| 40 | variables string |
| 41 | } |
| 42 | |
| 43 | func extractQuery(r *http.Request) (*query, error) { |
| 44 | if r.Method == "GET" { |
| 45 | if err := r.ParseForm(); err != nil { |
| 46 | return nil, err |
| 47 | } |
| 48 | q, ok := r.Form["query"] |
| 49 | if !ok || len(q) != 1 { |
| 50 | return nil, errors.New("Exactly one query must be provided") |
| 51 | } |
| 52 | return &query{query: q[0]}, nil |
| 53 | } else { |
| 54 | body, err := ioutil.ReadAll(r.Body) |
| 55 | if err != nil { |
| 56 | return nil, err |
| 57 | } |
| 58 | q, err := regogo.Get(string(body), "input.query") |
| 59 | if err != nil { |
| 60 | return nil, err |
| 61 | } |
| 62 | return &query{query: q.String()}, nil |
| 63 | } |
| 64 | } |
| 65 | |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 66 | func (m *MinioWebhook) graphqlHandler(w http.ResponseWriter, r *http.Request) { |
| 67 | glog.Infof("New GraphQL query received: %s", r.Method) |
| giolekva | 88d6e35 | 2020-04-30 13:32:38 +0400 | [diff] [blame] | 68 | q, err := extractQuery(r) |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 69 | if err != nil { |
| giolekva | 88d6e35 | 2020-04-30 13:32:38 +0400 | [diff] [blame] | 70 | glog.Error(err.Error()) |
| 71 | http.Error(w, "Could not extract query", http.StatusBadRequest) |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 72 | } |
| giolekva | 88d6e35 | 2020-04-30 13:32:38 +0400 | [diff] [blame] | 73 | resp, err := m.gql.RunQuery(q.query) |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 74 | if err != nil { |
| 75 | glog.Error(err) |
| 76 | http.Error(w, err.Error(), http.StatusInternalServerError) |
| 77 | return |
| 78 | } |
| 79 | io.WriteString(w, resp) |
| 80 | w.Header().Set("Content-Type", "application/json") |
| 81 | } |
| 82 | |
| 83 | func getKubeConfig() (*rest.Config, error) { |
| 84 | if *kubeconfig != "" { |
| 85 | return clientcmd.BuildConfigFromFlags("", *kubeconfig) |
| 86 | } else { |
| 87 | return rest.InClusterConfig() |
| 88 | } |
| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 89 | } |
| 90 | |
| 91 | func main() { |
| 92 | flag.Parse() |
| giolekva | c76b21b | 2020-04-18 19:28:43 +0400 | [diff] [blame] | 93 | |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 94 | gqlClient, err := schema.NewDgraphClient( |
| 95 | *dgraphGqlAddress, *dgraphSchemaAddress) |
| 96 | if err != nil { |
| 97 | panic(err) |
| 98 | } |
| giolekva | 26a8b5f | 2020-05-01 20:01:13 +0400 | [diff] [blame] | 99 | err = gqlClient.SetSchema(` |
| 100 | enum EventState { |
| 101 | NEW |
| 102 | PROCESSING |
| 103 | DONE |
| 104 | } |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 105 | |
| giolekva | 26a8b5f | 2020-05-01 20:01:13 +0400 | [diff] [blame] | 106 | type Foo { bar: Int }`) |
| 107 | if err != nil { |
| 108 | panic(err) |
| 109 | } |
| 110 | err = gqlClient.AddSchema(` |
| 111 | type Image { |
| 112 | id: ID! |
| 113 | objectPath: String! @search(by: [exact]) |
| 114 | } |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 115 | |
| giolekva | 26a8b5f | 2020-05-01 20:01:13 +0400 | [diff] [blame] | 116 | type ImageSegment { |
| 117 | id: ID! |
| 118 | upperLeftX: Float! |
| 119 | upperLeftY: Float! |
| 120 | lowerRightX: Float! |
| 121 | lowerRightY: Float! |
| 122 | sourceImage: Image! @hasInverse(field: segments) |
| 123 | } |
| 124 | |
| 125 | extend type Image { |
| 126 | segments: [ImageSegment] @hasInverse(field: sourceImage) |
| 127 | }`) |
| 128 | if err != nil { |
| 129 | panic(err) |
| 130 | } |
| giolekva | 8fe32df | 2020-05-12 13:31:46 +0400 | [diff] [blame^] | 131 | mw := MinioWebhook{gqlClient} |
| giolekva | fb52e0d | 2020-04-23 22:52:13 +0400 | [diff] [blame] | 132 | http.HandleFunc("/graphql", mw.graphqlHandler) |
| giolekva | 07f6be9 | 2020-04-16 21:09:30 +0400 | [diff] [blame] | 133 | log.Fatal(http.ListenAndServe(fmt.Sprintf(":%d", *port), nil)) |
| 134 | } |