-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhandlers.go
130 lines (114 loc) · 3.36 KB
/
handlers.go
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
124
125
126
127
128
129
130
package main
import (
"encoding/json"
"io/ioutil"
"log"
"net/http"
"strconv"
"github.com/gorilla/mux"
"github.com/xeipuuv/gojsonschema"
)
// Run contains the data needed for a "run" on a given lab instrument.
type Run struct {
InstrumentID int
Samples []Sample
}
// Sample represents the data items that will be analysed by a lab instrument.
type Sample struct {
ID int `json:"id"`
}
// ValidationError provides a data structure for 400 error responses.
type ValidationError struct {
Errors []string `json:"errors"`
}
// RunSuccessResponse defines the response for a new run, providing the new RunID.
type RunSuccessResponse struct {
RunID int `json:"runId"`
}
// Handlers is the receiver for methods that handle http requests.
// It will provide dependencies. Use NewHandlers() for construction.
type Handlers struct {
DB DB
Schema Schema
}
// NewHandlers is the constructor for Handlers. Unit tests may provide
// an alternative constructor to "inject" alternative dependencies.
func NewHandlers(db DB, schema Schema) Handlers {
h := Handlers{DB: db, Schema: schema}
return h
}
// RespondAsJSON has the boilerplate for the most typical responses.
func RespondAsJSON(payload interface{}, statusCode int, w http.ResponseWriter) {
json, err := json.Marshal(payload)
if err != nil {
log.Println("Error marshalling payload", err.Error())
w.WriteHeader(500)
return
}
w.Header().Set("Content-Type", "application/json")
w.WriteHeader(statusCode)
w.Write(json)
}
// Validate handles the details of validating a payload (request body) with a
// JSON Schema and building an error report. If valid is true, ignore the
// ValidationError object. If valid is false, return the ValidationError as
// a 400 (Bad Request) response.
func Validate(payload []byte, schema *gojsonschema.Schema) (valid bool, errors ValidationError) {
jsonLoader := gojsonschema.NewBytesLoader(payload)
result, err := schema.Validate(jsonLoader)
if err != nil {
v := ValidationError{[]string{"Malformed JSON"}}
return false, v
}
if !result.Valid() {
errors := make([]string, len(result.Errors()))
for i, desc := range result.Errors() {
errors[i] = desc.Description()
}
v := ValidationError{errors}
return false, v
}
return true, ValidationError{}
}
// Post is the request handler for submitting samples.
func (h Handlers) Post(w http.ResponseWriter, r *http.Request) {
// Extract instrumentID from URL.
vars := mux.Vars(r)
unparsedInstrumentID := vars["instrument_id"]
instrumentID, err := strconv.Atoi(unparsedInstrumentID)
if err != nil {
v := ValidationError{[]string{"Instrument ID in URL must be an integer."}}
RespondAsJSON(v, 400, w)
return
}
// Read body
defer r.Body.Close()
body, err := ioutil.ReadAll(r.Body)
if err != nil {
http.Error(w, err.Error(), 500)
return
}
// Validate
valid, errors := Validate(body, h.Schema.Samples)
if !valid {
RespondAsJSON(errors, 400, w)
return
}
// Data is good, we can unmarshal.
samples := make([]Sample, 0)
err = json.Unmarshal(body, &samples)
if err != nil {
http.Error(w, err.Error(), 500)
return
}
// Execute run (just saving to db for the scope of this exercise).
run := Run{instrumentID, samples}
runID, err := h.DB.SaveRun(run)
if err != nil {
http.Error(w, err.Error(), 500)
return
}
// Respond with newly created run ID.
success := RunSuccessResponse{runID}
RespondAsJSON(success, 200, w)
}