mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-26 12:01:08 +00:00 
			
		
		
		
	If a repository has git config --add push.pushOption submit=".sourcehut/*.yml" it failed when pushed because of the unknown submit push option. It will be ignored instead. Filtering out the push options is done in an earlier stage, when the hook command runs, before it submits the options map to the private endpoint. * move all the push options logic to modules/git/pushoptions * add 100% test coverage for modules/git/pushoptions Test coverage for the code paths from which code was moved to the modules/git/pushoptions package: * cmd/hook.go:runHookPreReceive * routers/private/hook_pre_receive.go:validatePushOptions tests/integration/git_push_test.go:TestOptionsGitPush runs through both. The test verifying the option is rejected was removed and, if added again, will fail because the option is now ignored instead of being rejected. * cmd/hook.go:runHookProcReceive * services/agit/agit.go:ProcReceive tests/integration/git_test.go: doCreateAgitFlowPull runs through both. It uses variations of AGit related push options. * cmd/hook.go:runHookPostReceive * routers/private/hook_post_receive.go:HookPostReceive tests/integration/git_test.go:doPushCreate called by TestGit/HTTP/sha1/PushCreate runs through both. Note that although it provides coverage for this code path it does not use push options. Fixes: https://codeberg.org/forgejo/forgejo/issues/3651
		
			
				
	
	
		
			196 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			196 lines
		
	
	
	
		
			5.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2021 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package cmd
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"bytes"
 | |
| 	"context"
 | |
| 	"io"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"os"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/modules/test"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| 	"github.com/stretchr/testify/require"
 | |
| 	"github.com/urfave/cli/v2"
 | |
| )
 | |
| 
 | |
| // Capture what's being written into a standard file descriptor.
 | |
| func captureOutput(t *testing.T, stdFD *os.File) (finish func() (output string)) {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	r, w, err := os.Pipe()
 | |
| 	require.NoError(t, err)
 | |
| 	resetStdout := test.MockVariableValue(stdFD, *w)
 | |
| 
 | |
| 	return func() (output string) {
 | |
| 		w.Close()
 | |
| 		resetStdout()
 | |
| 
 | |
| 		out, err := io.ReadAll(r)
 | |
| 		require.NoError(t, err)
 | |
| 		return string(out)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestPktLine(t *testing.T) {
 | |
| 	ctx := context.Background()
 | |
| 
 | |
| 	t.Run("Read", func(t *testing.T) {
 | |
| 		s := strings.NewReader("0000")
 | |
| 		r := bufio.NewReader(s)
 | |
| 		result, err := readPktLine(ctx, r, pktLineTypeFlush)
 | |
| 		assert.NoError(t, err)
 | |
| 		assert.Equal(t, pktLineTypeFlush, result.Type)
 | |
| 
 | |
| 		s = strings.NewReader("0006a\n")
 | |
| 		r = bufio.NewReader(s)
 | |
| 		result, err = readPktLine(ctx, r, pktLineTypeData)
 | |
| 		assert.NoError(t, err)
 | |
| 		assert.Equal(t, pktLineTypeData, result.Type)
 | |
| 		assert.Equal(t, []byte("a\n"), result.Data)
 | |
| 
 | |
| 		s = strings.NewReader("0004")
 | |
| 		r = bufio.NewReader(s)
 | |
| 		result, err = readPktLine(ctx, r, pktLineTypeData)
 | |
| 		assert.Error(t, err)
 | |
| 		assert.Nil(t, result)
 | |
| 
 | |
| 		data := strings.Repeat("x", 65516)
 | |
| 		r = bufio.NewReader(strings.NewReader("fff0" + data))
 | |
| 		result, err = readPktLine(ctx, r, pktLineTypeData)
 | |
| 		assert.NoError(t, err)
 | |
| 		assert.Equal(t, pktLineTypeData, result.Type)
 | |
| 		assert.Equal(t, []byte(data), result.Data)
 | |
| 
 | |
| 		r = bufio.NewReader(strings.NewReader("fff1a"))
 | |
| 		result, err = readPktLine(ctx, r, pktLineTypeData)
 | |
| 		assert.Error(t, err)
 | |
| 		assert.Nil(t, result)
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Write", func(t *testing.T) {
 | |
| 		w := bytes.NewBuffer([]byte{})
 | |
| 		err := writeFlushPktLine(ctx, w)
 | |
| 		assert.NoError(t, err)
 | |
| 		assert.Equal(t, []byte("0000"), w.Bytes())
 | |
| 
 | |
| 		w.Reset()
 | |
| 		err = writeDataPktLine(ctx, w, []byte("a\nb"))
 | |
| 		assert.NoError(t, err)
 | |
| 		assert.Equal(t, []byte("0007a\nb"), w.Bytes())
 | |
| 
 | |
| 		w.Reset()
 | |
| 		data := bytes.Repeat([]byte{0x05}, 288)
 | |
| 		err = writeDataPktLine(ctx, w, data)
 | |
| 		assert.NoError(t, err)
 | |
| 		assert.Equal(t, append([]byte("0124"), data...), w.Bytes())
 | |
| 
 | |
| 		w.Reset()
 | |
| 		err = writeDataPktLine(ctx, w, nil)
 | |
| 		assert.Error(t, err)
 | |
| 		assert.Empty(t, w.Bytes())
 | |
| 
 | |
| 		w.Reset()
 | |
| 		data = bytes.Repeat([]byte{0x64}, 65516)
 | |
| 		err = writeDataPktLine(ctx, w, data)
 | |
| 		assert.NoError(t, err)
 | |
| 		assert.Equal(t, append([]byte("fff0"), data...), w.Bytes())
 | |
| 
 | |
| 		w.Reset()
 | |
| 		err = writeDataPktLine(ctx, w, bytes.Repeat([]byte{0x64}, 65516+1))
 | |
| 		assert.Error(t, err)
 | |
| 		assert.Empty(t, w.Bytes())
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestDelayWriter(t *testing.T) {
 | |
| 	// Setup the environment.
 | |
| 	defer test.MockVariableValue(&setting.InternalToken, "Random")()
 | |
| 	defer test.MockVariableValue(&setting.InstallLock, true)()
 | |
| 	defer test.MockVariableValue(&setting.Git.VerbosePush, true)()
 | |
| 	require.NoError(t, os.Setenv("SSH_ORIGINAL_COMMAND", "true"))
 | |
| 
 | |
| 	// Setup the Stdin.
 | |
| 	f, err := os.OpenFile(t.TempDir()+"/stdin", os.O_RDWR|os.O_CREATE|os.O_EXCL, 0o666)
 | |
| 	require.NoError(t, err)
 | |
| 	_, err = f.Write([]byte("00000000000000000000 00000000000000000001 refs/head/main\n"))
 | |
| 	require.NoError(t, err)
 | |
| 	_, err = f.Seek(0, 0)
 | |
| 	require.NoError(t, err)
 | |
| 	defer test.MockVariableValue(os.Stdin, *f)()
 | |
| 
 | |
| 	// Setup the server that processes the hooks.
 | |
| 	ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
 | |
| 		time.Sleep(time.Millisecond * 600)
 | |
| 	}))
 | |
| 	defer ts.Close()
 | |
| 	defer test.MockVariableValue(&setting.LocalURL, ts.URL+"/")()
 | |
| 
 | |
| 	app := cli.NewApp()
 | |
| 	app.Commands = []*cli.Command{subcmdHookPreReceive}
 | |
| 
 | |
| 	t.Run("Should delay", func(t *testing.T) {
 | |
| 		defer test.MockVariableValue(&setting.Git.VerbosePushDelay, time.Millisecond*500)()
 | |
| 		finish := captureOutput(t, os.Stdout)
 | |
| 
 | |
| 		err = app.Run([]string{"./forgejo", "pre-receive"})
 | |
| 		require.NoError(t, err)
 | |
| 		out := finish()
 | |
| 
 | |
| 		require.Contains(t, out, "* Checking 1 references")
 | |
| 		require.Contains(t, out, "Checked 1 references in total")
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Shouldn't delay", func(t *testing.T) {
 | |
| 		defer test.MockVariableValue(&setting.Git.VerbosePushDelay, time.Second*5)()
 | |
| 		finish := captureOutput(t, os.Stdout)
 | |
| 
 | |
| 		err = app.Run([]string{"./forgejo", "pre-receive"})
 | |
| 		require.NoError(t, err)
 | |
| 		out := finish()
 | |
| 
 | |
| 		require.NoError(t, err)
 | |
| 		require.Empty(t, out)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestRunHookUpdate(t *testing.T) {
 | |
| 	app := cli.NewApp()
 | |
| 	app.Commands = []*cli.Command{subcmdHookUpdate}
 | |
| 
 | |
| 	t.Run("Removal of internal reference", func(t *testing.T) {
 | |
| 		defer test.MockVariableValue(&cli.OsExiter, func(code int) {})()
 | |
| 		defer test.MockVariableValue(&setting.IsProd, false)()
 | |
| 		finish := captureOutput(t, os.Stderr)
 | |
| 
 | |
| 		err := app.Run([]string{"./forgejo", "update", "refs/pull/1/head", "0a51ae26bc73c47e2f754560c40904cf14ed51a9", "0000000000000000000000000000000000000000"})
 | |
| 		out := finish()
 | |
| 		assert.Error(t, err)
 | |
| 
 | |
| 		assert.Contains(t, out, "The deletion of refs/pull/1/head is skipped as it's an internal reference.")
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Update of internal reference", func(t *testing.T) {
 | |
| 		err := app.Run([]string{"./forgejo", "update", "refs/pull/1/head", "0a51ae26bc73c47e2f754560c40904cf14ed51a9", "0000000000000000000000000000000000000001"})
 | |
| 		assert.NoError(t, err)
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Removal of branch", func(t *testing.T) {
 | |
| 		err := app.Run([]string{"./forgejo", "update", "refs/head/main", "0a51ae26bc73c47e2f754560c40904cf14ed51a9", "0000000000000000000000000000000000000000"})
 | |
| 		assert.NoError(t, err)
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Not enough arguments", func(t *testing.T) {
 | |
| 		err := app.Run([]string{"./forgejo", "update"})
 | |
| 		assert.NoError(t, err)
 | |
| 	})
 | |
| }
 |