-
Notifications
You must be signed in to change notification settings - Fork 17
/
handle_request.go
79 lines (66 loc) · 2.37 KB
/
handle_request.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
// Copyright 2023 Princess B33f Heavy Industries / Dave Shanley
// SPDX-License-Identifier: MIT
package daemon
import (
"github.com/pb33f/libopenapi-validator/parameters"
"github.com/pb33f/libopenapi-validator/requests"
"github.com/pb33f/libopenapi-validator/responses"
"github.com/pb33f/ranch/model"
"github.com/pb33f/ranch/service"
"github.com/pb33f/wiretap/shared"
"io"
"net/http"
"time"
)
func (ws *WiretapService) handleHttpRequest(request *model.Request, core service.FabricServiceCore) {
lowResponseChan := make(chan *http.Response)
lowErrorChan := make(chan error)
var returnedResponse *http.Response
var returnedError error
// create validators.
requestValidator := requests.NewRequestBodyValidator(ws.docModel)
paramValidator := parameters.NewParameterValidator(ws.docModel)
responseValidator := responses.NewResponseBodyValidator(ws.docModel)
// validate the request
go ws.validateRequest(request, requestValidator, paramValidator, responseValidator)
// call the API being requested.
go ws.callAPI(request.HttpRequest, lowResponseChan, lowErrorChan)
doneWaitingForResponse:
for {
select {
case resp, ok := <-lowResponseChan:
if ok {
returnedResponse = resp
}
break doneWaitingForResponse
case err := <-lowErrorChan:
returnedError = err
break doneWaitingForResponse
}
}
if returnedResponse == nil && returnedError != nil {
go ws.broadcastResponseError(request, cloneResponse(returnedResponse), returnedError)
core.SendErrorResponse(request, 500, returnedError.Error())
return
} else {
// validate response
go ws.validateResponse(request, responseValidator, cloneResponse(returnedResponse))
}
// send response back to client.
go func() {
config := ws.controlsStore.GetValue(shared.ConfigKey).(*shared.WiretapConfiguration)
if config.GlobalAPIDelay > 0 {
time.Sleep(time.Duration(config.GlobalAPIDelay) * time.Millisecond) // simulate a slow response.
}
body, _ := io.ReadAll(returnedResponse.Body)
headers := extractHeaders(returnedResponse)
// wiretap needs to work from anywhere, so allow everything.
headers["Access-Control-Allow-Origin"] = "*"
if returnedResponse.StatusCode >= 400 {
core.SendErrorResponseAsStringWithHeadersAndPayload(request, returnedResponse.StatusCode,
"HTTP Request failed", string(body), headers)
} else {
core.SendResponseAsStringWithHeaders(request, string(body), headers)
}
}()
}