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.")
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,21 +66,30 @@ func main() {
// 2. Build NFA from postfix representation (Thompson's algorithm)
// 3. Run the string against the NFA
iflen(flag.Args())<1||len(flag.Args())>2{// flag.Args() also strips out program name
iflen(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)
}
varrestring
re=flag.Args()[0]
varinputFile*os.File
varinputFiles[]*os.File
iflen(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{
varerrerror
inputFile,err=os.Open(flag.Args()[1])
iferr!=nil{
fmt.Printf("%s: No such file or directory\n",flag.Args()[1])
os.Exit(2)
inputFilenames:=flag.Args()[1:]
for_,inputFilename:=rangeinputFilenames{
inputFile,err:=os.Open(inputFilename)
iferr!=nil{
fmt.Printf("%s: No such file or directory\n",flag.Args()[1])
os.Exit(2)
}
inputFiles=append(inputFiles,inputFile)
}
}
@ -86,160 +97,169 @@ func main() {
varerrerror
varlinesReadbool// 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...)
iferr!=nil{
fmt.Println(err)
return
}
fortrue{
iflinesRead{
break
}
if!(*multiLineFlag){
// Read every string from stdin until we encounter an error. If the error isn't EOF, panic.
test_str,err=reader.ReadString('\n')
lineNum++
iferr!=nil{
for_,inputFile:=rangeinputFiles{
reader:=bufio.NewReader(inputFile)
linesRead=false
fortrue{
if!(*lineFlag)&&(!*onlyFlag){
color.New(color.FgMagenta).Fprintf(out,"%s: ",inputFile.Name())// The filename should be printed for every line, _except_ if we're not printing every line. This is the case with the lineFlag and onlyFlag
}
iflinesRead{
break
}
if!(*multiLineFlag){
// Read every string from stdin until we encounter an error. If the error isn't EOF, panic.
// If we are substituting, we need a different behavior, as follows:
// For every character in the test string:
// 1. Check if the index is the start of any matchIndex
// 2. If so, print the substitute text, and set our index to
// the corresponding end index.
// 3. If not, just print the character.
ifsubstituteFlagEnabled{
fori:=rangetest_str_runes{
inMatchIndex:=false
for_,m:=rangematchIndices{
ifi==m[0].StartIdx{
fmt.Fprintf(out,"%s",*substituteText)
i=m[0].EndIdx
inMatchIndex=true
break
// If we are substituting, we need a different behavior, as follows:
// For every character in the test string:
// 1. Check if the index is the start of any matchIndex
// 2. If so, print the substitute text, and set our index to
// the corresponding end index.
// 3. If not, just print the character.
ifsubstituteFlagEnabled{
fori:=rangetest_str_runes{
inMatchIndex:=false
for_,m:=rangematchIndices{
ifi==m[0].StartIdx{
fmt.Fprintf(out,"%s",*substituteText)
i=m[0].EndIdx
inMatchIndex=true
break
}
}
if!inMatchIndex{
fmt.Fprintf(out,"%c",test_str_runes[i])
}
}
if!inMatchIndex{
fmt.Fprintf(out,"%c",test_str_runes[i])
}
}
}else{
fori,c:=rangetest_str_runes{
ifindicesToPrint.contains(i){
color.New(color.FgRed).Fprintf(out,"%c",c)
// Newline after every match - only if -o is enabled and -v is disabled.
if*onlyFlag&&!(*invertFlag){
formatchIdxNum,idx:=rangematchIndices{
ifmatchIdxNum<len(matchIndices)-1{// Only print a newline afte printing a match, if there are multiple matches on the line, and we aren't on the last one. This is because the newline that gets added at the end will take care of that.
ifi+1==idx[0].EndIdx{// End index is one more than last index of match
// Newline after every match - only if -o is enabled and -v is disabled.
if*onlyFlag&&!(*invertFlag){
formatchIdxNum,idx:=rangematchIndices{
ifmatchIdxNum<len(matchIndices)-1{// Only print a newline afte printing a match, if there are multiple matches on the line, and we aren't on the last one. This is because the newline that gets added at the end will take care of that.
ifi+1==idx[0].EndIdx{// End index is one more than last index of match
fmt.Fprintf(out,"\n")
break
}
}
}
}
}
}else{
if!(*onlyFlag){
fmt.Fprintf(out,"%c",c)
}else{
if!(*onlyFlag){
fmt.Fprintf(out,"%c",c)
}
}
}
}
}
err=out.Flush()
iferr!=nil{
panic(err)
}
// If the last character in the string wasn't a newline, AND we either have don't -o set or we do (and we've matched something), then print a newline