|
|
|
@ -6,6 +6,7 @@ import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"os"
|
|
|
|
|
"slices"
|
|
|
|
|
|
|
|
|
|
"github.com/fatih/color"
|
|
|
|
|
|
|
|
|
@ -25,6 +26,8 @@ func main() {
|
|
|
|
|
multiLineFlag := flag.Bool("t", false, "Multi-line mode. Treats newline just like any character.")
|
|
|
|
|
printMatchesFlag := flag.Bool("p", false, "Prints start and end index of each match. Can only be used with '-t' for multi-line mode.")
|
|
|
|
|
caseInsensitiveFlag := flag.Bool("i", false, "Case-insensitive. Disregard the case of all characters.")
|
|
|
|
|
recursiveFlag := flag.Bool("r", false, "Recursively search all files in the given directory.")
|
|
|
|
|
lineNumFlag := flag.Bool("n", false, "For each line with a match, print the line number. Implies -l.")
|
|
|
|
|
matchNum := flag.Int("m", 0, "Print the match with the given index. Eg. -m 3 prints the third match.")
|
|
|
|
|
substituteText := flag.String("s", "", "Substitute the contents of each match with the given string. Overrides -o and -v")
|
|
|
|
|
flag.Parse()
|
|
|
|
@ -64,37 +67,49 @@ func main() {
|
|
|
|
|
// 2. Build NFA from postfix representation (Thompson's algorithm)
|
|
|
|
|
// 3. Run the string against the NFA
|
|
|
|
|
|
|
|
|
|
if len(flag.Args()) < 1 || len(flag.Args()) > 2 { // flag.Args() also strips out program name
|
|
|
|
|
if len(flag.Args()) < 1 { // flag.Args() also strips out program name
|
|
|
|
|
fmt.Println("ERROR: Missing cmdline args")
|
|
|
|
|
os.Exit(22)
|
|
|
|
|
}
|
|
|
|
|
if *recursiveFlag && len(flag.Args()) < 2 { // File/Directory must be provided with '-r'
|
|
|
|
|
fmt.Println("ERROR: Missing cmdline args")
|
|
|
|
|
os.Exit(22)
|
|
|
|
|
}
|
|
|
|
|
var re string
|
|
|
|
|
re = flag.Args()[0]
|
|
|
|
|
var inputFile *os.File
|
|
|
|
|
var inputFiles []*os.File
|
|
|
|
|
if len(flag.Args()) == 1 || flag.Args()[1] == "-" { // Either no file argument, or file argument is "-"
|
|
|
|
|
inputFile = os.Stdin
|
|
|
|
|
if !slices.Contains(inputFiles, os.Stdin) {
|
|
|
|
|
inputFiles = append(inputFiles, os.Stdin) // os.Stdin cannot be entered more than once into the file list
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
var err error
|
|
|
|
|
inputFile, err = os.Open(flag.Args()[1])
|
|
|
|
|
inputFilenames := flag.Args()[1:]
|
|
|
|
|
for _, inputFilename := range inputFilenames {
|
|
|
|
|
inputFile, err := os.Open(inputFilename)
|
|
|
|
|
if err != nil {
|
|
|
|
|
fmt.Printf("%s: No such file or directory\n", flag.Args()[1])
|
|
|
|
|
os.Exit(2)
|
|
|
|
|
}
|
|
|
|
|
inputFiles = append(inputFiles, inputFile)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var test_str string
|
|
|
|
|
var err error
|
|
|
|
|
var linesRead bool // Whether or not we have read the lines in the file
|
|
|
|
|
lineNum := 0 // Current line number
|
|
|
|
|
// Create reader for stdin and writer for stdout
|
|
|
|
|
reader := bufio.NewReader(inputFile)
|
|
|
|
|
// Create writer for stdout
|
|
|
|
|
out := bufio.NewWriter(os.Stdout)
|
|
|
|
|
|
|
|
|
|
// Compile regex
|
|
|
|
|
regComp, err := reg.Compile(re, flagsToCompile...)
|
|
|
|
|
if err != nil {
|
|
|
|
|
fmt.Println(err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, inputFile := range inputFiles {
|
|
|
|
|
reader := bufio.NewReader(inputFile)
|
|
|
|
|
linesRead = false
|
|
|
|
|
for true {
|
|
|
|
|
if linesRead {
|
|
|
|
|
break
|
|
|
|
@ -186,6 +201,8 @@ func main() {
|
|
|
|
|
if *lineFlag {
|
|
|
|
|
if !(*invertFlag) && len(matchIndices) == 0 || *invertFlag && len(matchIndices) > 0 {
|
|
|
|
|
continue
|
|
|
|
|
} else {
|
|
|
|
|
color.New(color.FgMagenta).Fprintf(out, "%s: ", inputFile.Name()) // Print filename
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -213,7 +230,7 @@ func main() {
|
|
|
|
|
} else {
|
|
|
|
|
for i, c := range test_str_runes {
|
|
|
|
|
if indicesToPrint.contains(i) {
|
|
|
|
|
color.New(color.FgRed).Fprintf(out, "%c", c)
|
|
|
|
|
color.New(color.FgRed, color.Bold).Fprintf(out, "%c", c)
|
|
|
|
|
// Newline after every match - only if -o is enabled and -v is disabled.
|
|
|
|
|
if *onlyFlag && !(*invertFlag) {
|
|
|
|
|
for matchIdxNum, idx := range matchIndices {
|
|
|
|
@ -243,3 +260,4 @@ func main() {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|