// A Go web‑server that converts IPv4 addresses into IPv6 addresses // inside a user‑defined /96 ULA prefix and outputs Windows 11‑ready fields. // // Features // -------- // 1. Single conversion page (http://localhost:8080/) – converts one IPv4. // 2. Range conversion page (http://localhost:8080/range) – converts a span of IPv4s into a table. // // Environment variables (all optional): // // ULA_PREFIX – /96 prefix to embed into, default "fd09:cafe:affe:4010::" // DNS1 – preferred DNS IPv6 address (default: ::53) // DNS2 – alternate DNS IPv6 address (default: ::54) // FORM_DEFAULT_IP – placeholder for the single‑convert page, default 172.16.0.0 // RANGE_LIMIT – max number of addresses allowed in /range (default 1024) // // Build & run: // // go run main.go package main import ( "encoding/json" "fmt" "html/template" "log" "net" "net/http" "os" "sort" "strconv" "strings" ) // --------------------------------------------------------------------------- // Configuration constants & globals // --------------------------------------------------------------------------- const ( listenAddr = ":8080" defaultPrefix = "fd09:cafe:affe:4010::" // fallback /96 defaultIP = "172.16.0.0" defaultLimit = 1024 // max. rows for range conversion ) var ( ulaPrefix string dns1, dns2 string pageIP string rangeLimit int singleTemplate *template.Template rangeTemplate *template.Template dhcpTemplate *template.Template dhcpServer string dhcpScope string dhcpNamePrefix string dhcpDomain string ) // --------------------------------------------------------------------------- // Template data structures // --------------------------------------------------------------------------- type singleData struct { IPv4, IPv6, Gateway, DNS1, DNS2, Error string HaveResult bool } type addrPair struct{ Name, IPv4, IPv6 string } type rangeData struct { Start, End string DhcpServer, DhcpScope, DhcpDomain string Rows []addrPair Error string HaveResult bool } var DuidHostnameList []payload type payload struct { Hostname string `json:"hostname"` DUIDs []string `json:"duids"` IAIDs []uint32 `json:"iaids"` } type payloadHelper struct { Hostname string DomainName string DhcpServer string DUID string IAID string CalculatedIPv4 string Dhcp4Scope string CalculatedIPv6 string Dhcp6Scope string } func octetsRaw(ip string) ([]string, error) { parts := strings.Split(ip, ".") if len(parts) != 4 { return nil, fmt.Errorf("ungültige IPv4-Adresse: %q", ip) } return parts, nil } func getenv(k, d string) string { if v := os.Getenv(k); v != "" { return v } return d } func enabled(k string, def bool) bool { b, err := strconv.ParseBool(strings.ToLower(os.Getenv(k))) if err != nil { return def } return b } func DhcpHelperFunc(xHostname string, xDUIDs []string, xIAIDs []uint32) []payloadHelper { /*IPv4*/ Ipv4Octets, _ := octetsRaw(defaultIP) rHostname := []rune(xHostname) qDUID := xDUIDs[0] qSegment1 := string(rHostname[2:4]) qSegment2 := string(rHostname[4:]) qCalculatedIPv4 := Ipv4Octets[0] + "." + Ipv4Octets[1] + "." + qSegment1 + "." + qSegment2 qCalculatedIPv6, _ := embedIPv4(qCalculatedIPv4) var res []payloadHelper for _, t := range xIAIDs { r := payloadHelper{ Hostname: xHostname, DUID: qDUID, CalculatedIPv4: qCalculatedIPv4, CalculatedIPv6: qCalculatedIPv6, Dhcp4Scope: dhcpScope, Dhcp6Scope: ulaPrefix, DomainName: dhcpDomain, DhcpServer: dhcpServer, IAID: fmt.Sprintf("%d", t), } res = append(res, r) } return res } func getDhcp() []payloadHelper { var result []payloadHelper sortByHostname(DuidHostnameList) for _, b := range DuidHostnameList { result = append(result, DhcpHelperFunc(b.Hostname, b.DUIDs, b.IAIDs)...) } return result } func register(w http.ResponseWriter, r *http.Request) { if r.Method != http.MethodPost { http.Error(w, "nur POST erlaubt", http.StatusMethodNotAllowed) return } var p payload if err := json.NewDecoder(r.Body).Decode(&p); err != nil { http.Error(w, "ungültiges JSON", http.StatusBadRequest) return } // --- hier kannst du speichern, weiterverarbeiten, loggen … --- log.Printf("neuer Client: %s → DUIDs=%v", p.Hostname, p.DUIDs) DuidHostnameList = append(DuidHostnameList, p) w.WriteHeader(http.StatusNoContent) // 204 } func getdhcp6(w http.ResponseWriter, r *http.Request) { temp := getDhcp() w.Header().Set("Content-Type", "text/html; charset=utf-8") _ = dhcpTemplate.Execute(w, temp) } func sortByHostname(p []payload) { sort.Slice(p, func(i, j int) bool { // Optional: case-insensitiv vergleichen return strings.ToLower(p[i].Hostname) < strings.ToLower(p[j].Hostname) }) } // --------------------------------------------------------------------------- // main – HTTP routing // --------------------------------------------------------------------------- func main() { initConfigAndTemplates() http.HandleFunc("/", handleSingle) http.HandleFunc("/convert", handleSingleConvert) http.HandleFunc("/range", handleRange) http.HandleFunc("/register", register) http.HandleFunc("/dhcp6", getdhcp6) log.Printf("Server läuft auf http://localhost%s (Präfix %s, DNS1 %s, DNS2 %s, Limit %d)", listenAddr, ulaPrefix, dns1, dns2, rangeLimit) log.Fatal(http.ListenAndServe(listenAddr, nil)) } // --------------------------------------------------------------------------- // Handlers – single conversion // --------------------------------------------------------------------------- func handleSingle(w http.ResponseWriter, r *http.Request) { renderSingle(w, singleData{}) } func handleSingleConvert(w http.ResponseWriter, r *http.Request) { if err := r.ParseForm(); err != nil { renderSingle(w, singleData{Error: "Ungültiges Formular"}) return } ipv4Str := r.FormValue("ipv4") ipv6, err := embedIPv4(ipv4Str) d := singleData{IPv4: ipv4Str} if err != nil { d.Error = err.Error() } else { d.HaveResult = true d.IPv6 = ipv6 d.Gateway = ulaPrefix + "1" d.DNS1, d.DNS2 = dns1, dns2 } renderSingle(w, d) } // --------------------------------------------------------------------------- // Handlers – range conversion // --------------------------------------------------------------------------- func handleRange(w http.ResponseWriter, r *http.Request) { switch r.Method { case http.MethodGet: renderRange(w, rangeData{}) return case http.MethodPost: if err := r.ParseForm(); err != nil { renderRange(w, rangeData{Error: "Ungültiges Formular"}) return } startStr := r.FormValue("start") endStr := r.FormValue("end") rows, err := convertRange(startStr, endStr) d := rangeData{Start: startStr, End: endStr, DhcpServer: dhcpServer, DhcpScope: dhcpScope, DhcpDomain: dhcpDomain} if err != nil { d.Error = err.Error() } else { d.HaveResult = true for _, b := range rows { //fmt.Println(a, b) octets := strings.Split(b.IPv4, ".") if len(octets) != 4 { fmt.Println("Ungültige IP-Adresse!") return } O3 := "" O4 := "" if len(octets[2]) < 2 { O3 = "0" + octets[2] } else { O3 = octets[2] } if len(octets[3]) < 2 { O4 = "0" + octets[3] } else { O4 = octets[3] } N := dhcpNamePrefix + O3 + O4 d.Rows = append(d.Rows, addrPair{IPv4: b.IPv4, IPv6: b.IPv6, Name: N}) } } renderRange(w, d) default: http.Error(w, "Method not allowed", http.StatusMethodNotAllowed) } } // convertRange returns slice of addrPairs for every IPv4 in [start,end]. func convertRange(start, end string) ([]addrPair, error) { sip := net.ParseIP(start).To4() eip := net.ParseIP(end).To4() if sip == nil || eip == nil { return nil, fmt.Errorf("start oder ende ist keine gültige ipv4-adresse") } su := ipToUint32(sip) eu := ipToUint32(eip) if su > eu { return nil, fmt.Errorf("start darf nicht größer als ende sein") } count := int(eu - su + 1) if count > rangeLimit { return nil, fmt.Errorf("bereich zu groß (%d > %d)", count, rangeLimit) } rows := make([]addrPair, 0, count) for cur := su; cur <= eu; cur++ { ipv4 := uint32ToIP(cur).String() ipv6, _ := embedIPv4(ipv4) // embedIPv4 kann hier nicht mehr fehlschlagen rows = append(rows, addrPair{IPv4: ipv4, IPv6: ipv6}) } return rows, nil } // --------------------------------------------------------------------------- // Helper functions // --------------------------------------------------------------------------- func embedIPv4(ipv4 string) (string, error) { ip := net.ParseIP(ipv4).To4() if ip == nil { return "", fmt.Errorf("%s ist keine gültige IPv4-Adresse", ipv4) } hi := uint16(ip[0])<<8 | uint16(ip[1]) lo := uint16(ip[2])<<8 | uint16(ip[3]) return fmt.Sprintf("%s%x:%x", ulaPrefix, hi, lo), nil } func ipToUint32(ip net.IP) uint32 { return uint32(ip[0])<<24 | uint32(ip[1])<<16 | uint32(ip[2])<<8 | uint32(ip[3]) } func uint32ToIP(u uint32) net.IP { return net.IPv4(byte(u>>24), byte(u>>16), byte(u>>8), byte(u)) } // --------------------------------------------------------------------------- // Config & templates // --------------------------------------------------------------------------- func initConfigAndTemplates() { // ULA prefix --------------------------------- ulaPrefix = getenv("ULA_PREFIX", "fd12:3456:789A:BCDE::") if ulaPrefix == "" { ulaPrefix = defaultPrefix } if !strings.HasSuffix(ulaPrefix, "::") { if strings.HasSuffix(ulaPrefix, ":") { ulaPrefix += ":" } else { ulaPrefix += "::" } } // DNS defaults -------------------------------- dns1 = getenv("DNS1", "fd12:3456:789A:BCDE::ac10:a") dns2 = getenv("DNS2", "fd12:3456:789A:BCDE::ac10:b") if dns1 == "" { dns1 = strings.Replace(ulaPrefix, "::", "::53", 1) } if dns2 == "" { dns2 = strings.Replace(ulaPrefix, "::", "::54", 1) } // Placeholder IP ------------------------------ pageIP = getenv("FORM_DEFAULT_IP", "172.16.0.0") if pageIP == "" { pageIP = defaultIP } dhcpScope = os.Getenv("DHCP_SCOPE") dhcpServer = os.Getenv("DHCP_SERVER") dhcpNamePrefix = os.Getenv("DHCP_NAME_PREFIX") dhcpDomain = os.Getenv("DHCP_DOMAIN") // Range limit --------------------------------- if limStr := os.Getenv("RANGE_LIMIT"); limStr != "" { if v, err := strconv.Atoi(limStr); err == nil && v > 0 { rangeLimit = v } else { rangeLimit = defaultLimit } } else { rangeLimit = defaultLimit } // Templates ----------------------------------- singleHTML := fmt.Sprintf(singlePageHTML, pageIP, ulaPrefix) rangeHTML := rangePageHTML singleTemplate = template.Must(template.New("single").Parse(singleHTML)) rangeTemplate = template.Must(template.New("range").Parse(rangeHTML)) dhcpTemplate = template.Must(template.New("range").Parse(rangeDHCP6HTML)) fmt.Println(ulaPrefix, dns1, dns2, pageIP, dhcpServer, dhcpScope, rangeLimit) //fmt.Println(singleHTML) //fmt.Println(rangeHTML) } // --------------------------------------------------------------------------- // Template rendering helpers // --------------------------------------------------------------------------- func renderSingle(w http.ResponseWriter, d singleData) { w.Header().Set("Content-Type", "text/html; charset=utf-8") _ = singleTemplate.Execute(w, d) } func renderRange(w http.ResponseWriter, d rangeData) { w.Header().Set("Content-Type", "text/html; charset=utf-8") _ = rangeTemplate.Execute(w, d) } // --------------------------------------------------------------------------- // HTML templates as raw strings with %s placeholders (prefix, etc.) // --------------------------------------------------------------------------- var singlePageHTML = ` IPv4 → IPv6-Mapper

IPv4 → IPv6 (einzeln)

» Bereich konvertierenDHCPv6
{{if .HaveResult}}

Windows 11-Eingaben

{{end}} {{if .Error}}

Fehler: {{.Error}}

{{end}}

Aktives Präfix: %s (/96)

` var rangePageHTML = ` IPv4-Range → IPv6

IPv4-Bereich → IPv6-Tabelle

« Einzelkonverter
{{if .HaveResult}} {{range .Rows}}{{end}}
IPv4IPv6DHCP-IPv4DHCP-IPv6
{{.IPv4}}{{.IPv6}}netsh DHCP Server {{$.DhcpServer}} Scope {{$.DhcpScope}} Add reservedip {{.IPv4}} "{{.Name}}.{{$.DhcpDomain}}" "" "DHCP"---
{{end}} {{if .Error}}

Fehler: {{.Error}}

{{end}} ` var rangeDHCP6HTML = ` DHCP-IPv6

DHCPv6 - DUID-Registrierung

{{range .}}{{end}}
HostnameDuidIPv4IPv6DHCP6
{{.Hostname}}{{.DUID}}{{.CalculatedIPv4}}{{.CalculatedIPv6}}Add-DhcpServerv6Reservation -ComputerName {{.DhcpServer}} -Prefix {{.Dhcp6Scope}} -IPAddress {{.CalculatedIPv6}} -ClientDuid {{.DUID}} -Iaid {{.IAID}} -Name {{.Hostname}}.{{.DomainName}} -Description "Auto-reserved after rollout"
` //Add-DhcpServerv6Reservation -ComputerName $Srv -Prefix $Prefix -IPAddress IPv6Address -ClientDuid $l.ClientDuid -Iaid $l.Iaid -Name $l.HostName -Description "Auto-reserved after rollout"