2025-01-29 19:49:51 +01:00
|
|
|
// Fail2ban UI - A Swiss made, management interface for Fail2ban.
|
|
|
|
|
//
|
|
|
|
|
// Copyright (C) 2025 Swissmakers GmbH (https://swissmakers.ch)
|
|
|
|
|
//
|
|
|
|
|
// Licensed under the GNU General Public License, Version 3 (GPL-3.0)
|
|
|
|
|
// You may not use this file except in compliance with the License.
|
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
|
//
|
|
|
|
|
// https://www.gnu.org/licenses/gpl-3.0.en.html
|
|
|
|
|
//
|
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
2025-01-25 16:21:14 +01:00
|
|
|
package fail2ban
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"bufio"
|
|
|
|
|
"fmt"
|
|
|
|
|
"os"
|
|
|
|
|
"regexp"
|
|
|
|
|
"time"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var (
|
|
|
|
|
// Typical fail2ban log line:
|
|
|
|
|
// 2023-01-20 10:15:30,123 fail2ban.actions [1234]: NOTICE [sshd] Ban 192.168.0.101
|
|
|
|
|
logRegex = regexp.MustCompile(`^(\S+\s+\S+) fail2ban\.actions.*?\[\d+\]: NOTICE\s+\[(\S+)\]\s+Ban\s+(\S+)`)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// BanEvent holds details about a ban
|
|
|
|
|
type BanEvent struct {
|
|
|
|
|
Time time.Time
|
|
|
|
|
Jail string
|
|
|
|
|
IP string
|
|
|
|
|
LogLine string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ParseBanLog returns a map[jailName]BanEvents and also the last 5 ban events overall.
|
|
|
|
|
func ParseBanLog(logPath string) (map[string][]BanEvent, error) {
|
|
|
|
|
file, err := os.Open(logPath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, fmt.Errorf("failed to open fail2ban log: %v", err)
|
|
|
|
|
}
|
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
|
|
eventsByJail := make(map[string][]BanEvent)
|
|
|
|
|
|
|
|
|
|
scanner := bufio.NewScanner(file)
|
|
|
|
|
for scanner.Scan() {
|
|
|
|
|
line := scanner.Text()
|
|
|
|
|
|
|
|
|
|
matches := logRegex.FindStringSubmatch(line)
|
|
|
|
|
if len(matches) == 4 {
|
|
|
|
|
// matches[1] -> "2023-01-20 10:15:30,123"
|
|
|
|
|
// matches[2] -> jail name, e.g. "sshd"
|
|
|
|
|
// matches[3] -> IP, e.g. "192.168.0.101"
|
|
|
|
|
timestampStr := matches[1]
|
|
|
|
|
jail := matches[2]
|
|
|
|
|
ip := matches[3]
|
|
|
|
|
|
|
|
|
|
// parse "2023-01-20 10:15:30,123" -> time.Time
|
|
|
|
|
parsedTime, err := time.Parse("2006-01-02 15:04:05,000", timestampStr)
|
|
|
|
|
if err != nil {
|
|
|
|
|
// If parse fails, skip or set parsedTime=zero
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ev := BanEvent{
|
|
|
|
|
Time: parsedTime,
|
|
|
|
|
Jail: jail,
|
|
|
|
|
IP: ip,
|
|
|
|
|
LogLine: line,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
eventsByJail[jail] = append(eventsByJail[jail], ev)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return eventsByJail, nil
|
|
|
|
|
}
|