package main import ( "fmt" "io" "os" "regexp" "strconv" "strings" "sync" "syscall" "github.com/elastic/go-libaudit/v2" "github.com/elastic/go-libaudit/v2/auparse" ) // 为每个事务id存储其信息,事务id在操作系统运行期间是唯一的 var eventTable sync.Map // 事件信息 var tmp any var ok bool var event Event var pEvent *Event var eventId, argc int // 要用的正则匹配列表 var ( syscallRegex = regexp.MustCompile(`audit\((\d+\.\d+):(\d+)\).*?syscall=(\d+)(?:.*?exit=([-+]?\d+))?.*?ppid=(\d+) pid=(\d+).*?subj=(.*?):(.*?):(.*?):(.*?) .*?$`) execveRegex = regexp.MustCompile(`audit\(\d+\.\d+:(\d+)\): argc=(\d+)`) argsRegex = regexp.MustCompile(`a\d+=("(.*?)"|([0-9a-fA-F]+))`) pathRegex = regexp.MustCompile(`audit\(\d+\.\d+:(\d+)\): item=(\d+) name="(.*?)" .*objtype=([A-Z]+) `) cwdRegex = regexp.MustCompile(`audit\(\d+\.\d+:(\d+)\): cwd="(.*?)"`) proctitleRegex = regexp.MustCompile(`audit\(\d+\.\d+:(\d+)\): proctitle=("(.*?)"|([0-9a-fA-F]+))$`) eoeRegex = regexp.MustCompile(`audit\(\d+\.\d+:(\d+)\)`) ) func orgnaze() { defer wg.Done() defer close(cookedChan) // 接收信息 var raw interface{} var rawEvent libaudit.RawAuditMessage var diagWriter io.Writer var f *os.File var err error var fileName string if *diag != "" { fileName = *diag } else { fileName = "godo.log" } f, err = os.OpenFile(fileName, os.O_CREATE|os.O_RDWR|os.O_TRUNC, 0o664) if err != nil { f = nil } defer f.Close() diagWriter = f for { raw, ok = <-rawChan if !ok { break } rawEvent = raw.(libaudit.RawAuditMessage) fmt.Fprintf(diagWriter, "type=%v msg=%s\n", rawEvent.Type, rawEvent.Data) switch rawEvent.Type { case auparse.AUDIT_SYSCALL: syscallRaw(rawEvent) case auparse.AUDIT_EXECVE: execve(rawEvent) case auparse.AUDIT_CWD: cwd(rawEvent) case auparse.AUDIT_PATH: path(rawEvent) case auparse.AUDIT_PROCTITLE: proctitle(rawEvent) case auparse.AUDIT_EOE: eoe(rawEvent) default: } } } func syscallRaw(rawEvent libaudit.RawAuditMessage) { if !syscallRegex.Match(rawEvent.Data) { return } var exit int var a [4]uint64 var subj [4]string // 捕获基础信息 match := syscallRegex.FindSubmatch(rawEvent.Data) event.timestamp, _ = getTimeFromStr(string(match[1])) eventId, _ = strconv.Atoi(string(match[2])) event.syscall, _ = strconv.Atoi(string(match[3])) if string(match[4]) == "" { // exit没捕获到 exit = 0 } else { exit, _ = strconv.Atoi(string(match[4])) } event.ppid, _ = strconv.Atoi(string(match[5])) event.pid, _ = strconv.Atoi(string(match[6])) // 几个subj,说不定会有用 for i := 0; i < 4; i++ { subj[i] = string(match[7+i]) } // 捕获参数 if !argsRegex.Match(rawEvent.Data) { fmt.Fprintf(os.Stderr, "Error: don't get args in syscall event!\n") return } argsMatch := argsRegex.FindAllSubmatch(rawEvent.Data, -1) for i := 0; i < 4; i++ { a[i], _ = strconv.ParseUint(string(argsMatch[i][3]), 16, 64) } switch syscallTable[event.syscall] { case "execve": eventTable.Store(eventId, &Event{ tag: EXECVE, timestamp: event.timestamp, syscall: event.syscall, // exit_code: a[0], // 为啥这么写? ppid: event.ppid, pid: event.pid, argc: 0, argv: make([]string, 0), cwd: "", }) case "open": // 检查打开的权限 if a[1]&(syscall.O_APPEND|syscall.O_WRONLY|syscall.O_RDWR|syscall.O_TRUNC) == 0 { break } // TRUNC应该被直接标记为改变,而不是打开 eventTable.Store(eventId, &Event{ tag: FILEOPEN, timestamp: event.timestamp, syscall: event.syscall, exit_code: exit, ppid: event.ppid, pid: event.pid, argc: 0, argv: make([]string, 0), cwd: "", syscallParam: a, srcPath: "", }) case "write": eventTable.Store(eventId, &Event{ tag: FILEWRITE, timestamp: event.timestamp, syscall: event.syscall, exit_code: exit, ppid: event.ppid, pid: event.pid, argc: 0, argv: make([]string, 0), cwd: "", syscallParam: a, }) case "close": // 文件关闭 eventTable.Store(eventId, &Event{ tag: FILECLOSE, timestamp: event.timestamp, syscall: event.syscall, exit_code: exit, ppid: event.ppid, pid: event.pid, argc: 0, argv: make([]string, 0), cwd: "", syscallParam: a, }) case "pivot_root": if subj[2] == "container_runtime_t" { eventTable.Store(eventId, &Event{ tag: PIVOTROOT, timestamp: event.timestamp, syscall: event.syscall, ppid: event.ppid, pid: event.pid, syscallParam: a, }) } } } func execve(rawEvent libaudit.RawAuditMessage) { if !execveRegex.Match(rawEvent.Data) { return } match := execveRegex.FindSubmatch(rawEvent.Data) eventId, _ = strconv.Atoi(string(match[1])) argc, _ = strconv.Atoi(string(match[2])) tmp, ok = eventTable.Load(eventId) if !ok { return } pEvent = tmp.(*Event) if argsRegex.Match(rawEvent.Data) { match := argsRegex.FindAllSubmatch(rawEvent.Data, -1) for i := 0; i < argc; i++ { if len(match[i][2]) == 0 { // 代表着匹配到的是十六进制数 str := hexToAscii(string(match[i][3])) pEvent.argv = append(pEvent.argv, str) } else { pEvent.argv = append(pEvent.argv, string(match[i][2])) } } pEvent.argc = argc } } func cwd(rawEvent libaudit.RawAuditMessage) { if !cwdRegex.Match(rawEvent.Data) { return } match := cwdRegex.FindSubmatch(rawEvent.Data) eventId, _ = strconv.Atoi(string(match[1])) tmp, ok = eventTable.Load(eventId) if !ok { return } tmp.(*Event).cwd = string(match[2]) } func proctitle(rawEvent libaudit.RawAuditMessage) { if !proctitleRegex.Match(rawEvent.Data) { return } var cmdline string match := proctitleRegex.FindSubmatch(rawEvent.Data) eventId, _ = strconv.Atoi(string(match[1])) tmp, ok = eventTable.Load(eventId) if !ok { return } pEvent = tmp.(*Event) if pEvent.argc == 0 { // 只有等于0,才证明没经过EXECVE提取参数,才允许使用PROCTITLE提取参数 if match[3] == nil { // PROCTITLE写的是十六进制,转换为字符串 cmdline = hexToAscii(string(match[4])) } else { cmdline = string(match[3]) } pEvent.argv = strings.Split(cmdline, " ") pEvent.argc = len(pEvent.argv) } } func eoe(rawEvent libaudit.RawAuditMessage) { if !eoeRegex.Match(rawEvent.Data) { return } match := eoeRegex.FindSubmatch(rawEvent.Data) eventId, _ = strconv.Atoi(string(match[1])) tmp, ok = eventTable.Load(eventId) if !ok { return } cooked := *(tmp.(*Event)) cookedChan <- cooked eventTable.Delete(eventId) // 死人别占地 } func path(rawEvent libaudit.RawAuditMessage) { if !pathRegex.Match(rawEvent.Data) { return } match := pathRegex.FindSubmatch(rawEvent.Data) eventId, _ = strconv.Atoi(string(match[1])) // item, _ := strconv.Atoi(string(match[2])) name := string(match[3]) objtype := string(match[4]) tmp, ok = eventTable.Load(eventId) if !ok { return } pEvent = tmp.(*Event) // 先看看是不是文件操作,再看是不是所在目录 if pEvent.tag != FILEOPEN || objtype == "PARENT" { return } if pEvent.cwd == "/" || name[0] == '/' { pEvent.srcPath = name } else { pEvent.srcPath = pEvent.cwd + "/" + name } // ATTENTION: 这里需要做路径简化,留给过滤清洗流程吧 }