mirror of https://github.com/go-gitea/gitea.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
215 lines
6.9 KiB
215 lines
6.9 KiB
// Copyright 2019 The Gitea Authors. All rights reserved. |
|
// SPDX-License-Identifier: MIT |
|
|
|
package integration |
|
|
|
import ( |
|
"context" |
|
"fmt" |
|
"net" |
|
"net/http" |
|
"net/url" |
|
"os" |
|
"path/filepath" |
|
"strconv" |
|
"testing" |
|
"time" |
|
|
|
"code.gitea.io/gitea/modules/git" |
|
"code.gitea.io/gitea/modules/git/gitcmd" |
|
"code.gitea.io/gitea/modules/setting" |
|
"code.gitea.io/gitea/modules/ssh" |
|
"code.gitea.io/gitea/modules/util" |
|
"code.gitea.io/gitea/tests" |
|
|
|
"github.com/stretchr/testify/assert" |
|
"github.com/stretchr/testify/require" |
|
) |
|
|
|
func withKeyFile(t *testing.T, keyname string, callback func(string)) { |
|
tmpDir := t.TempDir() |
|
|
|
err := os.Chmod(tmpDir, 0o700) |
|
assert.NoError(t, err) |
|
|
|
keyFile := filepath.Join(tmpDir, keyname) |
|
err = ssh.GenKeyPair(keyFile) |
|
assert.NoError(t, err) |
|
|
|
err = os.WriteFile(filepath.Join(tmpDir, "ssh"), []byte("#!/bin/bash\n"+ |
|
"ssh -o \"UserKnownHostsFile=/dev/null\" -o \"StrictHostKeyChecking=no\" -o \"IdentitiesOnly=yes\" -i \""+keyFile+"\" \"$@\""), 0o700) |
|
assert.NoError(t, err) |
|
|
|
// Setup ssh wrapper |
|
t.Setenv("GIT_SSH", filepath.Join(tmpDir, "ssh")) |
|
t.Setenv("GIT_SSH_COMMAND", |
|
"ssh -o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no -o IdentitiesOnly=yes -i \""+keyFile+"\"") |
|
t.Setenv("GIT_SSH_VARIANT", "ssh") |
|
|
|
callback(keyFile) |
|
} |
|
|
|
func createSSHUrl(gitPath string, u *url.URL) *url.URL { |
|
u2 := *u |
|
u2.Scheme = "ssh" |
|
u2.User = url.User("git") |
|
u2.Host = net.JoinHostPort(setting.SSH.ListenHost, strconv.Itoa(setting.SSH.ListenPort)) |
|
u2.Path = gitPath |
|
return &u2 |
|
} |
|
|
|
func onGiteaRun[T testing.TB](t T, callback func(T, *url.URL)) { |
|
defer tests.PrepareTestEnv(t, 1)() |
|
s := http.Server{ |
|
Handler: testWebRoutes, |
|
} |
|
|
|
u, err := url.Parse(setting.AppURL) |
|
assert.NoError(t, err) |
|
listener, err := net.Listen("tcp", u.Host) |
|
i := 0 |
|
for err != nil && i <= 10 { |
|
time.Sleep(100 * time.Millisecond) |
|
listener, err = net.Listen("tcp", u.Host) |
|
i++ |
|
} |
|
assert.NoError(t, err) |
|
u.Host = listener.Addr().String() |
|
|
|
defer func() { |
|
ctx, cancel := context.WithTimeout(t.Context(), 2*time.Minute) |
|
s.Shutdown(ctx) |
|
cancel() |
|
}() |
|
|
|
go s.Serve(listener) |
|
// Started by config go ssh.Listen(setting.SSH.ListenHost, setting.SSH.ListenPort, setting.SSH.ServerCiphers, setting.SSH.ServerKeyExchanges, setting.SSH.ServerMACs) |
|
|
|
callback(t, u) |
|
} |
|
|
|
func doGitClone(dstLocalPath string, u *url.URL) func(*testing.T) { |
|
return func(t *testing.T) { |
|
assert.NoError(t, git.Clone(t.Context(), u.String(), dstLocalPath, git.CloneRepoOptions{})) |
|
exist, err := util.IsExist(filepath.Join(dstLocalPath, "README.md")) |
|
assert.NoError(t, err) |
|
assert.True(t, exist) |
|
} |
|
} |
|
|
|
func doPartialGitClone(dstLocalPath string, u *url.URL) func(*testing.T) { |
|
return func(t *testing.T) { |
|
assert.NoError(t, git.Clone(t.Context(), u.String(), dstLocalPath, git.CloneRepoOptions{ |
|
Filter: "blob:none", |
|
})) |
|
exist, err := util.IsExist(filepath.Join(dstLocalPath, "README.md")) |
|
assert.NoError(t, err) |
|
assert.True(t, exist) |
|
} |
|
} |
|
|
|
func doGitCloneFail(u *url.URL) func(*testing.T) { |
|
return func(t *testing.T) { |
|
tmpDir := t.TempDir() |
|
assert.Error(t, git.Clone(t.Context(), u.String(), tmpDir, git.CloneRepoOptions{})) |
|
exist, err := util.IsExist(filepath.Join(tmpDir, "README.md")) |
|
assert.NoError(t, err) |
|
assert.False(t, exist) |
|
} |
|
} |
|
|
|
func doGitInitTestRepository(dstPath string) func(*testing.T) { |
|
return func(t *testing.T) { |
|
// Init repository in dstPath |
|
assert.NoError(t, git.InitRepository(t.Context(), dstPath, false, git.Sha1ObjectFormat.Name())) |
|
// forcibly set default branch to master |
|
_, _, err := gitcmd.NewCommand("symbolic-ref", "HEAD", git.BranchPrefix+"master").RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.NoError(t, err) |
|
assert.NoError(t, os.WriteFile(filepath.Join(dstPath, "README.md"), []byte("# Testing Repository\n\nOriginally created in: "+dstPath), 0o644)) |
|
assert.NoError(t, git.AddChanges(t.Context(), dstPath, true)) |
|
signature := git.Signature{ |
|
Email: "test@example.com", |
|
Name: "test", |
|
When: time.Now(), |
|
} |
|
assert.NoError(t, git.CommitChanges(t.Context(), dstPath, git.CommitChangesOptions{ |
|
Committer: &signature, |
|
Author: &signature, |
|
Message: "Initial Commit", |
|
})) |
|
} |
|
} |
|
|
|
func doGitAddRemote(dstPath, remoteName string, u *url.URL) func(*testing.T) { |
|
return func(t *testing.T) { |
|
_, _, err := gitcmd.NewCommand("remote", "add").AddDynamicArguments(remoteName, u.String()).RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.NoError(t, err) |
|
} |
|
} |
|
|
|
func doGitPushTestRepository(dstPath string, args ...string) func(*testing.T) { |
|
return func(t *testing.T) { |
|
_, _, err := gitcmd.NewCommand("push", "-u").AddArguments(gitcmd.ToTrustedCmdArgs(args)...).RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.NoError(t, err) |
|
} |
|
} |
|
|
|
func doGitPushTestRepositoryFail(dstPath string, args ...string) func(*testing.T) { |
|
return func(t *testing.T) { |
|
_, _, err := gitcmd.NewCommand("push").AddArguments(gitcmd.ToTrustedCmdArgs(args)...).RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.Error(t, err) |
|
} |
|
} |
|
|
|
type localGitAddCommitOptions struct { |
|
LocalRepoPath string |
|
CheckoutBranch string |
|
TreeFilePath string |
|
TreeFileContent string |
|
} |
|
|
|
func doGitCheckoutWriteFileCommit(opts localGitAddCommitOptions) func(*testing.T) { |
|
return func(t *testing.T) { |
|
doGitCheckoutBranch(opts.LocalRepoPath, opts.CheckoutBranch)(t) |
|
localFilePath := filepath.Join(opts.LocalRepoPath, opts.TreeFilePath) |
|
require.NoError(t, os.WriteFile(localFilePath, []byte(opts.TreeFileContent), 0o644)) |
|
require.NoError(t, git.AddChanges(t.Context(), opts.LocalRepoPath, true)) |
|
signature := git.Signature{ |
|
Email: "test@test.test", |
|
Name: "test", |
|
} |
|
require.NoError(t, git.CommitChanges(t.Context(), opts.LocalRepoPath, git.CommitChangesOptions{ |
|
Committer: &signature, |
|
Author: &signature, |
|
Message: fmt.Sprintf("update %s @ %s", opts.TreeFilePath, opts.CheckoutBranch), |
|
})) |
|
} |
|
} |
|
|
|
func doGitCreateBranch(dstPath, branch string) func(*testing.T) { |
|
return func(t *testing.T) { |
|
_, _, err := gitcmd.NewCommand("checkout", "-b").AddDynamicArguments(branch).RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.NoError(t, err) |
|
} |
|
} |
|
|
|
func doGitCheckoutBranch(dstPath string, args ...string) func(*testing.T) { |
|
return func(t *testing.T) { |
|
_, _, err := gitcmd.NewCommand().AddArguments("checkout").AddArguments(gitcmd.ToTrustedCmdArgs(args)...).RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.NoError(t, err) |
|
} |
|
} |
|
|
|
func doGitMerge(dstPath string, args ...string) func(*testing.T) { |
|
return func(t *testing.T) { |
|
_, _, err := gitcmd.NewCommand("merge").AddArguments(gitcmd.ToTrustedCmdArgs(args)...).RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.NoError(t, err) |
|
} |
|
} |
|
|
|
func doGitPull(dstPath string, args ...string) func(*testing.T) { |
|
return func(t *testing.T) { |
|
_, _, err := gitcmd.NewCommand().AddArguments("pull").AddArguments(gitcmd.ToTrustedCmdArgs(args)...).RunStdString(t.Context(), &gitcmd.RunOpts{Dir: dstPath}) |
|
assert.NoError(t, err) |
|
} |
|
}
|
|
|