forked from TrueCloudLab/certificates
api/render: initial implementation of the package
This commit is contained in:
parent
80abda22ee
commit
a31feae6d4
2 changed files with 111 additions and 0 deletions
58
api/render/render.go
Normal file
58
api/render/render.go
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
// Package render implements functionality related to response rendering.
|
||||||
|
package render
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"google.golang.org/protobuf/encoding/protojson"
|
||||||
|
"google.golang.org/protobuf/proto"
|
||||||
|
|
||||||
|
"github.com/smallstep/certificates/api/log"
|
||||||
|
)
|
||||||
|
|
||||||
|
// JSON writes the passed value into the http.ResponseWriter.
|
||||||
|
func JSON(w http.ResponseWriter, v interface{}) {
|
||||||
|
JSONStatus(w, v, http.StatusOK)
|
||||||
|
}
|
||||||
|
|
||||||
|
// JSONStatus writes the given value into the http.ResponseWriter and the
|
||||||
|
// given status is written as the status code of the response.
|
||||||
|
func JSONStatus(w http.ResponseWriter, v interface{}, status int) {
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
w.WriteHeader(status)
|
||||||
|
if err := json.NewEncoder(w).Encode(v); err != nil {
|
||||||
|
log.Error(w, err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
log.EnabledResponse(w, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ProtoJSON writes the passed value into the http.ResponseWriter.
|
||||||
|
func ProtoJSON(w http.ResponseWriter, m proto.Message) {
|
||||||
|
ProtoJSONStatus(w, m, http.StatusOK)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ProtoJSONStatus writes the given value into the http.ResponseWriter and the
|
||||||
|
// given status is written as the status code of the response.
|
||||||
|
func ProtoJSONStatus(w http.ResponseWriter, m proto.Message, status int) {
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
w.WriteHeader(status)
|
||||||
|
|
||||||
|
b, err := protojson.Marshal(m)
|
||||||
|
if err != nil {
|
||||||
|
log.Error(w, err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := w.Write(b); err != nil {
|
||||||
|
log.Error(w, err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// log.EnabledResponse(w, v)
|
||||||
|
}
|
53
api/render/render_test.go
Normal file
53
api/render/render_test.go
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
package render
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/smallstep/certificates/logging"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestJSON(t *testing.T) {
|
||||||
|
type args struct {
|
||||||
|
rw http.ResponseWriter
|
||||||
|
v interface{}
|
||||||
|
}
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
args args
|
||||||
|
ok bool
|
||||||
|
}{
|
||||||
|
{"ok", args{httptest.NewRecorder(), map[string]interface{}{"foo": "bar"}}, true},
|
||||||
|
{"fail", args{httptest.NewRecorder(), make(chan int)}, false},
|
||||||
|
}
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
rw := logging.NewResponseLogger(tt.args.rw)
|
||||||
|
JSON(rw, tt.args.v)
|
||||||
|
|
||||||
|
rr, ok := tt.args.rw.(*httptest.ResponseRecorder)
|
||||||
|
if !ok {
|
||||||
|
t.Error("ResponseWriter does not implement *httptest.ResponseRecorder")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
fields := rw.Fields()
|
||||||
|
if tt.ok {
|
||||||
|
if body := rr.Body.String(); body != "{\"foo\":\"bar\"}\n" {
|
||||||
|
t.Errorf(`Unexpected body = %v, want {"foo":"bar"}`, body)
|
||||||
|
}
|
||||||
|
if len(fields) != 0 {
|
||||||
|
t.Errorf("ResponseLogger fields = %v, wants 0 elements", fields)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if body := rr.Body.String(); body != "" {
|
||||||
|
t.Errorf("Unexpected body = %s, want empty string", body)
|
||||||
|
}
|
||||||
|
if len(fields) != 1 {
|
||||||
|
t.Errorf("ResponseLogger fields = %v, wants 1 element", fields)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue