2024-12-06 22:41:03 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"encoding/json"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"net"
|
2025-01-05 08:51:45 +00:00
|
|
|
"net/http"
|
2024-12-06 22:41:03 +00:00
|
|
|
"os"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/acidvegas/golcg"
|
2025-01-05 08:59:48 +00:00
|
|
|
"github.com/miekg/dns"
|
2024-12-06 22:41:03 +00:00
|
|
|
"github.com/rivo/tview"
|
|
|
|
)
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
const defaultResolversURL = "https://raw.githubusercontent.com/trickest/resolvers/refs/heads/main/resolvers.txt"
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
type Config struct {
|
2025-01-05 08:51:45 +00:00
|
|
|
concurrency int
|
|
|
|
timeout time.Duration
|
|
|
|
retries int
|
|
|
|
dnsServers []string
|
|
|
|
serverIndex int
|
|
|
|
debug bool
|
|
|
|
outputFile *os.File
|
|
|
|
mu sync.Mutex
|
|
|
|
lastDNSUpdate time.Time
|
|
|
|
updateMu sync.Mutex
|
2025-01-05 09:31:09 +00:00
|
|
|
loop bool
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type Stats struct {
|
|
|
|
processed uint64
|
|
|
|
total uint64
|
|
|
|
lastProcessed uint64
|
|
|
|
lastCheckTime time.Time
|
2025-01-05 09:31:09 +00:00
|
|
|
startTime time.Time
|
2024-12-06 22:41:03 +00:00
|
|
|
success uint64
|
|
|
|
failed uint64
|
2025-01-05 08:51:45 +00:00
|
|
|
cnames uint64
|
2024-12-06 22:41:03 +00:00
|
|
|
speedHistory []float64
|
|
|
|
mu sync.Mutex
|
2025-01-20 18:47:16 +00:00
|
|
|
round uint64
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stats) increment() {
|
|
|
|
atomic.AddUint64(&s.processed, 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stats) incrementSuccess() {
|
|
|
|
atomic.AddUint64(&s.success, 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Stats) incrementFailed() {
|
|
|
|
atomic.AddUint64(&s.failed, 1)
|
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
func (s *Stats) incrementCNAME() {
|
|
|
|
atomic.AddUint64(&s.cnames, 1)
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
func (c *Config) getNextServer() string {
|
2025-01-05 08:51:45 +00:00
|
|
|
if err := c.updateDNSServers(); err != nil {
|
|
|
|
fmt.Printf("Failed to update DNS servers: %v\n", err)
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
c.mu.Lock()
|
|
|
|
defer c.mu.Unlock()
|
|
|
|
|
|
|
|
if len(c.dnsServers) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
server := c.dnsServers[c.serverIndex]
|
|
|
|
c.serverIndex = (c.serverIndex + 1) % len(c.dnsServers)
|
|
|
|
return server
|
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
func fetchDefaultResolvers() ([]string, error) {
|
|
|
|
resp, err := http.Get(defaultResolversURL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to fetch default resolvers: %v", err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var resolvers []string
|
|
|
|
scanner := bufio.NewScanner(resp.Body)
|
|
|
|
for scanner.Scan() {
|
|
|
|
resolver := strings.TrimSpace(scanner.Text())
|
|
|
|
if resolver != "" {
|
|
|
|
resolvers = append(resolvers, resolver)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
|
|
return nil, fmt.Errorf("error reading default resolvers: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return resolvers, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func loadDNSServers(dnsFile string) ([]string, error) {
|
|
|
|
if dnsFile == "" {
|
|
|
|
resolvers, err := fetchDefaultResolvers()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if len(resolvers) == 0 {
|
|
|
|
return nil, fmt.Errorf("no default resolvers found")
|
|
|
|
}
|
|
|
|
return resolvers, nil
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
file, err := os.Open(dnsFile)
|
2024-12-06 22:41:03 +00:00
|
|
|
if err != nil {
|
2025-01-05 08:51:45 +00:00
|
|
|
return nil, fmt.Errorf("failed to open DNS servers file: %v", err)
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
var servers []string
|
|
|
|
scanner := bufio.NewScanner(file)
|
|
|
|
for scanner.Scan() {
|
|
|
|
server := strings.TrimSpace(scanner.Text())
|
2025-01-05 08:51:45 +00:00
|
|
|
if server != "" {
|
2024-12-06 22:41:03 +00:00
|
|
|
servers = append(servers, server)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
if err := scanner.Err(); err != nil {
|
|
|
|
return nil, fmt.Errorf("error reading DNS servers file: %v", err)
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
if len(servers) == 0 {
|
2025-01-05 08:51:45 +00:00
|
|
|
return nil, fmt.Errorf("no DNS servers found in file")
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
return servers, nil
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
type DNSResponse struct {
|
|
|
|
Names []string
|
|
|
|
Server string
|
|
|
|
RecordType string // "PTR" or "CNAME"
|
|
|
|
Target string // For CNAME records, stores the target
|
2025-01-05 08:59:48 +00:00
|
|
|
TTL uint32 // Add TTL field
|
2025-01-05 08:51:45 +00:00
|
|
|
}
|
2024-12-06 22:41:03 +00:00
|
|
|
|
2025-01-05 09:48:27 +00:00
|
|
|
func translateRcode(rcode int) string {
|
|
|
|
switch rcode {
|
|
|
|
case dns.RcodeSuccess:
|
|
|
|
return "Success"
|
|
|
|
case dns.RcodeFormatError:
|
|
|
|
return "Format Error"
|
|
|
|
case dns.RcodeServerFailure:
|
|
|
|
return "Server Failure"
|
|
|
|
case dns.RcodeNameError: // NXDOMAIN
|
|
|
|
return "No Such Domain"
|
|
|
|
case dns.RcodeNotImplemented:
|
|
|
|
return "Not Implemented"
|
|
|
|
case dns.RcodeRefused:
|
|
|
|
return "Query Refused"
|
|
|
|
default:
|
|
|
|
return fmt.Sprintf("DNS Error %d", rcode)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func lookupWithRetry(ip string, cfg *Config) (DNSResponse, string, error) {
|
2025-01-05 08:51:45 +00:00
|
|
|
var lastErr error
|
2025-01-05 09:48:27 +00:00
|
|
|
var lastServer string
|
2024-12-06 22:41:03 +00:00
|
|
|
|
|
|
|
for i := 0; i < cfg.retries; i++ {
|
2025-01-05 08:51:45 +00:00
|
|
|
server := cfg.getNextServer()
|
|
|
|
if server == "" {
|
2025-01-05 09:48:27 +00:00
|
|
|
return DNSResponse{}, "", fmt.Errorf("no DNS servers available")
|
2025-01-05 08:51:45 +00:00
|
|
|
}
|
2025-01-05 09:48:27 +00:00
|
|
|
lastServer = server
|
2025-01-05 08:51:45 +00:00
|
|
|
|
2025-01-05 08:59:48 +00:00
|
|
|
// Create DNS message
|
|
|
|
m := new(dns.Msg)
|
|
|
|
arpa, err := dns.ReverseAddr(ip)
|
|
|
|
if err != nil {
|
2025-01-05 09:48:27 +00:00
|
|
|
return DNSResponse{}, "", err
|
2025-01-05 08:51:45 +00:00
|
|
|
}
|
2025-01-05 08:59:48 +00:00
|
|
|
m.SetQuestion(arpa, dns.TypePTR)
|
|
|
|
m.RecursionDesired = true
|
2025-01-05 08:51:45 +00:00
|
|
|
|
2025-01-05 08:59:48 +00:00
|
|
|
// Create DNS client
|
|
|
|
c := new(dns.Client)
|
|
|
|
c.Timeout = cfg.timeout
|
2024-12-06 22:41:03 +00:00
|
|
|
|
2025-01-05 08:59:48 +00:00
|
|
|
// Make the query
|
|
|
|
r, _, err := c.Exchange(m, server)
|
|
|
|
if err != nil {
|
|
|
|
lastErr = err
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if r.Rcode != dns.RcodeSuccess {
|
2025-01-05 09:48:27 +00:00
|
|
|
lastErr = fmt.Errorf("%s", translateRcode(r.Rcode))
|
2025-01-05 08:59:48 +00:00
|
|
|
continue
|
|
|
|
}
|
2025-01-05 08:51:45 +00:00
|
|
|
|
2025-01-05 08:59:48 +00:00
|
|
|
// Process the response
|
|
|
|
if len(r.Answer) > 0 {
|
|
|
|
var names []string
|
|
|
|
var ttl uint32
|
|
|
|
var isCNAME bool
|
|
|
|
var target string
|
|
|
|
|
|
|
|
for _, ans := range r.Answer {
|
|
|
|
switch rr := ans.(type) {
|
|
|
|
case *dns.PTR:
|
|
|
|
names = append(names, rr.Ptr)
|
|
|
|
ttl = rr.Hdr.Ttl
|
|
|
|
case *dns.CNAME:
|
|
|
|
isCNAME = true
|
|
|
|
names = append(names, rr.Hdr.Name)
|
|
|
|
target = rr.Target
|
|
|
|
ttl = rr.Hdr.Ttl
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(names) > 0 {
|
|
|
|
if isCNAME {
|
2025-01-05 08:51:45 +00:00
|
|
|
return DNSResponse{
|
|
|
|
Names: names,
|
2025-01-05 09:48:27 +00:00
|
|
|
Server: server,
|
2025-01-05 08:51:45 +00:00
|
|
|
RecordType: "CNAME",
|
2025-01-05 08:59:48 +00:00
|
|
|
Target: strings.TrimSuffix(target, "."),
|
|
|
|
TTL: ttl,
|
2025-01-05 09:48:27 +00:00
|
|
|
}, server, nil
|
2025-01-05 08:51:45 +00:00
|
|
|
}
|
2025-01-05 08:59:48 +00:00
|
|
|
return DNSResponse{
|
|
|
|
Names: names,
|
2025-01-05 09:48:27 +00:00
|
|
|
Server: server,
|
2025-01-05 08:59:48 +00:00
|
|
|
RecordType: "PTR",
|
|
|
|
TTL: ttl,
|
2025-01-05 09:48:27 +00:00
|
|
|
}, server, nil
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
}
|
2025-01-05 08:51:45 +00:00
|
|
|
|
2025-01-05 08:59:48 +00:00
|
|
|
lastErr = fmt.Errorf("no PTR records found")
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
2025-01-05 08:51:45 +00:00
|
|
|
|
2025-01-05 09:48:27 +00:00
|
|
|
return DNSResponse{}, lastServer, lastErr
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func reverse(ss []string) []string {
|
|
|
|
reversed := make([]string, len(ss))
|
|
|
|
for i, s := range ss {
|
|
|
|
reversed[len(ss)-1-i] = s
|
|
|
|
}
|
|
|
|
return reversed
|
|
|
|
}
|
|
|
|
|
|
|
|
func colorizeIPInPtr(ptr, ip string) string {
|
|
|
|
specialHosts := []string{"localhost", "undefined.hostname.localhost", "unknown"}
|
|
|
|
for _, host := range specialHosts {
|
|
|
|
if strings.EqualFold(ptr, host) {
|
|
|
|
return "[gray]" + ptr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
octets := strings.Split(ip, ".")
|
|
|
|
|
|
|
|
patterns := []string{
|
|
|
|
strings.ReplaceAll(ip, ".", "\\."),
|
|
|
|
strings.Join(reverse(strings.Split(ip, ".")), "\\."),
|
|
|
|
strings.ReplaceAll(ip, ".", "-"),
|
|
|
|
strings.Join(reverse(strings.Split(ip, ".")), "-"),
|
|
|
|
}
|
|
|
|
|
|
|
|
zeroPadded := make([]string, 4)
|
|
|
|
for i, octet := range octets {
|
|
|
|
zeroPadded[i] = fmt.Sprintf("%03d", parseInt(octet))
|
|
|
|
}
|
|
|
|
patterns = append(patterns,
|
|
|
|
strings.Join(zeroPadded, "-"),
|
|
|
|
strings.Join(reverse(zeroPadded), "-"),
|
|
|
|
)
|
|
|
|
|
|
|
|
pattern := strings.Join(patterns, "|")
|
|
|
|
re := regexp.MustCompile("(" + pattern + ")")
|
|
|
|
|
|
|
|
matches := re.FindAllStringIndex(ptr, -1)
|
|
|
|
if matches == nil {
|
2025-01-05 09:31:09 +00:00
|
|
|
return "[white]" + ptr
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var result strings.Builder
|
|
|
|
lastEnd := 0
|
|
|
|
|
|
|
|
for _, match := range matches {
|
|
|
|
if match[0] > lastEnd {
|
2025-01-05 09:31:09 +00:00
|
|
|
result.WriteString("[white]")
|
2024-12-06 22:41:03 +00:00
|
|
|
result.WriteString(ptr[lastEnd:match[0]])
|
|
|
|
}
|
|
|
|
result.WriteString("[aqua]")
|
|
|
|
result.WriteString(ptr[match[0]:match[1]])
|
|
|
|
lastEnd = match[1]
|
|
|
|
}
|
|
|
|
|
|
|
|
if lastEnd < len(ptr) {
|
2025-01-05 09:31:09 +00:00
|
|
|
result.WriteString("[white]")
|
2024-12-06 22:41:03 +00:00
|
|
|
result.WriteString(ptr[lastEnd:])
|
|
|
|
}
|
|
|
|
|
|
|
|
finalResult := result.String()
|
2025-01-05 08:51:45 +00:00
|
|
|
|
|
|
|
if strings.HasSuffix(finalResult, ".in-addr.arpa") {
|
|
|
|
finalResult = finalResult[:len(finalResult)-13] + ".[blue]in-addr.arpa"
|
|
|
|
}
|
|
|
|
if strings.HasSuffix(finalResult, ".gov") {
|
|
|
|
finalResult = finalResult[:len(finalResult)-4] + ".[red]gov"
|
|
|
|
}
|
|
|
|
if strings.HasSuffix(finalResult, ".mil") {
|
|
|
|
finalResult = finalResult[:len(finalResult)-4] + ".[red]mil"
|
|
|
|
}
|
2024-12-06 22:41:03 +00:00
|
|
|
|
|
|
|
return finalResult
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseInt(s string) int {
|
|
|
|
num := 0
|
|
|
|
fmt.Sscanf(s, "%d", &num)
|
|
|
|
return num
|
|
|
|
}
|
|
|
|
|
|
|
|
const maxBufferLines = 1000
|
|
|
|
|
|
|
|
func worker(jobs <-chan string, wg *sync.WaitGroup, cfg *Config, stats *Stats, textView *tview.TextView, app *tview.Application) {
|
|
|
|
defer wg.Done()
|
|
|
|
for ip := range jobs {
|
|
|
|
timestamp := time.Now()
|
2025-01-05 08:51:45 +00:00
|
|
|
var response DNSResponse
|
|
|
|
var err error
|
2025-01-05 09:48:27 +00:00
|
|
|
var server string
|
2024-12-06 22:41:03 +00:00
|
|
|
|
|
|
|
if len(cfg.dnsServers) > 0 {
|
2025-01-05 09:48:27 +00:00
|
|
|
response, server, err = lookupWithRetry(ip, cfg)
|
|
|
|
if idx := strings.Index(server, ":"); idx != -1 {
|
|
|
|
server = server[:idx]
|
|
|
|
}
|
2024-12-06 22:41:03 +00:00
|
|
|
} else {
|
2025-01-05 08:51:45 +00:00
|
|
|
names, err := net.LookupAddr(ip)
|
|
|
|
if err == nil {
|
|
|
|
response = DNSResponse{Names: names, RecordType: "PTR"}
|
|
|
|
}
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
stats.increment()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
stats.incrementFailed()
|
|
|
|
if cfg.debug {
|
2025-01-20 07:52:11 +00:00
|
|
|
errRecord := formatErrorAsHostname(err)
|
2025-01-05 09:48:27 +00:00
|
|
|
timeStr := time.Now().Format("2006-01-02 15:04:05")
|
|
|
|
line := fmt.Sprintf("[gray]%s [gray]│[-] [purple]%15s[-] [gray]│[-] [aqua]%-15s[-] [gray]│[-] [red] ERR [-] [gray]│[-] [gray]%-6s[-] [gray]│[-] [gray]%s[-]\n",
|
|
|
|
timeStr,
|
2024-12-06 22:41:03 +00:00
|
|
|
ip,
|
2025-01-05 09:48:27 +00:00
|
|
|
server,
|
|
|
|
"",
|
2025-01-20 07:52:11 +00:00
|
|
|
errRecord)
|
2024-12-06 22:41:03 +00:00
|
|
|
app.QueueUpdateDraw(func() {
|
2025-01-05 09:48:27 +00:00
|
|
|
fmt.Fprint(textView, line)
|
2024-12-06 22:41:03 +00:00
|
|
|
textView.ScrollToEnd()
|
|
|
|
})
|
2025-01-20 07:52:11 +00:00
|
|
|
|
|
|
|
// Write to NDJSON if enabled
|
|
|
|
writeNDJSON(cfg, time.Now(), ip, server, errRecord, "ERR", "", 0)
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
if len(response.Names) == 0 {
|
2024-12-06 22:41:03 +00:00
|
|
|
stats.incrementFailed()
|
|
|
|
if cfg.debug {
|
2025-01-05 09:48:27 +00:00
|
|
|
timeStr := time.Now().Format("2006-01-02 15:04:05")
|
|
|
|
line := fmt.Sprintf("[gray]%s [gray]│[-] [purple]%15s[-] [gray]│[-] [aqua]%-15s[-] [gray]│[-] [red] ERR [-] [gray]│[-] [gray]%-6s[-] [gray]│[-] [red]No PTR record[-]\n",
|
|
|
|
timeStr,
|
|
|
|
ip,
|
|
|
|
server,
|
|
|
|
"")
|
2024-12-06 22:41:03 +00:00
|
|
|
app.QueueUpdateDraw(func() {
|
2025-01-05 09:48:27 +00:00
|
|
|
fmt.Fprint(textView, line)
|
2024-12-06 22:41:03 +00:00
|
|
|
textView.ScrollToEnd()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
stats.incrementSuccess()
|
|
|
|
|
|
|
|
ptr := ""
|
2025-01-05 08:51:45 +00:00
|
|
|
for _, name := range response.Names {
|
2024-12-06 22:41:03 +00:00
|
|
|
if cleaned := strings.TrimSpace(strings.TrimSuffix(name, ".")); cleaned != "" {
|
2025-01-20 07:52:11 +00:00
|
|
|
ptr = strings.ToLower(cleaned)
|
2024-12-06 22:41:03 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ptr == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2025-01-05 09:48:27 +00:00
|
|
|
writeNDJSON(cfg, timestamp, ip, server, ptr, response.RecordType, response.Target, response.TTL)
|
2024-12-06 22:41:03 +00:00
|
|
|
|
|
|
|
timeStr := time.Now().Format("2006-01-02 15:04:05")
|
2025-01-05 08:51:45 +00:00
|
|
|
recordTypeColor := "[blue] PTR [-]"
|
|
|
|
if response.RecordType == "CNAME" {
|
|
|
|
stats.incrementCNAME()
|
|
|
|
recordTypeColor = "[fuchsia]CNAME[-]"
|
2025-01-20 07:52:11 +00:00
|
|
|
ptr = fmt.Sprintf("%s -> %s", strings.ToLower(ptr), strings.ToLower(response.Target))
|
2025-01-05 08:51:45 +00:00
|
|
|
}
|
2024-12-06 22:41:03 +00:00
|
|
|
|
|
|
|
var line string
|
|
|
|
if len(cfg.dnsServers) > 0 {
|
2025-01-05 09:31:09 +00:00
|
|
|
line = fmt.Sprintf("[gray]%s [gray]│[-] [purple]%15s[-] [gray]│[-] [aqua]%-15s[-] [gray]│[-] %-5s [gray]│[-] %s [gray]│[-] %s\n",
|
2024-12-06 22:41:03 +00:00
|
|
|
timeStr,
|
|
|
|
ip,
|
2025-01-05 09:48:27 +00:00
|
|
|
server,
|
2025-01-05 08:51:45 +00:00
|
|
|
recordTypeColor,
|
2025-01-05 09:31:09 +00:00
|
|
|
colorizeTTL(response.TTL),
|
2024-12-06 22:41:03 +00:00
|
|
|
colorizeIPInPtr(ptr, ip))
|
|
|
|
} else {
|
2025-01-05 09:31:09 +00:00
|
|
|
line = fmt.Sprintf("[gray]%s [gray]│[-] [purple]%15s[-] [gray]│[-] %-5s [gray]│[-] %s [gray]│[-] %s\n",
|
2024-12-06 22:41:03 +00:00
|
|
|
timeStr,
|
|
|
|
ip,
|
2025-01-05 08:51:45 +00:00
|
|
|
recordTypeColor,
|
2025-01-05 09:31:09 +00:00
|
|
|
colorizeTTL(response.TTL),
|
2024-12-06 22:41:03 +00:00
|
|
|
colorizeIPInPtr(ptr, ip))
|
|
|
|
}
|
|
|
|
|
|
|
|
app.QueueUpdateDraw(func() {
|
|
|
|
fmt.Fprint(textView, line)
|
|
|
|
content := textView.GetText(false)
|
|
|
|
lines := strings.Split(content, "\n")
|
|
|
|
if len(lines) > maxBufferLines {
|
|
|
|
newContent := strings.Join(lines[len(lines)-maxBufferLines:], "\n")
|
|
|
|
textView.Clear()
|
|
|
|
fmt.Fprint(textView, newContent)
|
|
|
|
}
|
|
|
|
textView.ScrollToEnd()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseShardArg(shard string) (int, int, error) {
|
|
|
|
if shard == "" {
|
|
|
|
return 1, 1, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
parts := strings.Split(shard, "/")
|
|
|
|
if len(parts) != 2 {
|
|
|
|
return 0, 0, fmt.Errorf("invalid shard format (expected n/total)")
|
|
|
|
}
|
|
|
|
|
|
|
|
shardNum, err := strconv.Atoi(parts[0])
|
|
|
|
if err != nil {
|
|
|
|
return 0, 0, fmt.Errorf("invalid shard number: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
totalShards, err := strconv.Atoi(parts[1])
|
|
|
|
if err != nil {
|
|
|
|
return 0, 0, fmt.Errorf("invalid total shards: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if shardNum < 1 || shardNum > totalShards {
|
|
|
|
return 0, 0, fmt.Errorf("shard number must be between 1 and total shards")
|
|
|
|
}
|
|
|
|
|
|
|
|
return shardNum, totalShards, nil
|
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
func (c *Config) updateDNSServers() error {
|
|
|
|
c.updateMu.Lock()
|
|
|
|
defer c.updateMu.Unlock()
|
|
|
|
|
|
|
|
if time.Since(c.lastDNSUpdate) < 24*time.Hour {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
resolvers, err := fetchDefaultResolvers()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(resolvers) == 0 {
|
|
|
|
return fmt.Errorf("no resolvers found in update")
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, server := range resolvers {
|
|
|
|
if !strings.Contains(server, ":") {
|
|
|
|
resolvers[i] = server + ":53"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
c.mu.Lock()
|
|
|
|
c.dnsServers = resolvers
|
|
|
|
c.serverIndex = 0
|
|
|
|
c.lastDNSUpdate = time.Now()
|
|
|
|
c.mu.Unlock()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
func main() {
|
|
|
|
concurrency := flag.Int("c", 100, "Concurrency level")
|
|
|
|
timeout := flag.Duration("t", 2*time.Second, "Timeout for DNS queries")
|
|
|
|
retries := flag.Int("r", 2, "Number of retries for failed lookups")
|
|
|
|
dnsFile := flag.String("dns", "", "File containing DNS servers (one per line)")
|
|
|
|
debug := flag.Bool("debug", false, "Show unsuccessful lookups")
|
|
|
|
outputPath := flag.String("o", "", "Path to NDJSON output file")
|
|
|
|
seed := flag.Int64("s", 0, "Seed for IP generation (0 for random)")
|
|
|
|
shard := flag.String("shard", "", "Shard specification (e.g., 1/4 for first shard of 4)")
|
2025-01-05 09:31:09 +00:00
|
|
|
loop := flag.Bool("l", false, "Loop continuously after completion")
|
2025-01-05 12:05:00 +00:00
|
|
|
jsonOutput := flag.Bool("j", false, "Output NDJSON to stdout (no TUI)")
|
2024-12-06 22:41:03 +00:00
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
shardNum, totalShards, err := parseShardArg(*shard)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error parsing shard argument: %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if *seed == 0 {
|
|
|
|
*seed = time.Now().UnixNano()
|
|
|
|
}
|
|
|
|
|
2025-01-05 08:51:45 +00:00
|
|
|
servers, err := loadDNSServers(*dnsFile)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error loading DNS servers: %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, server := range servers {
|
|
|
|
if !strings.Contains(server, ":") {
|
|
|
|
servers[i] = server + ":53"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
cfg := &Config{
|
2025-01-05 08:51:45 +00:00
|
|
|
concurrency: *concurrency,
|
|
|
|
timeout: *timeout,
|
|
|
|
retries: *retries,
|
|
|
|
debug: *debug,
|
|
|
|
dnsServers: servers,
|
|
|
|
lastDNSUpdate: time.Now(),
|
2025-01-05 09:31:09 +00:00
|
|
|
loop: *loop,
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if *outputPath != "" {
|
|
|
|
f, err := os.OpenFile(*outputPath, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0644)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error opening output file: %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
cfg.outputFile = f
|
|
|
|
defer f.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
app := tview.NewApplication()
|
|
|
|
|
|
|
|
textView := tview.NewTextView().
|
|
|
|
SetDynamicColors(true).
|
|
|
|
SetScrollable(true).
|
|
|
|
SetChangedFunc(func() {
|
|
|
|
app.Draw()
|
|
|
|
})
|
|
|
|
textView.SetBorder(true).SetTitle(" PTR Records ")
|
|
|
|
|
|
|
|
progress := tview.NewTextView().
|
|
|
|
SetDynamicColors(true).
|
|
|
|
SetTextAlign(tview.AlignLeft)
|
|
|
|
progress.SetBorder(true).SetTitle(" Progress ")
|
|
|
|
|
|
|
|
flex := tview.NewFlex().
|
|
|
|
SetDirection(tview.FlexRow).
|
|
|
|
AddItem(textView, 0, 1, false).
|
2025-01-05 09:31:09 +00:00
|
|
|
AddItem(progress, 4, 0, false)
|
2024-12-06 22:41:03 +00:00
|
|
|
|
|
|
|
stats := &Stats{
|
|
|
|
total: 1 << 32,
|
|
|
|
lastCheckTime: time.Now(),
|
2025-01-05 09:31:09 +00:00
|
|
|
startTime: time.Now(),
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
const movingAverageWindow = 5
|
|
|
|
stats.speedHistory = make([]float64, 0, movingAverageWindow)
|
|
|
|
stats.lastCheckTime = time.Now()
|
|
|
|
|
|
|
|
for {
|
|
|
|
processed := atomic.LoadUint64(&stats.processed)
|
|
|
|
success := atomic.LoadUint64(&stats.success)
|
|
|
|
failed := atomic.LoadUint64(&stats.failed)
|
|
|
|
|
|
|
|
now := time.Now()
|
|
|
|
duration := now.Sub(stats.lastCheckTime).Seconds()
|
|
|
|
|
|
|
|
if duration >= 1.0 {
|
|
|
|
stats.mu.Lock()
|
|
|
|
speed := float64(processed-stats.lastProcessed) / duration
|
|
|
|
stats.speedHistory = append(stats.speedHistory, speed)
|
|
|
|
if len(stats.speedHistory) > movingAverageWindow {
|
|
|
|
stats.speedHistory = stats.speedHistory[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
var avgSpeed float64
|
|
|
|
for _, s := range stats.speedHistory {
|
|
|
|
avgSpeed += s
|
|
|
|
}
|
|
|
|
avgSpeed /= float64(len(stats.speedHistory))
|
|
|
|
|
|
|
|
stats.lastProcessed = processed
|
|
|
|
stats.lastCheckTime = now
|
|
|
|
stats.mu.Unlock()
|
|
|
|
|
|
|
|
percent := float64(processed) / float64(stats.total) * 100
|
|
|
|
|
|
|
|
app.QueueUpdateDraw(func() {
|
|
|
|
var width int
|
|
|
|
_, _, width, _ = progress.GetInnerRect()
|
2024-12-06 23:11:19 +00:00
|
|
|
if width <= 0 {
|
2024-12-06 22:41:03 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2025-01-05 09:31:09 +00:00
|
|
|
// First line: stats
|
2025-01-20 18:47:16 +00:00
|
|
|
statsLine := fmt.Sprintf(" [aqua]Elapsed:[:-] [white]%s [gray]│[-] [aqua]Round:[:-] [white]%d [gray]│[-] [aqua]Count:[:-] [white]%s [gray]│[-] [aqua]Progress:[:-] [darkgray]%.2f%%[-] [gray]│[-] [aqua]Rate:[:-] %s [gray]│[-] [aqua]CNAMEs:[:-] [yellow]%s[-][darkgray] (%.1f%%)[-] [gray]│[-] [aqua]Successful:[:-] [green]✓ %s[-][darkgray] (%.1f%%)[-] [gray]│[-] [aqua]Failed:[:-] [red]✗ %s[-][darkgray] (%.1f%%)[-]\n",
|
2025-01-05 09:31:09 +00:00
|
|
|
formatDuration(time.Since(stats.startTime)),
|
2025-01-20 18:47:16 +00:00
|
|
|
atomic.LoadUint64(&stats.round)+1,
|
2024-12-06 22:41:03 +00:00
|
|
|
formatNumber(processed),
|
|
|
|
percent,
|
|
|
|
colorizeSpeed(avgSpeed),
|
2025-01-05 08:51:45 +00:00
|
|
|
formatNumber(atomic.LoadUint64(&stats.cnames)),
|
2025-01-05 09:31:09 +00:00
|
|
|
float64(atomic.LoadUint64(&stats.cnames))/float64(processed)*100,
|
2024-12-06 22:41:03 +00:00
|
|
|
formatNumber(success),
|
|
|
|
float64(success)/float64(processed)*100,
|
|
|
|
formatNumber(failed),
|
|
|
|
float64(failed)/float64(processed)*100)
|
|
|
|
|
2025-01-05 09:31:09 +00:00
|
|
|
// Second line: progress bar
|
|
|
|
barWidth := width - 3 // -3 for the [] and space
|
2024-12-06 23:11:19 +00:00
|
|
|
if barWidth < 1 {
|
|
|
|
progress.Clear()
|
2025-01-05 09:31:09 +00:00
|
|
|
fmt.Fprint(progress, statsLine)
|
2024-12-06 23:11:19 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
filled := int(float64(barWidth) * (percent / 100))
|
|
|
|
if filled > barWidth {
|
|
|
|
filled = barWidth
|
|
|
|
}
|
|
|
|
|
2025-01-05 09:31:09 +00:00
|
|
|
barLine := fmt.Sprintf(" [%s%s]",
|
|
|
|
strings.Repeat("█", filled),
|
|
|
|
strings.Repeat("░", barWidth-filled))
|
2024-12-06 22:41:03 +00:00
|
|
|
|
2025-01-05 09:31:09 +00:00
|
|
|
// Combine both lines with explicit newline
|
2024-12-06 22:41:03 +00:00
|
|
|
progress.Clear()
|
2025-01-05 09:31:09 +00:00
|
|
|
fmt.Fprintf(progress, "%s%s", statsLine, barLine)
|
2024-12-06 22:41:03 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2025-01-05 12:05:00 +00:00
|
|
|
if *jsonOutput {
|
|
|
|
// JSON-only mode
|
|
|
|
jobs := make(chan string, cfg.concurrency)
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
// Start workers
|
|
|
|
for i := 0; i < cfg.concurrency; i++ {
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
for ip := range jobs {
|
|
|
|
var response DNSResponse
|
|
|
|
var err error
|
|
|
|
var server string
|
|
|
|
|
|
|
|
if len(cfg.dnsServers) > 0 {
|
|
|
|
response, server, err = lookupWithRetry(ip, cfg)
|
|
|
|
if idx := strings.Index(server, ":"); idx != -1 {
|
|
|
|
server = server[:idx]
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
names, err := net.LookupAddr(ip)
|
|
|
|
if err == nil {
|
|
|
|
response = DNSResponse{Names: names, RecordType: "PTR"}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-20 07:52:11 +00:00
|
|
|
if err != nil {
|
|
|
|
if cfg.debug {
|
|
|
|
errRecord := formatErrorAsHostname(err)
|
|
|
|
record := struct {
|
|
|
|
Seen string `json:"seen"`
|
|
|
|
IP string `json:"ip"`
|
|
|
|
Nameserver string `json:"nameserver"`
|
|
|
|
Record string `json:"record"`
|
|
|
|
RecordType string `json:"record_type"`
|
|
|
|
TTL uint32 `json:"ttl"`
|
|
|
|
}{
|
|
|
|
Seen: time.Now().Format(time.RFC3339),
|
|
|
|
IP: ip,
|
|
|
|
Nameserver: server,
|
|
|
|
Record: errRecord,
|
|
|
|
RecordType: "ERR",
|
|
|
|
TTL: 0,
|
|
|
|
}
|
|
|
|
if data, err := json.Marshal(record); err == nil {
|
|
|
|
fmt.Println(string(data))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(response.Names) == 0 {
|
|
|
|
if cfg.debug {
|
|
|
|
record := struct {
|
|
|
|
Seen string `json:"seen"`
|
|
|
|
IP string `json:"ip"`
|
|
|
|
Nameserver string `json:"nameserver"`
|
|
|
|
Record string `json:"record"`
|
|
|
|
RecordType string `json:"record_type"`
|
|
|
|
TTL uint32 `json:"ttl"`
|
|
|
|
}{
|
|
|
|
Seen: time.Now().Format(time.RFC3339),
|
|
|
|
IP: ip,
|
|
|
|
Nameserver: server,
|
|
|
|
Record: "FAIL.NO-PTR-RECORD.in-addr.arpa",
|
|
|
|
RecordType: "ERR",
|
|
|
|
TTL: 0,
|
|
|
|
}
|
|
|
|
if data, err := json.Marshal(record); err == nil {
|
|
|
|
fmt.Println(string(data))
|
|
|
|
}
|
|
|
|
}
|
2025-01-05 12:05:00 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
ptr := ""
|
|
|
|
for _, name := range response.Names {
|
|
|
|
if cleaned := strings.TrimSpace(strings.TrimSuffix(name, ".")); cleaned != "" {
|
|
|
|
ptr = cleaned
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ptr == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
record := struct {
|
|
|
|
Seen string `json:"seen"`
|
|
|
|
IP string `json:"ip"`
|
|
|
|
Nameserver string `json:"nameserver"`
|
|
|
|
Record string `json:"record"`
|
|
|
|
RecordType string `json:"record_type"`
|
|
|
|
TTL uint32 `json:"ttl"`
|
|
|
|
}{
|
|
|
|
Seen: time.Now().Format(time.RFC3339),
|
|
|
|
IP: ip,
|
|
|
|
Nameserver: server,
|
|
|
|
Record: response.Target,
|
|
|
|
RecordType: response.RecordType,
|
|
|
|
TTL: response.TTL,
|
|
|
|
}
|
|
|
|
|
|
|
|
if response.RecordType != "CNAME" {
|
|
|
|
record.Record = ptr
|
|
|
|
}
|
|
|
|
|
|
|
|
if data, err := json.Marshal(record); err == nil {
|
|
|
|
fmt.Println(string(data))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Feed IPs to workers
|
|
|
|
for {
|
|
|
|
stream, err := golcg.IPStream("0.0.0.0/0", shardNum, totalShards, int(*seed), nil)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Fprintf(os.Stderr, "Error creating IP stream: %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for ip := range stream {
|
|
|
|
jobs <- ip
|
|
|
|
}
|
|
|
|
|
|
|
|
if !cfg.loop {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close(jobs)
|
|
|
|
wg.Wait()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
jobs := make(chan string, cfg.concurrency)
|
|
|
|
|
2025-01-05 09:31:09 +00:00
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
stream, err := golcg.IPStream("0.0.0.0/0", shardNum, totalShards, int(*seed), nil)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error creating IP stream: %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
for ip := range stream {
|
|
|
|
jobs <- ip
|
|
|
|
}
|
|
|
|
|
|
|
|
if !cfg.loop {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close(jobs)
|
|
|
|
}()
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
var wg sync.WaitGroup
|
|
|
|
for i := 0; i < cfg.concurrency; i++ {
|
|
|
|
wg.Add(1)
|
|
|
|
go worker(jobs, &wg, cfg, stats, textView, app)
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
wg.Wait()
|
|
|
|
app.Stop()
|
|
|
|
}()
|
|
|
|
|
|
|
|
if err := app.SetRoot(flex, true).EnableMouse(true).Run(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func formatNumber(n uint64) string {
|
|
|
|
s := fmt.Sprint(n)
|
|
|
|
parts := make([]string, 0)
|
|
|
|
for i := len(s); i > 0; i -= 3 {
|
|
|
|
start := i - 3
|
|
|
|
if start < 0 {
|
|
|
|
start = 0
|
|
|
|
}
|
|
|
|
parts = append([]string{s[start:i]}, parts...)
|
|
|
|
}
|
2025-01-05 09:36:53 +00:00
|
|
|
return strings.Join(parts, ",")
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func colorizeSpeed(speed float64) string {
|
|
|
|
switch {
|
|
|
|
case speed >= 500:
|
|
|
|
return fmt.Sprintf("[green]%5.0f/s[-]", speed)
|
|
|
|
case speed >= 350:
|
|
|
|
return fmt.Sprintf("[yellow]%5.0f/s[-]", speed)
|
|
|
|
case speed >= 200:
|
|
|
|
return fmt.Sprintf("[orange]%5.0f/s[-]", speed)
|
|
|
|
case speed >= 100:
|
|
|
|
return fmt.Sprintf("[red]%5.0f/s[-]", speed)
|
|
|
|
default:
|
|
|
|
return fmt.Sprintf("[gray]%5.0f/s[-]", speed)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func visibleLength(s string) int {
|
|
|
|
noColors := regexp.MustCompile(`\[[a-zA-Z:-]*\]`).ReplaceAllString(s, "")
|
|
|
|
return len(noColors)
|
|
|
|
}
|
|
|
|
|
2025-01-05 08:59:48 +00:00
|
|
|
func writeNDJSON(cfg *Config, timestamp time.Time, ip, server, ptr, recordType, target string, ttl uint32) {
|
2024-12-06 22:41:03 +00:00
|
|
|
if cfg.outputFile == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
record := struct {
|
2025-01-05 12:05:00 +00:00
|
|
|
Seen string `json:"seen"`
|
|
|
|
IP string `json:"ip"`
|
|
|
|
Nameserver string `json:"nameserver"`
|
|
|
|
Record string `json:"record"`
|
2025-01-05 08:51:45 +00:00
|
|
|
RecordType string `json:"record_type"`
|
2025-01-05 08:59:48 +00:00
|
|
|
TTL uint32 `json:"ttl"`
|
2024-12-06 22:41:03 +00:00
|
|
|
}{
|
2025-01-05 12:05:00 +00:00
|
|
|
Seen: timestamp.Format(time.RFC3339),
|
|
|
|
IP: ip,
|
|
|
|
Nameserver: server,
|
|
|
|
Record: target, // For CNAME records, use the target
|
2025-01-05 08:51:45 +00:00
|
|
|
RecordType: recordType,
|
2025-01-05 08:59:48 +00:00
|
|
|
TTL: ttl,
|
2024-12-06 22:41:03 +00:00
|
|
|
}
|
|
|
|
|
2025-01-05 12:05:00 +00:00
|
|
|
// If it's not a CNAME, use the PTR record
|
|
|
|
if recordType != "CNAME" {
|
|
|
|
record.Record = ptr
|
|
|
|
}
|
|
|
|
|
2024-12-06 22:41:03 +00:00
|
|
|
if data, err := json.Marshal(record); err == nil {
|
|
|
|
cfg.mu.Lock()
|
|
|
|
cfg.outputFile.Write(data)
|
|
|
|
cfg.outputFile.Write([]byte("\n"))
|
|
|
|
cfg.mu.Unlock()
|
|
|
|
}
|
|
|
|
}
|
2025-01-05 09:31:09 +00:00
|
|
|
|
|
|
|
func formatDuration(d time.Duration) string {
|
|
|
|
d = d.Round(time.Second)
|
|
|
|
|
|
|
|
days := d / (24 * time.Hour)
|
|
|
|
d -= days * 24 * time.Hour
|
|
|
|
|
|
|
|
hours := d / time.Hour
|
|
|
|
d -= hours * time.Hour
|
|
|
|
|
|
|
|
minutes := d / time.Minute
|
|
|
|
d -= minutes * time.Minute
|
|
|
|
|
|
|
|
seconds := d / time.Second
|
|
|
|
|
|
|
|
var result string
|
|
|
|
|
|
|
|
if days > 0 {
|
|
|
|
if hours > 0 && minutes > 0 {
|
|
|
|
result = fmt.Sprintf("%dd %dh %dm", days, hours, minutes)
|
|
|
|
} else if hours > 0 {
|
|
|
|
result = fmt.Sprintf("%dd %dh", days, hours)
|
|
|
|
} else {
|
|
|
|
result = fmt.Sprintf("%dd", days)
|
|
|
|
}
|
|
|
|
} else if hours > 0 {
|
|
|
|
if minutes > 0 {
|
|
|
|
result = fmt.Sprintf("%dh %dm", hours, minutes)
|
|
|
|
} else {
|
|
|
|
result = fmt.Sprintf("%dh", hours)
|
|
|
|
}
|
|
|
|
} else if minutes > 0 {
|
|
|
|
if seconds > 0 {
|
|
|
|
result = fmt.Sprintf("%dm %ds", minutes, seconds)
|
|
|
|
} else {
|
|
|
|
result = fmt.Sprintf("%dm", minutes)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
result = fmt.Sprintf("%ds", seconds)
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func colorizeTTL(ttl uint32) string {
|
|
|
|
switch {
|
|
|
|
case ttl >= 86400: // 1 day or more
|
|
|
|
return fmt.Sprintf("[#00FF00::b]%-6d[-]", ttl) // Bright green with bold
|
|
|
|
case ttl >= 3600: // 1 hour or more
|
|
|
|
return fmt.Sprintf("[yellow]%-6d[-]", ttl)
|
|
|
|
case ttl >= 300: // 5 minutes or more
|
|
|
|
return fmt.Sprintf("[orange]%-6d[-]", ttl)
|
|
|
|
case ttl >= 60: // 1 minute or more
|
|
|
|
return fmt.Sprintf("[red]%-6d[-]", ttl)
|
|
|
|
default: // Less than 60 seconds
|
|
|
|
return fmt.Sprintf("[gray]%-6d[-]", ttl)
|
|
|
|
}
|
|
|
|
}
|
2025-01-20 07:52:11 +00:00
|
|
|
|
|
|
|
func formatErrorAsHostname(err error) string {
|
|
|
|
errMsg := err.Error()
|
|
|
|
if idx := strings.LastIndex(errMsg, ": "); idx != -1 {
|
|
|
|
errMsg = errMsg[idx+2:]
|
|
|
|
}
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case strings.Contains(errMsg, "i/o timeout"):
|
|
|
|
return "FAIL.TIMEOUT.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "Server Failure"):
|
|
|
|
return "FAIL.SERVER-FAILURE.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "No Such Domain"):
|
|
|
|
return "FAIL.NON-AUTHORITATIVE.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "refused"):
|
|
|
|
return "FAIL.REFUSED.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "no such host"):
|
|
|
|
return "FAIL.NO-SUCH-HOST.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "connection refused"):
|
|
|
|
return "FAIL.CONNECTION-REFUSED.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "network is unreachable"):
|
|
|
|
return "FAIL.NETWORK-UNREACHABLE.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "no route to host"):
|
|
|
|
return "FAIL.NO-ROUTE.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "Format error"):
|
|
|
|
return "FAIL.FORMAT-ERROR.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "Not Implemented"):
|
|
|
|
return "FAIL.NOT-IMPLEMENTED.in-addr.arpa"
|
|
|
|
case strings.Contains(errMsg, "truncated"):
|
|
|
|
return "FAIL.TRUNCATED.in-addr.arpa"
|
|
|
|
default:
|
|
|
|
return fmt.Sprintf("FAIL.%s.in-addr.arpa", strings.ReplaceAll(strings.ToUpper(errMsg), " ", "-"))
|
|
|
|
}
|
|
|
|
}
|