Skip to content

Commit

Permalink
Fix DNS resolver "additional answers" logic
Browse files Browse the repository at this point in the history
- Guard against nil pointer dereference
- Loop until all additional answers are drained
  • Loading branch information
rod-hynes committed Aug 19, 2022
1 parent b8c5c19 commit cb4e5e2
Showing 1 changed file with 15 additions and 5 deletions.
20 changes: 15 additions & 5 deletions psiphon/common/resolver/resolver.go
Original file line number Diff line number Diff line change
Expand Up @@ -838,11 +838,21 @@ func (r *Resolver) ResolveIP(
// arrived concurrent with the first answer. This receive avoids a race
// condition where inFlight may now be 0, with additional answers
// enqueued, in which case the following await branch is not taken.
select {
case nextAnswer := <-answerChan:
result.IPs = append(result.IPs, nextAnswer.IPs...)
result.TTLs = append(result.TTLs, nextAnswer.TTLs...)
default:
//
// It's possible for the attempts loop to exit with no received answer due
// to timeouts or cancellation while, concurrently, an answer is sent to
// the channel. In this case, when result == nil, we ignore the answers
// and leave this as a failed resolve.
if result != nil {
for loop := true; loop; {
select {
case nextAnswer := <-answerChan:
result.IPs = append(result.IPs, nextAnswer.IPs...)
result.TTLs = append(result.TTLs, nextAnswer.TTLs...)
default:
loop = false
}
}
}

// When we have an answer, await -- for a short time,
Expand Down

0 comments on commit cb4e5e2

Please sign in to comment.