Skip to content

Commit

Permalink
Adapt httpboot plugin to respond with client specific UKIs (#155)
Browse files Browse the repository at this point in the history
* Add httpboot plugin

* Adapt httpboot plugin to respond with client specific UKIs

* Add LinkLayerAddress in the XFF

* Support overwrite for IPv6 httpboot plugin

---------

Co-authored-by: Stefan Catargiu <[email protected]>
  • Loading branch information
hardikdr and 5kt authored Jun 17, 2024
1 parent 74a3083 commit e611122
Showing 1 changed file with 112 additions and 12 deletions.
124 changes: 112 additions & 12 deletions plugins/httpboot/plugin.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,18 +4,23 @@
package httpboot

import (
"encoding/json"
"fmt"
"io"
"net/http"
"net/url"
"strings"

"github.com/coredhcp/coredhcp/handler"
"github.com/coredhcp/coredhcp/logger"
"github.com/coredhcp/coredhcp/plugins"
"github.com/insomniacslk/dhcp/dhcpv4"
"github.com/insomniacslk/dhcp/dhcpv6"
"net/url"
"strings"
)

var bootFile4 string
var bootFile6 string
var useBootService bool

var log = logger.GetLogger("plugins/httpboot")

Expand All @@ -25,25 +30,35 @@ var Plugin = plugins.Plugin{
Setup4: setup4,
}

func parseArgs(args ...string) (*url.URL, error) {
func parseArgs(args ...string) (*url.URL, bool, error) {
if len(args) != 1 {
return nil, fmt.Errorf("Exactly one argument must be passed to the httpboot plugin, got %d", len(args))
return nil, false, fmt.Errorf("exactly one argument must be passed to the httpboot plugin, got %d", len(args))
}
arg := args[0]
useBootService := strings.HasPrefix(arg, "bootservice:")
if useBootService {
arg = strings.TrimPrefix(arg, "bootservice:")
}
return url.Parse(args[0])
parsedURL, err := url.Parse(arg)
if err != nil {
return nil, false, fmt.Errorf("invalid URL: %v", err)
}
return parsedURL, useBootService, nil
}

func setup6(args ...string) (handler.Handler6, error) {
u, err := parseArgs(args...)
u, usebootservice, err := parseArgs(args...)
if err != nil {
return nil, err
return nil, fmt.Errorf("invalid configuration: %v", err)
}
bootFile6 = u.String()
log.Printf("loaded httpboot plugin for DHCPv6.")
useBootService = usebootservice
log.Printf("Configured httpboot plugin with URL: %s, useBootService: %t", bootFile6, useBootService)
return Handler6, nil
}

func setup4(args ...string) (handler.Handler4, error) {
u, err := parseArgs(args...)
u, _, err := parseArgs(args...)
if err != nil {
return nil, err
}
Expand All @@ -54,9 +69,26 @@ func setup4(args ...string) (handler.Handler4, error) {

func Handler6(req, resp dhcpv6.DHCPv6) (dhcpv6.DHCPv6, bool) {
log.Debugf("Received DHCPv6 request: %s", req.Summary())

var ukiURL string
if !useBootService {
ukiURL = bootFile6
} else {
clientIPs, err := extractClientIP6(req)
if err != nil {
log.Errorf("failed to extract ClientIP, Error: %v Request: %v ", err, req)
return resp, false
}
ukiURL, err = fetchUKIURL(bootFile6, clientIPs)
if err != nil {
log.Errorf("failed to fetch UKI URL: %v", err)
return resp, false
}
}

decap, err := req.GetInnerMessage()
if err != nil {
log.Errorf("Could not decapsulate request: %v", err)
log.Errorf("could not decapsulate request: %v", err)
return nil, true
}

Expand All @@ -65,7 +97,7 @@ func Handler6(req, resp dhcpv6.DHCPv6) (dhcpv6.DHCPv6, bool) {
if strings.Contains(vc, "HTTPClient") {
bf := &dhcpv6.OptionGeneric{
OptionCode: dhcpv6.OptionBootfileURL,
OptionData: []byte(bootFile6),
OptionData: []byte(ukiURL),
}
resp.AddOption(bf)
vid := &dhcpv6.OptionGeneric{
Expand All @@ -84,12 +116,19 @@ func Handler6(req, resp dhcpv6.DHCPv6) (dhcpv6.DHCPv6, bool) {

func Handler4(req, resp *dhcpv4.DHCPv4) (*dhcpv4.DHCPv4, bool) {
log.Debugf("Received DHCPv4 request: %s", req.Summary())

ukiURL, err := fetchUKIURL(bootFile4, []string{req.ClientIPAddr.String()})
if err != nil {
log.Errorf("failed to fetch UKI URL: %v", err)
return resp, false
}

if req.GetOneOption(dhcpv4.OptionClassIdentifier) != nil {
vc := req.GetOneOption(dhcpv4.OptionClassIdentifier)
if strings.Contains(string(vc), "HTTPClient") {
bf := &dhcpv4.Option{
Code: dhcpv4.OptionBootfileName,
Value: dhcpv4.String(bootFile4),
Value: dhcpv4.String(ukiURL),
}
resp.Options.Update(*bf)
vid := &dhcpv4.Option{
Expand All @@ -101,3 +140,64 @@ func Handler4(req, resp *dhcpv4.DHCPv4) (*dhcpv4.DHCPv4, bool) {
}
return resp, false
}

func extractClientIP6(req dhcpv6.DHCPv6) ([]string, error) {
if req.IsRelay() {
relayMsg, ok := req.(*dhcpv6.RelayMessage)
if !ok {
return nil, fmt.Errorf("failed to cast the DHCPv6 request to a RelayMessage")
}

var addresses []string
if relayMsg.LinkAddr != nil {
addresses = append(addresses, relayMsg.LinkAddr.String())
}

if _, linkLayerAddress := relayMsg.Options.ClientLinkLayerAddress(); linkLayerAddress != nil {
addresses = append(addresses, linkLayerAddress.String())
}

if len(addresses) == 0 {
return nil, fmt.Errorf("no client IP or link-layer address found in the relay message")
}

return addresses, nil
}
return nil, fmt.Errorf("received non-relay DHCPv6 request, client IP cannot be extracted from non-relayed messages")
}

func fetchUKIURL(url string, clientIPs []string) (string, error) {
client := &http.Client{}
req, err := http.NewRequest("GET", url, nil)
if err != nil {
return "", err
}

xForwardedFor := strings.Join(clientIPs, ", ")
req.Header.Set("X-Forwarded-For", xForwardedFor)

resp, err := client.Do(req)
if err != nil {
log.Errorf("HTTP request failed: %v", err)
return "", err
}
defer resp.Body.Close()

body, err := io.ReadAll(resp.Body)
if err != nil {
return "", err
}

var data struct {
UKIURL string `json:"UKIURL"`
}
if err := json.Unmarshal(body, &data); err != nil {
return "", err
}

if data.UKIURL == "" {
return "", fmt.Errorf("received empty UKI URL")
}

return data.UKIURL, nil
}

0 comments on commit e611122

Please sign in to comment.