// 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" "sync" ) // --------------------------------------------------------------------------- // 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 // Persistenter Teil 👇 DuidHostnameList []payload duidFile string duidMu sync.RWMutex ) // --------------------------------------------------------------------------- // 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 } 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 loadDuidList() { duidMu.Lock() defer duidMu.Unlock() f, err := os.Open(duidFile) if err != nil { if os.IsNotExist(err) { DuidHostnameList = nil return } log.Printf("duid-load: %v", err) return } defer f.Close() var tmp []payload if err := json.NewDecoder(f).Decode(&tmp); err != nil { log.Printf("duid-load: decode: %v", err) return } DuidHostnameList = tmp log.Printf("duid-load: %d Einträge geladen", len(DuidHostnameList)) } func saveDuidList() { duidMu.RLock() data, err := json.MarshalIndent(DuidHostnameList, "", " ") duidMu.RUnlock() if err != nil { log.Printf("duid-save: marshal: %v", err) return } tmpFile := duidFile + ".tmp" if err := os.WriteFile(tmpFile, data, 0o644); err != nil { log.Printf("duid-save: write tmp: %v", err) return } if err := os.Rename(tmpFile, duidFile); err != nil { log.Printf("duid-save: rename: %v", err) return } } func uniqueStrings(in []string) []string { m := make(map[string]struct{}, len(in)) var out []string for _, s := range in { if s == "" { continue } if _, ok := m[s]; ok { continue } m[s] = struct{}{} out = append(out, s) } return out } func uniqueU32(in []uint32) []uint32 { m := make(map[uint32]struct{}, len(in)) var out []uint32 for _, v := range in { if _, ok := m[v]; ok { continue } m[v] = struct{}{} out = append(out, v) } return out } // fügt neu ein ODER aktualisiert bestehenden Host func upsertPayload(p payload) { duidMu.Lock() defer duidMu.Unlock() for i := range DuidHostnameList { if strings.EqualFold(DuidHostnameList[i].Hostname, p.Hostname) { // merge DuidHostnameList[i].DUIDs = uniqueStrings(append(DuidHostnameList[i].DUIDs, p.DUIDs...)) DuidHostnameList[i].IAIDs = uniqueU32(append(DuidHostnameList[i].IAIDs, p.IAIDs...)) return } } // neu p.DUIDs = uniqueStrings(p.DUIDs) p.IAIDs = uniqueU32(p.IAIDs) DuidHostnameList = append(DuidHostnameList, p) } 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 { Ipv4Octets, _ := octetsRaw(defaultIP) // evtl. pageIP nehmen? rHostname := []rune(xHostname) if len(rHostname) < 6 { // 2+4 // fallback: nimm letzte zwei Oktette 0.0 qCalculatedIPv4 := Ipv4Octets[0] + "." + Ipv4Octets[1] + ".0.0" qCalculatedIPv6, _ := embedIPv4(qCalculatedIPv4) return []payloadHelper{{ Hostname: xHostname, DUID: firstOrEmpty(xDUIDs), CalculatedIPv4: qCalculatedIPv4, CalculatedIPv6: qCalculatedIPv6, Dhcp4Scope: dhcpScope, Dhcp6Scope: ulaPrefix, DomainName: dhcpDomain, DhcpServer: dhcpServer, IAID: "0", }} } qDUID := firstOrEmpty(xDUIDs) 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 { res = append(res, payloadHelper{ Hostname: xHostname, DUID: qDUID, CalculatedIPv4: qCalculatedIPv4, CalculatedIPv6: qCalculatedIPv6, Dhcp4Scope: dhcpScope, Dhcp6Scope: ulaPrefix, DomainName: dhcpDomain, DhcpServer: dhcpServer, IAID: fmt.Sprintf("%d", t), }) } return res } func firstOrEmpty(xs []string) string { if len(xs) == 0 { return "" } return xs[0] } func getDhcp() []payloadHelper { duidMu.RLock() defer duidMu.RUnlock() cp := make([]payload, len(DuidHostnameList)) copy(cp, DuidHostnameList) sortByHostname(cp) var result []payloadHelper for _, b := range cp { 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 } if p.Hostname == "" { http.Error(w, "hostname fehlt", http.StatusBadRequest) return } if len(p.DUIDs) == 0 { // optional: nicht hart abbrechen – aber wenigstens loggen log.Printf("register: Host %s ohne DUIDs", p.Hostname) } upsertPayload(p) saveDuidList() // 👈 direkt nach Update persistieren log.Printf("client registriert/aktualisiert: %s → DUIDs=%v IAIDs=%v", p.Hostname, p.DUIDs, p.IAIDs) log.Println(p) w.WriteHeader(http.StatusNoContent) } 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() // Persistenz initialisieren duidFile = getenv("DUID_DB_FILE", "duids.json") loadDuidList() 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-Eingaben

Verwenden Sie ausschließlich den 'alten' Dialog über die klassische Systemsteuerung! Nicht das 'moderne' UI über Einstellungen, da sonst zusätzliche IP-Adressen gelöscht werden!

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

Fehler: {{.Error}}

{{end}}

Aktives Präfix: %s (/64)

` var rangePageHTML = ` IPv4-Range → IPv6

IPv4-Bereich → IPv6-Tabelle

« Einzelkonverter
{{if .HaveResult}} {{range .Rows}}{{end}} //Add-DhcpServerv4Reservation -ComputerName $Srv -ScopeId $Prefix -IPAddress IPv6Address -ClientId $l.ClientDuid -Name $l.HostName -Description "Auto-reserved after rollout" -Type Dhcp
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-DhcpServerv4Reservation -ComputerName $Srv -ScopeId $Prefix -IPAddress IPv6Address -ClientId $l.ClientDuid -Name $l.HostName -Description "Auto-reserved after rollout" -Type Dhcp //Add-DhcpServerv6Reservation -ComputerName $Srv -Prefix $Prefix -IPAddress IPv6Address -ClientDuid $l.ClientDuid -Iaid $l.Iaid -Name $l.HostName -Description "Auto-reserved after rollout"