multiLineFlag:=flag.Bool("t",false,"Multi-line mode. Treats newline just like any character.")
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.")
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.")
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.")
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")
substituteText:=flag.String("s","","Substitute the contents of each match with the given string. Overrides -o and -v")
flag.Parse()
flag.Parse()
@ -64,21 +66,30 @@ func main() {
// 2. Build NFA from postfix representation (Thompson's algorithm)
// 2. Build NFA from postfix representation (Thompson's algorithm)
// 3. Run the string against the NFA
// 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")
fmt.Println("ERROR: Missing cmdline args")
os.Exit(22)
os.Exit(22)
}
}
varrestring
varrestring
re=flag.Args()[0]
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 "-"
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{
}else{
varerrerror
inputFilenames:=flag.Args()[1:]
inputFile,err=os.Open(flag.Args()[1])
for_,inputFilename:=rangeinputFilenames{
iferr!=nil{
inputFile,err:=os.Open(inputFilename)
fmt.Printf("%s: No such file or directory\n",flag.Args()[1])
iferr!=nil{
os.Exit(2)
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
varerrerror
varlinesReadbool// Whether or not we have read the lines in the file
varlinesReadbool// Whether or not we have read the lines in the file
lineNum:=0// Current line number
lineNum:=0// Current line number
// Create reader for stdin and writer for stdout
// Create writer for stdout
reader:=bufio.NewReader(inputFile)
out:=bufio.NewWriter(os.Stdout)
out:=bufio.NewWriter(os.Stdout)
// Compile regex
regComp,err:=reg.Compile(re,flagsToCompile...)
regComp,err:=reg.Compile(re,flagsToCompile...)
iferr!=nil{
iferr!=nil{
fmt.Println(err)
fmt.Println(err)
return
return
}
}
fortrue{
iflinesRead{
for_,inputFile:=rangeinputFiles{
break
reader:=bufio.NewReader(inputFile)
}
linesRead=false
if!(*multiLineFlag){
fortrue{
// Read every string from stdin until we encounter an error. If the error isn't EOF, panic.
if!(*lineFlag)&&(!*onlyFlag){
test_str,err=reader.ReadString('\n')
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
lineNum++
}
iferr!=nil{
iflinesRead{
break
}
if!(*multiLineFlag){
// Read every string from stdin until we encounter an error. If the error isn't EOF, panic.
// Newline after every match - only if -o is enabled and -v is disabled.
fori,c:=rangetest_str_runes{
if*onlyFlag&&!(*invertFlag){
ifindicesToPrint.contains(i){
formatchIdxNum,idx:=rangematchIndices{
color.New(color.FgRed).Fprintf(out,"%c",c)
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.
// Newline after every match - only if -o is enabled and -v is disabled.
ifi+1==idx[0].EndIdx{// End index is one more than last index of match
if*onlyFlag&&!(*invertFlag){
fmt.Fprintf(out,"\n")
formatchIdxNum,idx:=rangematchIndices{
break
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{
}else{
if!(*onlyFlag){
if!(*onlyFlag){
fmt.Fprintf(out,"%c",c)
fmt.Fprintf(out,"%c",c)
}
}
}
}
}
}
}
}
err=out.Flush()
err=out.Flush()
iferr!=nil{
iferr!=nil{
panic(err)
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
// 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