| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424 |
- // Copyright 2019 the Kilo authors
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package mesh
- import (
- "errors"
- "net"
- "sort"
- "github.com/go-kit/kit/log"
- "github.com/go-kit/kit/log/level"
- "github.com/kilo-io/kilo/pkg/wireguard"
- )
- const (
- logicalLocationPrefix = "location:"
- nodeLocationPrefix = "node:"
- )
- // Topology represents the logical structure of the overlay network.
- type Topology struct {
- // key is the private key of the node creating the topology.
- key []byte
- port uint32
- // Location is the logical location of the local host.
- location string
- segments []*segment
- peers []*Peer
- // hostname is the hostname of the local host.
- hostname string
- // leader represents whether or not the local host
- // is the segment leader.
- leader bool
- // persistentKeepalive is the interval in seconds of the emission
- // of keepalive packets by the local node to its peers.
- persistentKeepalive int
- // privateIP is the private IP address of the local node.
- privateIP *net.IPNet
- // subnet is the Pod subnet of the local node.
- subnet *net.IPNet
- // wireGuardCIDR is the allocated CIDR of the WireGuard
- // interface of the local node within the Kilo subnet.
- // If the local node is not the leader of a location, then
- // the IP is the 0th address in the subnet, i.e. the CIDR
- // is equal to the Kilo subnet.
- wireGuardCIDR *net.IPNet
- // discoveredEndpoints is the updated map of valid discovered Endpoints
- discoveredEndpoints map[string]*wireguard.Endpoint
- logger log.Logger
- }
- type segment struct {
- allowedIPs []*net.IPNet
- endpoint *wireguard.Endpoint
- key []byte
- persistentKeepalive int
- // Location is the logical location of this segment.
- location string
- // cidrs is a slice of subnets of all peers in the segment.
- cidrs []*net.IPNet
- // hostnames is a slice of the hostnames of the peers in the segment.
- hostnames []string
- // leader is the index of the leader of the segment.
- leader int
- // privateIPs is a slice of private IPs of all peers in the segment.
- privateIPs []net.IP
- // wireGuardIP is the allocated IP address of the WireGuard
- // interface on the leader of the segment.
- wireGuardIP net.IP
- // allowedLocationIPs are not part of the cluster and are not peers.
- // They are directly routable from nodes within the segment.
- // A classic example is a printer that ought to be routable from other locations.
- allowedLocationIPs []*net.IPNet
- }
- // NewTopology creates a new Topology struct from a given set of nodes and peers.
- func NewTopology(nodes map[string]*Node, peers map[string]*Peer, granularity Granularity, hostname string, port uint32, key []byte, subnet *net.IPNet, persistentKeepalive int, logger log.Logger) (*Topology, error) {
- if logger == nil {
- logger = log.NewNopLogger()
- }
- topoMap := make(map[string][]*Node)
- for _, node := range nodes {
- var location string
- switch granularity {
- case LogicalGranularity:
- location = logicalLocationPrefix + node.Location
- // Put node in a different location, if no private
- // IP was found.
- if node.InternalIP == nil {
- location = nodeLocationPrefix + node.Name
- }
- case FullGranularity:
- location = nodeLocationPrefix + node.Name
- }
- topoMap[location] = append(topoMap[location], node)
- }
- var localLocation string
- switch granularity {
- case LogicalGranularity:
- localLocation = logicalLocationPrefix + nodes[hostname].Location
- if nodes[hostname].InternalIP == nil {
- localLocation = nodeLocationPrefix + hostname
- }
- case FullGranularity:
- localLocation = nodeLocationPrefix + hostname
- }
- t := Topology{key: key, port: port, hostname: hostname, location: localLocation, persistentKeepalive: persistentKeepalive, privateIP: nodes[hostname].InternalIP, subnet: nodes[hostname].Subnet, wireGuardCIDR: subnet, discoveredEndpoints: make(map[string]*wireguard.Endpoint), logger: logger}
- for location := range topoMap {
- // Sort the location so the result is stable.
- sort.Slice(topoMap[location], func(i, j int) bool {
- return topoMap[location][i].Name < topoMap[location][j].Name
- })
- leader := findLeader(topoMap[location])
- if location == localLocation && topoMap[location][leader].Name == hostname {
- t.leader = true
- }
- var allowedIPs []*net.IPNet
- allowedLocationIPsMap := make(map[string]struct{})
- var allowedLocationIPs []*net.IPNet
- var cidrs []*net.IPNet
- var hostnames []string
- var privateIPs []net.IP
- for _, node := range topoMap[location] {
- // Allowed IPs should include:
- // - the node's allocated subnet
- // - the node's WireGuard IP
- // - the node's internal IP
- // - IPs that were specified by the allowed-location-ips annotation
- allowedIPs = append(allowedIPs, node.Subnet)
- for _, ip := range node.AllowedLocationIPs {
- if _, ok := allowedLocationIPsMap[ip.String()]; !ok {
- allowedLocationIPs = append(allowedLocationIPs, ip)
- allowedLocationIPsMap[ip.String()] = struct{}{}
- }
- }
- if node.InternalIP != nil {
- allowedIPs = append(allowedIPs, oneAddressCIDR(node.InternalIP.IP))
- privateIPs = append(privateIPs, node.InternalIP.IP)
- }
- cidrs = append(cidrs, node.Subnet)
- hostnames = append(hostnames, node.Name)
- }
- // The sorting has no function, but makes testing easier.
- sort.Slice(allowedLocationIPs, func(i, j int) bool {
- return allowedLocationIPs[i].String() < allowedLocationIPs[j].String()
- })
- t.segments = append(t.segments, &segment{
- allowedIPs: allowedIPs,
- endpoint: topoMap[location][leader].Endpoint,
- key: topoMap[location][leader].Key,
- persistentKeepalive: topoMap[location][leader].PersistentKeepalive,
- location: location,
- cidrs: cidrs,
- hostnames: hostnames,
- leader: leader,
- privateIPs: privateIPs,
- allowedLocationIPs: allowedLocationIPs,
- })
- }
- // Sort the Topology segments so the result is stable.
- sort.Slice(t.segments, func(i, j int) bool {
- return t.segments[i].location < t.segments[j].location
- })
- for _, peer := range peers {
- t.peers = append(t.peers, peer)
- }
- // Sort the Topology peers so the result is stable.
- sort.Slice(t.peers, func(i, j int) bool {
- return t.peers[i].Name < t.peers[j].Name
- })
- // We need to defensively deduplicate peer allowed IPs. If two peers claim the same IP,
- // the WireGuard configuration could flap, causing the interface to churn.
- t.peers = deduplicatePeerIPs(t.peers)
- // Copy the host node DiscoveredEndpoints in the topology as a starting point.
- for key := range nodes[hostname].DiscoveredEndpoints {
- t.discoveredEndpoints[key] = nodes[hostname].DiscoveredEndpoints[key]
- }
- // Allocate IPs to the segment leaders in a stable, coordination-free manner.
- a := newAllocator(*subnet)
- for _, segment := range t.segments {
- ipNet := a.next()
- if ipNet == nil {
- return nil, errors.New("failed to allocate an IP address; ran out of IP addresses")
- }
- segment.wireGuardIP = ipNet.IP
- segment.allowedIPs = append(segment.allowedIPs, oneAddressCIDR(ipNet.IP))
- if t.leader && segment.location == t.location {
- t.wireGuardCIDR = &net.IPNet{IP: ipNet.IP, Mask: subnet.Mask}
- }
- // Now that the topology is ordered, update the discoveredEndpoints map
- // add new ones by going through the ordered topology: segments, nodes
- for _, node := range topoMap[segment.location] {
- for key := range node.DiscoveredEndpoints {
- if _, ok := t.discoveredEndpoints[key]; !ok {
- t.discoveredEndpoints[key] = node.DiscoveredEndpoints[key]
- }
- }
- }
- // Check for intersecting IPs in allowed location IPs
- segment.allowedLocationIPs = t.filterAllowedLocationIPs(segment.allowedLocationIPs, segment.location)
- }
- return &t, nil
- }
- func intersect(n1, n2 *net.IPNet) bool {
- return n1.Contains(n2.IP) || n2.Contains(n1.IP)
- }
- func (t *Topology) filterAllowedLocationIPs(ips []*net.IPNet, location string) (ret []*net.IPNet) {
- CheckIPs:
- for _, ip := range ips {
- for _, s := range t.segments {
- // Check if allowed location IPs are also allowed in other locations.
- if location != s.location {
- for _, i := range s.allowedLocationIPs {
- if intersect(ip, i) {
- level.Warn(t.logger).Log("msg", "overlapping allowed location IPnets", "IP", ip.String(), "IP2", i.String(), "segment-location", s.location)
- continue CheckIPs
- }
- }
- }
- // Check if allowed location IPs intersect with the allowed IPs.
- for _, i := range s.allowedIPs {
- if intersect(ip, i) {
- level.Warn(t.logger).Log("msg", "overlapping allowed location IPnet with allowed IPnets", "IP", ip.String(), "IP2", i.String(), "segment-location", s.location)
- continue CheckIPs
- }
- }
- // Check if allowed location IPs intersect with the private IPs of the segment.
- for _, i := range s.privateIPs {
- if ip.Contains(i) {
- level.Warn(t.logger).Log("msg", "overlapping allowed location IPnet with privateIP", "IP", ip.String(), "IP2", i.String(), "segment-location", s.location)
- continue CheckIPs
- }
- }
- }
- // Check if allowed location IPs intersect with allowed IPs of peers.
- for _, p := range t.peers {
- for _, i := range p.AllowedIPs {
- if intersect(ip, i) {
- level.Warn(t.logger).Log("msg", "overlapping allowed location IPnet with peer IPnet", "IP", ip.String(), "IP2", i.String(), "peer", p.Name)
- continue CheckIPs
- }
- }
- }
- ret = append(ret, ip)
- }
- return
- }
- func (t *Topology) updateEndpoint(endpoint *wireguard.Endpoint, key []byte, persistentKeepalive int) *wireguard.Endpoint {
- // Do not update non-nat peers
- if persistentKeepalive == 0 {
- return endpoint
- }
- e, ok := t.discoveredEndpoints[string(key)]
- if ok {
- return e
- }
- return endpoint
- }
- // Conf generates a WireGuard configuration file for a given Topology.
- func (t *Topology) Conf() *wireguard.Conf {
- c := &wireguard.Conf{
- Interface: &wireguard.Interface{
- PrivateKey: t.key,
- ListenPort: t.port,
- },
- }
- for _, s := range t.segments {
- if s.location == t.location {
- continue
- }
- peer := &wireguard.Peer{
- AllowedIPs: append(s.allowedIPs, s.allowedLocationIPs...),
- Endpoint: t.updateEndpoint(s.endpoint, s.key, s.persistentKeepalive),
- PersistentKeepalive: t.persistentKeepalive,
- PublicKey: s.key,
- }
- c.Peers = append(c.Peers, peer)
- }
- for _, p := range t.peers {
- peer := &wireguard.Peer{
- AllowedIPs: p.AllowedIPs,
- Endpoint: t.updateEndpoint(p.Endpoint, p.PublicKey, p.PersistentKeepalive),
- PersistentKeepalive: t.persistentKeepalive,
- PresharedKey: p.PresharedKey,
- PublicKey: p.PublicKey,
- }
- c.Peers = append(c.Peers, peer)
- }
- return c
- }
- // AsPeer generates the WireGuard peer configuration for the local location of the given Topology.
- // This configuration can be used to configure this location as a peer of another WireGuard interface.
- func (t *Topology) AsPeer() *wireguard.Peer {
- for _, s := range t.segments {
- if s.location != t.location {
- continue
- }
- return &wireguard.Peer{
- AllowedIPs: s.allowedIPs,
- Endpoint: s.endpoint,
- PublicKey: s.key,
- }
- }
- return nil
- }
- // PeerConf generates a WireGuard configuration file for a given peer in a Topology.
- func (t *Topology) PeerConf(name string) *wireguard.Conf {
- var pka int
- var psk []byte
- for i := range t.peers {
- if t.peers[i].Name == name {
- pka = t.peers[i].PersistentKeepalive
- psk = t.peers[i].PresharedKey
- break
- }
- }
- c := &wireguard.Conf{}
- for _, s := range t.segments {
- peer := &wireguard.Peer{
- AllowedIPs: s.allowedIPs,
- Endpoint: s.endpoint,
- PersistentKeepalive: pka,
- PresharedKey: psk,
- PublicKey: s.key,
- }
- c.Peers = append(c.Peers, peer)
- }
- for i := range t.peers {
- if t.peers[i].Name == name {
- continue
- }
- peer := &wireguard.Peer{
- AllowedIPs: t.peers[i].AllowedIPs,
- PersistentKeepalive: pka,
- PublicKey: t.peers[i].PublicKey,
- Endpoint: t.peers[i].Endpoint,
- }
- c.Peers = append(c.Peers, peer)
- }
- return c
- }
- // oneAddressCIDR takes an IP address and returns a CIDR
- // that contains only that address.
- func oneAddressCIDR(ip net.IP) *net.IPNet {
- return &net.IPNet{IP: ip, Mask: net.CIDRMask(len(ip)*8, len(ip)*8)}
- }
- // findLeader selects a leader for the nodes in a segment;
- // it will select the first node that says it should lead
- // or the first node in the segment if none have volunteered,
- // always preferring those with a public external IP address,
- func findLeader(nodes []*Node) int {
- var leaders, public []int
- for i := range nodes {
- if nodes[i].Leader {
- if isPublic(nodes[i].Endpoint.IP) {
- return i
- }
- leaders = append(leaders, i)
- }
- if isPublic(nodes[i].Endpoint.IP) {
- public = append(public, i)
- }
- }
- if len(leaders) != 0 {
- return leaders[0]
- }
- if len(public) != 0 {
- return public[0]
- }
- return 0
- }
- func deduplicatePeerIPs(peers []*Peer) []*Peer {
- ps := make([]*Peer, len(peers))
- ips := make(map[string]struct{})
- for i, peer := range peers {
- p := Peer{
- Name: peer.Name,
- Peer: wireguard.Peer{
- Endpoint: peer.Endpoint,
- PersistentKeepalive: peer.PersistentKeepalive,
- PresharedKey: peer.PresharedKey,
- PublicKey: peer.PublicKey,
- },
- }
- for _, ip := range peer.AllowedIPs {
- if _, ok := ips[ip.String()]; ok {
- continue
- }
- p.AllowedIPs = append(p.AllowedIPs, ip)
- ips[ip.String()] = struct{}{}
- }
- ps[i] = &p
- }
- return ps
- }
|