aboutsummaryrefslogtreecommitdiff
path: root/monitor.go
blob: c6d883d35f4024ced913a82bd8bf25c0e6c615da (plain) (blame)
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
package main

import (
	"net"
	"time"
)

func monitor4(conf *config, update4 chan<- net.IPAddr) {
	refresh := time.NewTicker(conf.Interval)

	var prevAddr4 *net.IPAddr
	for range refresh.C {
		link, err := net.InterfaceByName(conf.Link4)
		if err != nil {
			logger.Printf("can't find interface %s: %w", conf.Link4, err)
			continue
		}

		addrs, err := link.Addrs()
		if err != nil {
			const format = "can't read IPv4 addresses from %s: %w"
			logger.Printf(format, conf.Link4, err)
			continue
		}

		var addr4 *net.IPAddr
		for _, netAddr := range addrs {
			addr := netAddr.(*net.IPAddr)

			if addr.To4() != nil && !addr.IP.IsPrivate() {
				addr4 = addr
				break
			}
		}

		if !addr4.IP.Equal(prevAddr4) {
			logger.Println("detected new IPv4 address:", addr4)
			update4 <- addr4

			prevAddr4 = addr4
		}
	}
}

func monitor6(conf *config, update6 chan<- net.IPNet) {
	refresh := time.NewTicker(conf.Interval)

	var prevPrefix6 *net.IPNet
	for range refresh.C {
		link, err := net.InterfaceByName(conf.Link6)
		if err != nil {
			logger.Printf("can't find interface %s: %w", conf.Link6, err)
			continue
		}

		addrs, err := link.Addrs()
		if err != nil {
			const format = "can't read IPv6 addresses from %s: %w"
			logger.Printf(format, conf.Link6, err)
			continue
		}

		var prefix6 *net.IPNet
		for _, netAddr := range addrs {
			addr := netAddr.(*net.IPAddr)

			if addr.To4() == nil && addr.IP.IsGlobalUnicast() {
				cidr := net.CIDRMask(conf.PrefixLen, 128)
				prefix6 = addr.IP.Mask(cidr)

				break
			}
		}

		if !prefix6.Equal(prevPrefix6) {
			logger.Println("detected new IPv6 address:", prefix6)
			update6 <- prefix6

			prevPrefix6 = prefix6
		}
	}
}