// Copyright 2022 Harness Inc. All rights reserved. // Use of this source code is governed by the Polyform Free Trial License // that can be found in the LICENSE.md file for this repository. package service import ( "bufio" "bytes" "context" "fmt" "io" "github.com/harness/gitness/gitrpc/diff" "github.com/harness/gitness/gitrpc/internal/streamio" "github.com/harness/gitness/gitrpc/internal/types" "github.com/harness/gitness/gitrpc/rpc" ) type DiffService struct { rpc.UnimplementedDiffServiceServer adapter GitAdapter reposRoot string reposTempDir string } func NewDiffService(adapter GitAdapter, reposRoot string, reposTempDir string) (*DiffService, error) { return &DiffService{ adapter: adapter, reposRoot: reposRoot, reposTempDir: reposTempDir, }, nil } func (s DiffService) RawDiff(request *rpc.DiffRequest, stream rpc.DiffService_RawDiffServer) error { sw := streamio.NewWriter(func(p []byte) error { return stream.Send(&rpc.RawDiffResponse{Data: p}) }) return s.rawDiff(stream.Context(), request, sw) } func (s DiffService) rawDiff(ctx context.Context, request *rpc.DiffRequest, w io.Writer) error { err := validateDiffRequest(request) if err != nil { return err } base := request.GetBase() repoPath := getFullPathForRepo(s.reposRoot, base.GetRepoUid()) args := make([]string, 0, 4) args = append(args, "--full-index") if request.GetMergeBase() { args = append(args, "--merge-base") } return s.adapter.RawDiff(ctx, repoPath, request.GetBaseRef(), request.GetHeadRef(), w, args...) } func validateDiffRequest(in *rpc.DiffRequest) error { if in.GetBase() == nil { return types.ErrBaseCannotBeEmpty } if in.GetBaseRef() == "" { return types.ErrEmptyBaseRef } if in.GetHeadRef() == "" { return types.ErrEmptyHeadRef } return nil } func (s DiffService) DiffShortStat(ctx context.Context, r *rpc.DiffRequest) (*rpc.DiffShortStatResponse, error) { err := validateDiffRequest(r) if err != nil { return nil, fmt.Errorf("failed to validate request for short diff statistic, error: %v", err) } base := r.GetBase() repoPath := getFullPathForRepo(s.reposRoot, base.GetRepoUid()) stat, err := s.adapter.DiffShortStat(ctx, repoPath, r.GetBaseRef(), r.GetHeadRef(), r.GetMergeBase()) if err != nil { return nil, processGitErrorf(err, "failed to fetch short statistics "+ "between %s and %s", r.GetBaseRef(), r.GetHeadRef()) } return &rpc.DiffShortStatResponse{ Files: int32(stat.Files), Additions: int32(stat.Additions), Deletions: int32(stat.Deletions), }, nil } func (s DiffService) GetDiffHunkHeaders( ctx context.Context, r *rpc.GetDiffHunkHeadersRequest, ) (*rpc.GetDiffHunkHeadersResponse, error) { base := r.GetBase() repoPath := getFullPathForRepo(s.reposRoot, base.GetRepoUid()) hunkHeaders, err := s.adapter.GetDiffHunkHeaders(ctx, repoPath, r.TargetCommitSha, r.SourceCommitSha) if err != nil { return nil, processGitErrorf(err, "failed to get diff hunk headers between two commits") } return &rpc.GetDiffHunkHeadersResponse{ Files: mapDiffFileHunkHeaders(hunkHeaders), }, nil } func (s DiffService) DiffCut( ctx context.Context, r *rpc.DiffCutRequest, ) (*rpc.DiffCutResponse, error) { base := r.GetBase() repoPath := getFullPathForRepo(s.reposRoot, base.GetRepoUid()) mergeBase, _, err := s.adapter.GetMergeBase(ctx, repoPath, "", r.TargetBranch, r.SourceBranch) if err != nil { return nil, processGitErrorf(err, "failed to find merge base") } sourceCommits, err := s.adapter.ListCommitSHAs(ctx, repoPath, r.SourceBranch, 0, 1, types.CommitFilter{AfterRef: r.TargetBranch}) if err != nil || len(sourceCommits) == 0 { return nil, processGitErrorf(err, "failed to get list of source branch commits") } diffHunkHeader, linesHunk, err := s.adapter.DiffCut(ctx, repoPath, r.TargetCommitSha, r.SourceCommitSha, r.Path, types.DiffCutParams{ LineStart: int(r.LineStart), LineStartNew: r.LineStartNew, LineEnd: int(r.LineEnd), LineEndNew: r.LineEndNew, BeforeLines: 2, AfterLines: 2, LineLimit: 40, }) if err != nil { return nil, processGitErrorf(err, "failed to get diff hunk") } return &rpc.DiffCutResponse{ HunkHeader: mapHunkHeader(diffHunkHeader), LinesHeader: linesHunk.HunkHeader.String(), Lines: linesHunk.Lines, MergeBaseSha: mergeBase, LatestSourceSha: sourceCommits[0], }, nil } func (s DiffService) Diff(request *rpc.DiffRequest, stream rpc.DiffService_DiffServer) error { done := make(chan bool) defer close(done) pr, pw := io.Pipe() defer pr.Close() parser := diff.Parser{ Reader: bufio.NewReader(pr), } go func() { defer pw.Close() err := s.rawDiff(stream.Context(), request, pw) if err != nil { return } }() return parser.Parse(func(f *diff.File) { streamDiffFile(f, request.IncludePatch, stream) }) } func streamDiffFile(f *diff.File, includePatch bool, stream rpc.DiffService_DiffServer) { var status rpc.DiffResponse_FileStatus switch f.Type { case diff.FileAdd: status = rpc.DiffResponse_ADDED case diff.FileChange: status = rpc.DiffResponse_MODIFIED case diff.FileDelete: status = rpc.DiffResponse_DELETED case diff.FileRename: status = rpc.DiffResponse_RENAMED default: status = rpc.DiffResponse_UNDEFINED } patch := bytes.Buffer{} if includePatch { for _, sec := range f.Sections { for _, line := range sec.Lines { if line.Type != diff.DiffLinePlain { patch.WriteString(line.Content) } } } } stream.Send(&rpc.DiffResponse{ Path: f.Path, OldPath: f.OldPath, Sha: f.SHA, OldSha: f.OldSHA, Status: status, Additions: int32(f.NumAdditions()), Deletions: int32(f.NumDeletions()), Changes: int32(f.NumChanges()), Patch: patch.Bytes(), }) }