2024-03-24 16:05:00 +00:00
|
|
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
|
|
|
package git
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
2024-04-12 03:36:34 +00:00
|
|
|
"slices"
|
2024-03-24 16:05:00 +00:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"code.gitea.io/gitea/modules/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
type GrepResult struct {
|
|
|
|
Filename string
|
|
|
|
LineNumbers []int
|
|
|
|
LineCodes []string
|
|
|
|
}
|
|
|
|
|
|
|
|
type GrepOptions struct {
|
|
|
|
RefName string
|
2024-03-25 13:25:22 +00:00
|
|
|
MaxResultLimit int
|
2024-03-24 16:05:00 +00:00
|
|
|
ContextLineNumber int
|
|
|
|
IsFuzzy bool
|
2024-04-12 03:36:34 +00:00
|
|
|
MaxLineLength int // the maximum length of a line to parse, exceeding chars will be truncated
|
2024-05-03 09:13:48 +00:00
|
|
|
PathspecList []string
|
2024-03-24 16:05:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func GrepSearch(ctx context.Context, repo *Repository, search string, opts GrepOptions) ([]*GrepResult, error) {
|
|
|
|
stdoutReader, stdoutWriter, err := os.Pipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("unable to create os pipe to grep: %w", err)
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
_ = stdoutReader.Close()
|
|
|
|
_ = stdoutWriter.Close()
|
|
|
|
}()
|
|
|
|
|
|
|
|
/*
|
|
|
|
The output is like this ( "^@" means \x00):
|
|
|
|
|
|
|
|
HEAD:.air.toml
|
|
|
|
6^@bin = "gitea"
|
|
|
|
|
|
|
|
HEAD:.changelog.yml
|
|
|
|
2^@repo: go-gitea/gitea
|
|
|
|
*/
|
|
|
|
var results []*GrepResult
|
|
|
|
cmd := NewCommand(ctx, "grep", "--null", "--break", "--heading", "--fixed-strings", "--line-number", "--ignore-case", "--full-name")
|
|
|
|
cmd.AddOptionValues("--context", fmt.Sprint(opts.ContextLineNumber))
|
|
|
|
if opts.IsFuzzy {
|
|
|
|
words := strings.Fields(search)
|
|
|
|
for _, word := range words {
|
|
|
|
cmd.AddOptionValues("-e", strings.TrimLeft(word, "-"))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
cmd.AddOptionValues("-e", strings.TrimLeft(search, "-"))
|
|
|
|
}
|
|
|
|
cmd.AddDynamicArguments(util.IfZero(opts.RefName, "HEAD"))
|
2024-05-03 09:13:48 +00:00
|
|
|
cmd.AddDashesAndList(opts.PathspecList...)
|
2024-03-25 13:25:22 +00:00
|
|
|
opts.MaxResultLimit = util.IfZero(opts.MaxResultLimit, 50)
|
2024-03-24 16:05:00 +00:00
|
|
|
stderr := bytes.Buffer{}
|
|
|
|
err = cmd.Run(&RunOpts{
|
|
|
|
Dir: repo.Path,
|
|
|
|
Stdout: stdoutWriter,
|
|
|
|
Stderr: &stderr,
|
|
|
|
PipelineFunc: func(ctx context.Context, cancel context.CancelFunc) error {
|
|
|
|
_ = stdoutWriter.Close()
|
|
|
|
defer stdoutReader.Close()
|
|
|
|
|
|
|
|
isInBlock := false
|
2024-04-12 03:36:34 +00:00
|
|
|
rd := bufio.NewReaderSize(stdoutReader, util.IfZero(opts.MaxLineLength, 16*1024))
|
2024-03-24 16:05:00 +00:00
|
|
|
var res *GrepResult
|
2024-04-12 03:36:34 +00:00
|
|
|
for {
|
|
|
|
lineBytes, isPrefix, err := rd.ReadLine()
|
|
|
|
if isPrefix {
|
|
|
|
lineBytes = slices.Clone(lineBytes)
|
|
|
|
for isPrefix && err == nil {
|
|
|
|
_, isPrefix, err = rd.ReadLine()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(lineBytes) == 0 && err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
line := string(lineBytes) // the memory of lineBytes is mutable
|
2024-03-24 16:05:00 +00:00
|
|
|
if !isInBlock {
|
|
|
|
if _ /* ref */, filename, ok := strings.Cut(line, ":"); ok {
|
|
|
|
isInBlock = true
|
|
|
|
res = &GrepResult{Filename: filename}
|
|
|
|
results = append(results, res)
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if line == "" {
|
2024-03-25 13:25:22 +00:00
|
|
|
if len(results) >= opts.MaxResultLimit {
|
2024-03-24 16:05:00 +00:00
|
|
|
cancel()
|
|
|
|
break
|
|
|
|
}
|
|
|
|
isInBlock = false
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if line == "--" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if lineNum, lineCode, ok := strings.Cut(line, "\x00"); ok {
|
|
|
|
lineNumInt, _ := strconv.Atoi(lineNum)
|
|
|
|
res.LineNumbers = append(res.LineNumbers, lineNumInt)
|
|
|
|
res.LineCodes = append(res.LineCodes, lineCode)
|
|
|
|
}
|
|
|
|
}
|
2024-04-12 03:36:34 +00:00
|
|
|
return nil
|
2024-03-24 16:05:00 +00:00
|
|
|
},
|
|
|
|
})
|
2024-03-25 13:25:22 +00:00
|
|
|
// git grep exits by cancel (killed), usually it is caused by the limit of results
|
|
|
|
if IsErrorExitCode(err, -1) && stderr.Len() == 0 {
|
|
|
|
return results, nil
|
|
|
|
}
|
2024-03-24 16:05:00 +00:00
|
|
|
// git grep exits with 1 if no results are found
|
|
|
|
if IsErrorExitCode(err, 1) && stderr.Len() == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
if err != nil && !errors.Is(err, context.Canceled) {
|
|
|
|
return nil, fmt.Errorf("unable to run git grep: %w, stderr: %s", err, stderr.String())
|
|
|
|
}
|
|
|
|
return results, nil
|
|
|
|
}
|