File: transfer.go

package info (click to toggle)
golang-github-mdlayher-ndp 1.1.0-1
  • links: PTS, VCS
  • area: main
  • in suites: experimental, forky, sid, trixie
  • size: 244 kB
  • sloc: makefile: 3
file content (121 lines) | stat: -rw-r--r-- 2,400 bytes parent folder | download | duplicates (2)
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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
package ndpcmd

import (
	"context"
	"errors"
	"fmt"
	"log"
	"net"
	"net/netip"
	"time"

	"github.com/mdlayher/ndp"
)

func sendReceiveLoop(
	ctx context.Context,
	c *ndp.Conn,
	ll *log.Logger,
	m ndp.Message,
	dst netip.Addr,
	check func(m ndp.Message) bool,
) error {
	for i := 0; ; i++ {
		msg, from, err := sendReceive(ctx, c, m, dst, check)
		switch err {
		case context.Canceled:
			fmt.Println()
			ll.Printf("canceled, sent %d message(s)", i+1)
			return err
		case errRetry:
			fmt.Print(".")
			continue
		case nil:
			fmt.Println()
			printMessage(ll, msg, from)
			return nil
		default:
			return err
		}
	}
}

func receiveLoop(
	ctx context.Context,
	c *ndp.Conn,
	ll *log.Logger,
	check func(m ndp.Message) bool,
	recv func(ll *log.Logger, msg ndp.Message, from netip.Addr),
) error {
	if recv == nil {
		recv = printMessage
	}

	var count int
	for {
		msg, from, err := receive(ctx, c, check)
		switch err {
		case context.Canceled:
			ll.Printf("received %d message(s)", count)
			return nil
		case errRetry:
			continue
		case nil:
			count++
			recv(ll, msg, from)
		default:
			return err
		}
	}
}

var errRetry = errors.New("retry")

func sendReceive(
	ctx context.Context,
	c *ndp.Conn,
	m ndp.Message,
	dst netip.Addr,
	check func(m ndp.Message) bool,
) (ndp.Message, netip.Addr, error) {
	if err := c.WriteTo(m, nil, dst); err != nil {
		return nil, netip.Addr{}, fmt.Errorf("failed to write message: %v", err)
	}

	return receive(ctx, c, check)
}

func receive(
	ctx context.Context,
	c *ndp.Conn,
	check func(m ndp.Message) bool,
) (ndp.Message, netip.Addr, error) {
	if err := c.SetReadDeadline(time.Now().Add(1 * time.Second)); err != nil {
		return nil, netip.Addr{}, fmt.Errorf("failed to set deadline: %v", err)
	}

	msg, _, from, err := c.ReadFrom()
	if err == nil {
		if check != nil && !check(msg) {
			// Read a message, but it isn't the one we want.  Keep trying.
			return nil, netip.Addr{}, errRetry
		}

		// Got a message that passed the check, if check was not nil.
		return msg, from, nil
	}

	// Was the context canceled already?
	select {
	case <-ctx.Done():
		return nil, netip.Addr{}, ctx.Err()
	default:
	}

	// Was the error caused by a read timeout, and should the loop continue?
	if nerr, ok := err.(net.Error); ok && nerr.Timeout() {
		return nil, netip.Addr{}, errRetry
	}

	return nil, netip.Addr{}, fmt.Errorf("failed to read message: %v", err)
}