package blocker import ( "fmt" "os/exec" "strings" "github.com/d3m0k1d/BanForge/internal/logger" ) type Nftables struct { logger *logger.Logger config string } func NewNftables(logger *logger.Logger, config string) *Nftables { return &Nftables{ logger: logger, config: config, } } func (n *Nftables) Ban(ip string) error { err := validateIP(ip) if err != nil { return err } cmd := exec.Command("sudo", "nft", "add", "rule", "inet", "banforge", "banned", "ip", "saddr", ip, "drop") output, err := cmd.CombinedOutput() if err != nil { n.logger.Error("failed to ban IP", "ip", ip, "error", err.Error(), "output", string(output)) return err } n.logger.Info("IP banned", "ip", ip) err = saveNftablesConfig(n.config) if err != nil { n.logger.Error("failed to save config", "config_path", n.config, "error", err.Error()) return err } n.logger.Info("config saved", "config_path", n.config) return nil } func (n *Nftables) Unban(ip string) error { err := validateIP(ip) if err != nil { return err } handle, err := n.findRuleHandle(ip) if err != nil { n.logger.Error("failed to find rule handle", "ip", ip, "error", err.Error()) return err } if handle == "" { n.logger.Warn("no rule found for IP", "ip", ip) return fmt.Errorf("no rule found for IP %s", ip) } // #nosec G204 - handle is extracted from nftables output and validated cmd := exec.Command("sudo", "nft", "delete", "rule", "inet", "banforge", "banned", "handle", handle) output, err := cmd.CombinedOutput() if err != nil { n.logger.Error("failed to unban IP", "ip", ip, "handle", handle, "error", err.Error(), "output", string(output)) return err } n.logger.Info("IP unbanned", "ip", ip, "handle", handle) err = saveNftablesConfig(n.config) if err != nil { n.logger.Error("failed to save config", "config_path", n.config, "error", err.Error()) return err } n.logger.Info("config saved", "config_path", n.config) return nil } func SetupNftables(config string) error { err := validateConfigPath(config) if err != nil { return err } cmd := exec.Command("sudo", "nft", "list", "table", "inet", "banforge") if err := cmd.Run(); err != nil { cmd = exec.Command("sudo", "nft", "add", "table", "inet", "banforge") output, err := cmd.CombinedOutput() if err != nil { return fmt.Errorf("failed to create table: %s", string(output)) } } cmd = exec.Command("sudo", "nft", "list", "chain", "inet", "banforge", "input") if err := cmd.Run(); err != nil { script := "sudo nft 'add chain inet banforge input { type filter hook input priority 0; policy accept; }'" cmd = exec.Command("bash", "-c", script) output, err := cmd.CombinedOutput() if err != nil { return fmt.Errorf("failed to create input chain: %s", string(output)) } } err = saveNftablesConfig(config) if err != nil { return fmt.Errorf("failed to save nftables config: %w", err) } return nil } func (n *Nftables) findRuleHandle(ip string) (string, error) { cmd := exec.Command("sudo", "nft", "-a", "list", "chain", "inet", "banforge", "banned") output, err := cmd.CombinedOutput() if err != nil { return "", fmt.Errorf("failed to list chain rules: %w", err) } lines := strings.Split(string(output), "\n") for _, line := range lines { if strings.Contains(line, ip) && strings.Contains(line, "drop") { if idx := strings.Index(line, "# handle"); idx != -1 { parts := strings.Fields(line[idx:]) if len(parts) >= 3 && parts[1] == "handle" { return parts[2], nil } } } } return "", nil } func saveNftablesConfig(configPath string) error { err := validateConfigPath(configPath) if err != nil { return err } cmd := exec.Command("sudo", "nft", "list", "ruleset") output, err := cmd.CombinedOutput() if err != nil { return fmt.Errorf("failed to get nftables ruleset: %w", err) } cmd = exec.Command("sudo", "tee", configPath) stdin, err := cmd.StdinPipe() if err != nil { return fmt.Errorf("failed to create stdin pipe: %w", err) } if err := cmd.Start(); err != nil { return fmt.Errorf("failed to start tee command: %w", err) } _, err = stdin.Write(output) if err != nil { return fmt.Errorf("failed to write to config file: %w", err) } err = stdin.Close() if err != nil { return fmt.Errorf("failed to close stdin pipe: %w", err) } if err := cmd.Wait(); err != nil { return fmt.Errorf("failed to save config: %w", err) } return nil }