// 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 }