Skip to content

Commit

Permalink
Merge pull request #3316 from getlantern/issue3315
Browse files Browse the repository at this point in the history
Cleaner approach to dual fetching closes #3315
  • Loading branch information
fffw committed Nov 18, 2015
2 parents ba0a9fe + 8c411b1 commit f750e8e
Showing 1 changed file with 33 additions and 29 deletions.
62 changes: 33 additions & 29 deletions src/github.com/getlantern/flashlight/util/http.go
Original file line number Diff line number Diff line change
Expand Up @@ -145,8 +145,8 @@ func (df *dualFetcher) Do(req *http.Request) (*http.Response, error) {

// Create channels for the final response or error. The response channel will be filled
// in the case of any successful response as well as a non-error response for the second
// response received. The error channel will only be filled if the first response non-successful
// and the second is an error.
// response received. The error channel will only be filled if the first response is
// unsuccessful and the second is an error.
finalResponseCh := make(chan *http.Response, 1)
finalErrorCh := make(chan error, 1)

Expand All @@ -161,38 +161,42 @@ func (df *dualFetcher) Do(req *http.Request) (*http.Response, error) {
}

func readResponses(finalResponse chan *http.Response, responses chan *http.Response, finalErr chan error, errs chan error) {
for i := 0; i < 2; i++ {
select {
case resp := <-responses:
if i == 1 {
log.Debug("Got second response -- sending")
finalResponse <- resp
} else if success(resp) {
log.Debug("Got good response")
finalResponse <- resp
select {
case <-responses:
log.Debug("Closing second response body")
_ = resp.Body.Close()
return
case <-errs:
log.Debug("Ignoring error on second response")
return
}
} else {
log.Debugf("Got bad first response -- wait for second")
// Note that the caller is responsible for closing the
// response body of the response they receive.
select {
case resp := <-responses:
if success(resp) {
log.Debug("Got good first response")
finalResponse <- resp

// Just ignore the second response, but still process it.
select {
case resp := <-responses:
log.Debug("Closing second response body")
_ = resp.Body.Close()
return
case <-errs:
log.Debug("Ignoring error on second response")
return
}
case err := <-errs:
log.Debugf("Got an error: %v", err)
if i == 1 {
// In this case all requests have errored, so our final response
// is an error.
} else {
log.Debugf("Got bad first response -- wait for second")
_ = resp.Body.Close()
// Just use whatever we get from the second response.
select {
case resp := <-responses:
finalResponse <- resp
case err := <-errs:
finalErr <- err
}
}
case err := <-errs:
log.Debugf("Got an error: %v", err)
// Just use whatever we get from the second response.
select {
case resp := <-responses:
finalResponse <- resp
case err := <-errs:
finalErr <- err
}
}
}

Expand Down

0 comments on commit f750e8e

Please sign in to comment.