Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. // Copyright 2021 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. // Package private includes all internal routes. The package name internal is ideal but Golang is not allowed, so we use private as package name instead.
  5. package private
  6. import (
  7. "bufio"
  8. "context"
  9. "fmt"
  10. "io"
  11. "os"
  12. asymkey_model "code.gitea.io/gitea/models/asymkey"
  13. "code.gitea.io/gitea/modules/git"
  14. "code.gitea.io/gitea/modules/log"
  15. )
  16. // _________ .__ __
  17. // \_ ___ \ ____ _____ _____ |__|/ |_
  18. // / \ \/ / _ \ / \ / \| \ __\
  19. // \ \___( <_> ) Y Y \ Y Y \ || |
  20. // \______ /\____/|__|_| /__|_| /__||__|
  21. // \/ \/ \/
  22. // ____ ____ .__ _____.__ __ .__
  23. // \ \ / /___________|__|/ ____\__| ____ _____ _/ |_|__| ____ ____
  24. // \ Y // __ \_ __ \ \ __\| |/ ___\\__ \\ __\ |/ _ \ / \
  25. // \ /\ ___/| | \/ || | | \ \___ / __ \| | | ( <_> ) | \
  26. // \___/ \___ >__| |__||__| |__|\___ >____ /__| |__|\____/|___| /
  27. // \/ \/ \/ \/
  28. //
  29. // This file contains commit verification functions for refs passed across in hooks
  30. func verifyCommits(oldCommitID, newCommitID string, repo *git.Repository, env []string) error {
  31. stdoutReader, stdoutWriter, err := os.Pipe()
  32. if err != nil {
  33. log.Error("Unable to create os.Pipe for %s", repo.Path)
  34. return err
  35. }
  36. defer func() {
  37. _ = stdoutReader.Close()
  38. _ = stdoutWriter.Close()
  39. }()
  40. // This is safe as force pushes are already forbidden
  41. err = git.NewCommand("rev-list", oldCommitID+"..."+newCommitID).
  42. RunInDirTimeoutEnvFullPipelineFunc(env, -1, repo.Path,
  43. stdoutWriter, nil, nil,
  44. func(ctx context.Context, cancel context.CancelFunc) error {
  45. _ = stdoutWriter.Close()
  46. err := readAndVerifyCommitsFromShaReader(stdoutReader, repo, env)
  47. if err != nil {
  48. log.Error("%v", err)
  49. cancel()
  50. }
  51. _ = stdoutReader.Close()
  52. return err
  53. })
  54. if err != nil && !isErrUnverifiedCommit(err) {
  55. log.Error("Unable to check commits from %s to %s in %s: %v", oldCommitID, newCommitID, repo.Path, err)
  56. }
  57. return err
  58. }
  59. func readAndVerifyCommitsFromShaReader(input io.ReadCloser, repo *git.Repository, env []string) error {
  60. scanner := bufio.NewScanner(input)
  61. for scanner.Scan() {
  62. line := scanner.Text()
  63. err := readAndVerifyCommit(line, repo, env)
  64. if err != nil {
  65. log.Error("%v", err)
  66. return err
  67. }
  68. }
  69. return scanner.Err()
  70. }
  71. func readAndVerifyCommit(sha string, repo *git.Repository, env []string) error {
  72. stdoutReader, stdoutWriter, err := os.Pipe()
  73. if err != nil {
  74. log.Error("Unable to create pipe for %s: %v", repo.Path, err)
  75. return err
  76. }
  77. defer func() {
  78. _ = stdoutReader.Close()
  79. _ = stdoutWriter.Close()
  80. }()
  81. hash := git.MustIDFromString(sha)
  82. return git.NewCommand("cat-file", "commit", sha).
  83. RunInDirTimeoutEnvFullPipelineFunc(env, -1, repo.Path,
  84. stdoutWriter, nil, nil,
  85. func(ctx context.Context, cancel context.CancelFunc) error {
  86. _ = stdoutWriter.Close()
  87. commit, err := git.CommitFromReader(repo, hash, stdoutReader)
  88. if err != nil {
  89. return err
  90. }
  91. verification := asymkey_model.ParseCommitWithSignature(commit)
  92. if !verification.Verified {
  93. cancel()
  94. return &errUnverifiedCommit{
  95. commit.ID.String(),
  96. }
  97. }
  98. return nil
  99. })
  100. }
  101. type errUnverifiedCommit struct {
  102. sha string
  103. }
  104. func (e *errUnverifiedCommit) Error() string {
  105. return fmt.Sprintf("Unverified commit: %s", e.sha)
  106. }
  107. func isErrUnverifiedCommit(err error) bool {
  108. _, ok := err.(*errUnverifiedCommit)
  109. return ok
  110. }