Skip to content

Commit 3119cc0

Browse files
committedSep 25, 2024
add logscanner.go, tool that allow parse build logs 300x times faster than check_error.py
1 parent e1942df commit 3119cc0

File tree

3 files changed

+196
-0
lines changed

3 files changed

+196
-0
lines changed
 

‎errors.txt

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
Segmentation fault
2+
A fatal error has been detected by the Java Runtime Environment
3+
error:(.*)field(.*)has incomplete type
4+
error:(.*)conflicting types for
5+
No rule to make target (.*)
6+
failed to allocate (.*) bytes for output file: Cannot allocate memory
7+
cp: cannot stat (.*):(.*)
8+
can't find file to patch at input line (.*)
9+
No matching package to install: (.*)
10+
package (.*) requires (.*) but none of the providers can be installed$
11+
unable to execute command: (.*)
12+
(.*) is a protected member of (.*)
13+
undefined reference to (.*)
14+
no matching function for call to (.*)
15+
bytecode stream in file (.*) generated with (.*)
16+
Could not find a configuration file for package (.*)
17+
variable has incomplete type (.*)
18+
File must begin with (.*)
19+
Bad source: (.*)
20+
File not found: (.*)
21+
Installed \(but unpackaged\) file\(s\) found
22+
cannot find -l(.*)
23+
implicit declaration of function (.*)
24+
'(.*)' file not found
25+
use of undeclared identifier (.*)
26+
function cannot return function type (.*)
27+
unknown type name (.*)
28+
incomplete definition of type (.*)
29+
Problem encountered: Man pages cannot be built: (.*)
30+
format string is not a string literal (.*)
31+
Failed to find required (.*) component (.*)
32+
Package (.*), required by (.*), not found
33+
CMake Error at (.*)
34+
error:(.*)
35+
Couldn't find include (.*)
36+
(.*) Stop

‎logscanner.go

+158
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,158 @@
1+
package main
2+
3+
import (
4+
"bufio"
5+
"compress/gzip"
6+
"flag"
7+
"fmt"
8+
"io"
9+
"os"
10+
"regexp"
11+
"strings"
12+
"sync"
13+
)
14+
15+
// Load patterns from a file
16+
func loadPatterns(filename string) ([]string, error) {
17+
file, err := os.Open(filename)
18+
if err != nil {
19+
return nil, err
20+
}
21+
defer file.Close()
22+
23+
var patterns []string
24+
scanner := bufio.NewScanner(file)
25+
for scanner.Scan() {
26+
pattern := strings.TrimSpace(scanner.Text())
27+
if pattern != "" {
28+
patterns = append(patterns, pattern)
29+
}
30+
}
31+
32+
if err := scanner.Err(); err != nil {
33+
return nil, err
34+
}
35+
36+
return patterns, nil
37+
}
38+
39+
// Check if the line matches any exclusion patterns
40+
func isExcluded(line string, notAnError *regexp.Regexp) bool {
41+
return notAnError.MatchString(line)
42+
}
43+
44+
// Check if the line matches any error patterns and write to output if found
45+
func checkLine(line string, errorPattern, notAnError *regexp.Regexp, mu *sync.Mutex, wg *sync.WaitGroup, output *os.File, outputToFile bool) {
46+
defer wg.Done()
47+
if errorPattern.MatchString(line) && !isExcluded(line, notAnError) {
48+
mu.Lock()
49+
if outputToFile {
50+
output.WriteString(line + "\n") // Write to file
51+
} else {
52+
fmt.Println("Found error:", line) // Print to console
53+
}
54+
mu.Unlock()
55+
}
56+
}
57+
58+
// Process the log file
59+
func processLogFile(logFile string, errorPatterns, notErrorPatterns []string, outputFile *os.File, outputToFile bool) {
60+
file, err := os.Open(logFile)
61+
if err != nil {
62+
fmt.Println("Error opening log file:", err)
63+
return
64+
}
65+
defer file.Close()
66+
67+
var reader io.Reader
68+
69+
// Check if the file is gzipped
70+
if strings.HasSuffix(logFile, ".gz") {
71+
gzipReader, err := gzip.NewReader(file)
72+
if err != nil {
73+
fmt.Println("Error opening gzip file:", err)
74+
return
75+
}
76+
defer gzipReader.Close()
77+
reader = gzipReader
78+
} else {
79+
reader = file
80+
}
81+
82+
bufReader := bufio.NewReader(reader)
83+
var wg sync.WaitGroup
84+
var mu sync.Mutex
85+
86+
// Combine all error patterns into one regex
87+
combinedErrorPattern := strings.Join(errorPatterns, "|")
88+
errorRegex := regexp.MustCompile(combinedErrorPattern)
89+
90+
// Combine all not-error patterns into one regex
91+
combinedNotErrorPattern := strings.Join(notErrorPatterns, "|")
92+
notErrorRegex := regexp.MustCompile(combinedNotErrorPattern)
93+
94+
// Read and check each line
95+
for {
96+
line, err := bufReader.ReadString('\n')
97+
if err != nil {
98+
if err == io.EOF {
99+
break // Exit loop on EOF
100+
}
101+
fmt.Println("Error reading log file:", err)
102+
return
103+
}
104+
wg.Add(1)
105+
go checkLine(strings.TrimSpace(line), errorRegex, notErrorRegex, &mu, &wg, outputFile, outputToFile)
106+
}
107+
108+
wg.Wait() // Wait for all goroutines to finish
109+
}
110+
111+
func main() {
112+
// Define command-line flags
113+
logFile := flag.String("log", "", "Path to the log file (supports .gz files)")
114+
errorsFile := flag.String("errors", "errors.txt", "Path to the error patterns file")
115+
notErrorsFile := flag.String("ignore", "", "Path to the ignore patterns file")
116+
outputFile := flag.String("output", "", "Path to the output file for found errors")
117+
flag.Parse()
118+
119+
// Validate input
120+
if *logFile == "" {
121+
fmt.Println("Log file must be specified using --log")
122+
return
123+
}
124+
125+
// Load error patterns from the specified file
126+
errorPatterns, err := loadPatterns(*errorsFile)
127+
if err != nil {
128+
fmt.Printf("Error loading error patterns from %s: %v\n", *errorsFile, err)
129+
return
130+
}
131+
132+
// Load not-error patterns from the specified file if provided
133+
var notErrorPatterns []string
134+
if *notErrorsFile != "" {
135+
notErrorPatterns, err = loadPatterns(*notErrorsFile)
136+
if err != nil {
137+
fmt.Printf("Error loading not error patterns from %s: %v\n", *notErrorsFile, err)
138+
return
139+
}
140+
}
141+
142+
// Determine if we are writing to an output file or printing to the console
143+
outputToFile := false
144+
var output *os.File
145+
146+
if *outputFile != "" {
147+
outputToFile = true
148+
output, err = os.Create(*outputFile)
149+
if err != nil {
150+
fmt.Printf("Error creating output file %s: %v\n", *outputFile, err)
151+
return
152+
}
153+
defer output.Close()
154+
}
155+
156+
processLogFile(*logFile, errorPatterns, notErrorPatterns, output, outputToFile)
157+
}
158+

‎not_errors.txt

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
warning: File not found:
2+
e,error:

0 commit comments

Comments
 (0)
Please sign in to comment.