mirror of
https://github.com/yunionio/cloudpods.git
synced 2026-05-10 23:58:59 +08:00
151 lines
3.9 KiB
Go
151 lines
3.9 KiB
Go
// Copyright 2019 Yunion
|
|
//
|
|
// 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 dhcp
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"runtime/debug"
|
|
|
|
"golang.org/x/net/bpf"
|
|
|
|
"yunion.io/x/log"
|
|
)
|
|
|
|
type DHCPServer struct {
|
|
Address string
|
|
Port int
|
|
conn *Conn
|
|
}
|
|
|
|
// net.ListenPacket
|
|
func NewDHCPServer(address string, port int) (*DHCPServer, error) {
|
|
dhcpAddr := fmt.Sprintf("%s:%d", address, port)
|
|
dhcpConn, err := NewConn(dhcpAddr, false)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("New DHCP connection error: %v", err)
|
|
}
|
|
return &DHCPServer{
|
|
Address: address,
|
|
Port: port,
|
|
conn: dhcpConn,
|
|
}, nil
|
|
}
|
|
|
|
// udp socket
|
|
func NewDHCPServer3(address string, port int) (*DHCPServer, error) {
|
|
dhcpConn, err := NewSocketConn(address, port)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("New DHCP connection error: %v", err)
|
|
}
|
|
return &DHCPServer{
|
|
Address: address,
|
|
Port: port,
|
|
conn: dhcpConn,
|
|
}, nil
|
|
}
|
|
|
|
// raw socket
|
|
func NewDHCPServer2(iface string, dhcpServerPort, dhcpRelayPort uint16) (*DHCPServer, *Conn, error) {
|
|
// ip and udp and port 67 and port 68
|
|
bpf := []bpf.RawInstruction{
|
|
{0x28, 0, 0, 0x0000000c},
|
|
{0x15, 0, 13, 0x00000800},
|
|
{0x30, 0, 0, 0x00000017},
|
|
{0x15, 0, 11, 0x00000011},
|
|
{0x28, 0, 0, 0x00000014},
|
|
{0x45, 9, 0, 0x00001fff},
|
|
{0xb1, 0, 0, 0x0000000e},
|
|
{0x48, 0, 0, 0x0000000e},
|
|
{0x15, 0, 2, uint32(dhcpServerPort)},
|
|
{0x48, 0, 0, 0x00000010},
|
|
{0x15, 3, 4, uint32(dhcpRelayPort)},
|
|
{0x15, 0, 3, uint32(dhcpRelayPort)},
|
|
{0x48, 0, 0, 0x00000010},
|
|
{0x15, 0, 1, uint32(dhcpServerPort)},
|
|
{0x6, 0, 0, 0x00040000},
|
|
{0x6, 0, 0, 0x00000000},
|
|
}
|
|
conn, err := NewRawSocketConn(iface, bpf, dhcpServerPort)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
return &DHCPServer{
|
|
conn: conn,
|
|
}, conn, nil
|
|
}
|
|
|
|
type DHCPHandler interface {
|
|
ServeDHCP(pkt Packet, addr *net.UDPAddr, intf *net.Interface) (Packet, []string, error)
|
|
}
|
|
|
|
func (s *DHCPServer) ListenAndServe(handler DHCPHandler) error {
|
|
defer s.conn.Close()
|
|
return s.serveDHCP(handler)
|
|
}
|
|
|
|
func (s *DHCPServer) GetConn() *Conn {
|
|
return s.conn
|
|
}
|
|
|
|
func (s *DHCPServer) serveDHCP(handler DHCPHandler) error {
|
|
for {
|
|
pkt, addr, mac, intf, err := s.conn.RecvDHCP()
|
|
if err != nil {
|
|
log.Errorf("Receiving DHCP packet: %s", err)
|
|
continue
|
|
}
|
|
// if intf == nil {
|
|
// return fmt.Errorf("Received DHCP packet with no interface information (this is a violation of dhcp4.Conn's contract)")
|
|
// }
|
|
|
|
go func() {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
log.Errorf("Serve panic error: %v", r)
|
|
debug.PrintStack()
|
|
}
|
|
}()
|
|
|
|
resp, targets, err := handler.ServeDHCP(pkt, addr, intf)
|
|
if err != nil {
|
|
log.Warningf("[DHCP] handler serve error: %v", err)
|
|
return
|
|
}
|
|
if resp == nil {
|
|
// log.Warningf("[DHCP] hander response null packet")
|
|
return
|
|
}
|
|
if len(targets) > 0 {
|
|
targetUdpAddr := *addr
|
|
for _, target := range targets {
|
|
log.Debugf("[DHCP] Send packet back to %s", target)
|
|
targetUdpAddr.IP = net.ParseIP(target)
|
|
resp.SetGIAddr(targetUdpAddr.IP)
|
|
if err = s.conn.SendDHCP(resp, &targetUdpAddr, mac, intf); err != nil {
|
|
log.Errorf("[DHCP] failed to response packet for %s: %v", pkt.CHAddr(), err)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
//log.Debugf("[DHCP] send response packet: %s to interface: %#v", resp.DebugString(), intf)
|
|
if err = s.conn.SendDHCP(resp, addr, mac, intf); err != nil {
|
|
log.Errorf("[DHCP] failed to response packet for %s: %v", pkt.CHAddr(), err)
|
|
return
|
|
}
|
|
}()
|
|
}
|
|
}
|