Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat/ping for wireguard #35

Open
wants to merge 5 commits into
base: main
Choose a base branch
from

Conversation

ainghazal
Copy link
Collaborator

some modifications to make ping work for the wireguard experiment branch

this is needed so that we can pass vpn.Dialer to HTTPConfig as part of
the urlgetter experiment in OONI probe-cli.
@ainghazal
Copy link
Collaborator Author

please note that only the latest commit addresses the wireguard modifications. this branch piles up commits from the security fixes, and is needed for compat reasons, but this will be rebased after the other PRs are merged.

Copy link
Contributor

@bassosimone bassosimone left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have asked a bunch of questions to better understand the diff!

Thank you for working on this! 🙌

if err != nil {
fmt.Println("ERROR", err)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please, do not use fmt.Println to print errors. If there is a desire to log errors, we should have a customizable logger such that we can correctly redirect logs when running on mobile devices. Alternatively, if logging errors is useful when debugging and we do not care about bringing logs to users, I would recommend using the log package in the standard library, which defaults to printing on the standard error. We cannot print on the standard output because that is reserved for communication between ooniprobe and the Desktop app. Any message on the standard output that does not conform to the expected logging message would confuse the desktop app.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

understood, sorry about that. This one was probably a debugging slip; there were other places that were intended to print to console to mimick the ping utility, but this is a library and any output function should live in the cmd entrypoint. fixed.

n, err := p.conn.Read(buf)
if err != nil {
var netErr *net.OpError
if errors.As(err, &netErr) && netErr.Timeout() {
if errors.As(err, &netErr) && netErr.Timeout() || err.Error() == "i/o timeout" {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please, make sure we are matching the end of the error message by using strings.HasSuffix. It's quite common in Go to chain error messages and prefix additional context. For this reason, it would probably be better to try and use strings.HasSuffix. I would also recommend defining the specific error string in a constant rather than using the raw string directly inside the check.

@@ -501,7 +508,7 @@ func (p *Pinger) recvICMP(recv chan<- *packet) error {
select {
case <-p.done:
return nil
case recv <- &packet{bytes: buf, nbytes: n}:
case recv <- &packet{bytes: buf[:], nbytes: n}:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The buf[:] idiom creates a new slice using the same underlying array. If the intent is to copy the array to make sure it is not otherwise modified, I think you want append([]byte{}, buf...).

Please, see this playground for more information about the difference between the two idioms.

Assuming the intent here is to duplicate the bytes, I would like to better understand why. Because we are not aiming for high high performance, I would say we should just allocate a new buffer for each message and accept that we transfer its ownership when we deliver it over a channel.

(You may have seen the buf[:] idiom being used when the backing buffer is a var buf [256]byte rather than a buf := make([]byte. 256). The reason why it's necessary to use buf[:] in such a case is that you have a function that accepts a []byte - i.e., a slice - and so you need to convert from an array to a slice, and you end up doing that using buf[:].)

@@ -572,15 +585,47 @@ func (p *Pinger) processPacket(recv *packet) error {
return nil
}

func (p *Pinger) parseEchoReplyFromICMP(data []byte, from *net.IP) *Packet {
replyPacket, err := icmp.ParseMessage(1, data[:])
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same comment as above

func (p *Pinger) parseEchoReplyFromICMP(data []byte, from *net.IP) *Packet {
replyPacket, err := icmp.ParseMessage(1, data[:])
if err != nil {
log.Printf("error: %v\n", err)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for using log here. I'd like to encourage you to replace fmt.Printf with log.Printf throughout the code, otherwise we'll have troubles when we integrate this code with ooniprobe.

replyPacket, err := icmp.ParseMessage(1, data[:])
if err != nil {
log.Printf("error: %v\n", err)
return &Packet{}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't like this pattern much; I'd rather return *Packet, error or, if the error does not matter, a *Packet, bool mainly because I think we should leverage the fact that Go allows returning multiple values to avoid returning special-case canary values.

if err := parser.DecodeLayers(data, &decoded); err != nil {
log.Printf("error: %v\n", err)
return &Packet{}
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I admit I am a bit lost in what you are actually doing here. Would you mind adding some comments to the code explaining why you're parsing an ICMP packet and then apparently parsing it again a second time?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It would also be useful to know, more in general, why we need a different parsing strategy with wireguard and we should perhaps consider whether we can adopt the same strategy also when using openvpn.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also, I am kinda lost in this thinking about why we're performing wireguard related changes in this library which does not seem to otherwise deal with wireguard. Is this because we're reusing openvpn code for wireguard?

eventListener chan uint16
eventListener chan uint8

failed bool
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what does this field do?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this is an unrelated change, I will force-push (moved to #38)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants