mirror of
https://github.com/harness/drone.git
synced 2025-04-28 05:36:58 +00:00
152 lines
4.6 KiB
Go
152 lines
4.6 KiB
Go
// 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 gitea
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/harness/gitness/gitrpc/internal/types"
|
|
|
|
gitea "code.gitea.io/gitea/modules/git"
|
|
"github.com/rs/zerolog/log"
|
|
)
|
|
|
|
// Logs the error and message, returns either the provided message or a git equivalent if possible.
|
|
// Always logs the full message with error as warning.
|
|
func processGiteaErrorf(err error, format string, args ...interface{}) error {
|
|
// create fallback error returned if we can't map it
|
|
fallbackErr := fmt.Errorf(format, args...)
|
|
|
|
// always log internal error together with message.
|
|
log.Warn().Msgf("%v: [GITEA] %v", fallbackErr, err)
|
|
|
|
// check if it's a RunStdError error (contains raw git error)
|
|
var runStdErr gitea.RunStdError
|
|
if errors.As(err, &runStdErr) {
|
|
return mapGiteaRunStdError(runStdErr, fallbackErr)
|
|
}
|
|
|
|
switch {
|
|
case gitea.IsErrNotExist(err):
|
|
return types.ErrNotFound
|
|
default:
|
|
return fallbackErr
|
|
}
|
|
}
|
|
|
|
// TODO: Improve gitea error handling.
|
|
// Doubt this will work for all std errors, as git doesn't seem to have nice error codes.
|
|
func mapGiteaRunStdError(err gitea.RunStdError, fallback error) error {
|
|
switch {
|
|
// exit status 128 - fatal: A branch named 'mybranch' already exists.
|
|
// exit status 128 - fatal: cannot lock ref 'refs/heads/a': 'refs/heads/a/b' exists; cannot create 'refs/heads/a'
|
|
case err.IsExitCode(128) && strings.Contains(err.Stderr(), "exists"):
|
|
return types.ErrAlreadyExists
|
|
|
|
// exit status 128 - fatal: 'a/bc/d/' is not a valid branch name.
|
|
case err.IsExitCode(128) && strings.Contains(err.Stderr(), "not a valid"):
|
|
return types.ErrInvalidArgument
|
|
|
|
// exit status 1 - error: branch 'mybranch' not found.
|
|
case err.IsExitCode(1) && strings.Contains(err.Stderr(), "not found"):
|
|
return types.ErrNotFound
|
|
|
|
// exit status 128 - fatal: ambiguous argument 'branch1...branch2': unknown revision or path not in the working tree.
|
|
case err.IsExitCode(128) && strings.Contains(err.Stderr(), "unknown revision"):
|
|
return types.ErrNotFound
|
|
|
|
default:
|
|
return fallback
|
|
}
|
|
}
|
|
|
|
func mapGiteaRawRef(raw map[string]string) (map[types.GitReferenceField]string, error) {
|
|
res := make(map[types.GitReferenceField]string, len(raw))
|
|
for k, v := range raw {
|
|
gitRefField, err := types.ParseGitReferenceField(k)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
res[gitRefField] = v
|
|
}
|
|
|
|
return res, nil
|
|
}
|
|
|
|
func mapToGiteaReferenceSortingArgument(s types.GitReferenceField, o types.SortOrder) string {
|
|
sortBy := string(types.GitReferenceFieldRefName)
|
|
desc := o == types.SortOrderDesc
|
|
|
|
if s == types.GitReferenceFieldCreatorDate {
|
|
sortBy = string(types.GitReferenceFieldCreatorDate)
|
|
if o == types.SortOrderDefault {
|
|
desc = true
|
|
}
|
|
}
|
|
|
|
if desc {
|
|
return "-" + sortBy
|
|
}
|
|
|
|
return sortBy
|
|
}
|
|
|
|
func mapGiteaCommit(giteaCommit *gitea.Commit) (*types.Commit, error) {
|
|
if giteaCommit == nil {
|
|
return nil, fmt.Errorf("gitea commit is nil")
|
|
}
|
|
|
|
author, err := mapGiteaSignature(giteaCommit.Author)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to map gitea author: %w", err)
|
|
}
|
|
committer, err := mapGiteaSignature(giteaCommit.Committer)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to map gitea commiter: %w", err)
|
|
}
|
|
return &types.Commit{
|
|
SHA: giteaCommit.ID.String(),
|
|
Title: giteaCommit.Summary(),
|
|
// remove potential tailing newlines from message
|
|
Message: strings.TrimRight(giteaCommit.Message(), "\n"),
|
|
Author: author,
|
|
Committer: committer,
|
|
}, nil
|
|
}
|
|
|
|
func mapGiteaNodeToTreeNodeModeAndType(giteaMode gitea.EntryMode) (types.TreeNodeType, types.TreeNodeMode, error) {
|
|
switch giteaMode {
|
|
case gitea.EntryModeBlob:
|
|
return types.TreeNodeTypeBlob, types.TreeNodeModeFile, nil
|
|
case gitea.EntryModeSymlink:
|
|
return types.TreeNodeTypeBlob, types.TreeNodeModeSymlink, nil
|
|
case gitea.EntryModeExec:
|
|
return types.TreeNodeTypeBlob, types.TreeNodeModeExec, nil
|
|
case gitea.EntryModeCommit:
|
|
return types.TreeNodeTypeCommit, types.TreeNodeModeCommit, nil
|
|
case gitea.EntryModeTree:
|
|
return types.TreeNodeTypeTree, types.TreeNodeModeTree, nil
|
|
default:
|
|
return types.TreeNodeTypeBlob, types.TreeNodeModeFile,
|
|
fmt.Errorf("received unknown tree node mode from gitea: '%s'", giteaMode.String())
|
|
}
|
|
}
|
|
|
|
func mapGiteaSignature(giteaSignature *gitea.Signature) (types.Signature, error) {
|
|
if giteaSignature == nil {
|
|
return types.Signature{}, fmt.Errorf("gitea signature is nil")
|
|
}
|
|
|
|
return types.Signature{
|
|
Identity: types.Identity{
|
|
Name: giteaSignature.Name,
|
|
Email: giteaSignature.Email,
|
|
},
|
|
When: giteaSignature.When,
|
|
}, nil
|
|
}
|