mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-31 02:46:04 +01:00 
			
		
		
		
	Fix some mirror bugs (#18649)
* Fix some mirror bugs * Remove unnecessary code * Fix lint * rename stdard url * Allow more charactors in git ssh protocol url * improve the detection * support ipv6 for git url parse * Fix bug * Fix template * Fix bug * fix template * Fix tmpl * Fix tmpl * Fix parse ssh with interface * Rename functions name Co-authored-by: zeripath <art27@cantab.net>
This commit is contained in:
		| @@ -19,12 +19,6 @@ import ( | |||||||
| // ErrMirrorNotExist mirror does not exist error | // ErrMirrorNotExist mirror does not exist error | ||||||
| var ErrMirrorNotExist = errors.New("Mirror does not exist") | var ErrMirrorNotExist = errors.New("Mirror does not exist") | ||||||
|  |  | ||||||
| // RemoteMirrorer defines base methods for pull/push mirrors. |  | ||||||
| type RemoteMirrorer interface { |  | ||||||
| 	GetRepository() *Repository |  | ||||||
| 	GetRemoteName() string |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Mirror represents mirror information of a repository. | // Mirror represents mirror information of a repository. | ||||||
| type Mirror struct { | type Mirror struct { | ||||||
| 	ID          int64       `xorm:"pk autoincr"` | 	ID          int64       `xorm:"pk autoincr"` | ||||||
|   | |||||||
| @@ -6,11 +6,12 @@ package git | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
| 	"net/url" |  | ||||||
|  | 	giturl "code.gitea.io/gitea/modules/git/url" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // GetRemoteAddress returns the url of a specific remote of the repository. | // GetRemoteAddress returns remote url of git repository in the repoPath with special remote name | ||||||
| func GetRemoteAddress(ctx context.Context, repoPath, remoteName string) (*url.URL, error) { | func GetRemoteAddress(ctx context.Context, repoPath, remoteName string) (string, error) { | ||||||
| 	var cmd *Command | 	var cmd *Command | ||||||
| 	if CheckGitVersionAtLeast("2.7") == nil { | 	if CheckGitVersionAtLeast("2.7") == nil { | ||||||
| 		cmd = NewCommand(ctx, "remote", "get-url", remoteName) | 		cmd = NewCommand(ctx, "remote", "get-url", remoteName) | ||||||
| @@ -20,11 +21,20 @@ func GetRemoteAddress(ctx context.Context, repoPath, remoteName string) (*url.UR | |||||||
|  |  | ||||||
| 	result, _, err := cmd.RunStdString(&RunOpts{Dir: repoPath}) | 	result, _, err := cmd.RunStdString(&RunOpts{Dir: repoPath}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return "", err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if len(result) > 0 { | 	if len(result) > 0 { | ||||||
| 		result = result[:len(result)-1] | 		result = result[:len(result)-1] | ||||||
| 	} | 	} | ||||||
| 	return url.Parse(result) | 	return result, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // GetRemoteURL returns the url of a specific remote of the repository. | ||||||
|  | func GetRemoteURL(ctx context.Context, repoPath, remoteName string) (*giturl.GitURL, error) { | ||||||
|  | 	addr, err := GetRemoteAddress(ctx, repoPath, remoteName) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return giturl.Parse(addr) | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										90
									
								
								modules/git/url/url.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								modules/git/url/url.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | |||||||
|  | // Copyright 2022 The Gitea Authors. All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT-style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  |  | ||||||
|  | package url | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	stdurl "net/url" | ||||||
|  | 	"strings" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // ErrWrongURLFormat represents an error with wrong url format | ||||||
|  | type ErrWrongURLFormat struct { | ||||||
|  | 	URL string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (err ErrWrongURLFormat) Error() string { | ||||||
|  | 	return fmt.Sprintf("git URL %s format is wrong", err.URL) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // GitURL represents a git URL | ||||||
|  | type GitURL struct { | ||||||
|  | 	*stdurl.URL | ||||||
|  | 	extraMark int // 0 no extra 1 scp 2 file path with no prefix | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // String returns the URL's string | ||||||
|  | func (u *GitURL) String() string { | ||||||
|  | 	switch u.extraMark { | ||||||
|  | 	case 0: | ||||||
|  | 		return u.URL.String() | ||||||
|  | 	case 1: | ||||||
|  | 		return fmt.Sprintf("%s@%s:%s", u.User.Username(), u.Host, u.Path) | ||||||
|  | 	case 2: | ||||||
|  | 		return u.Path | ||||||
|  | 	default: | ||||||
|  | 		return "" | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Parse parse all kinds of git URL | ||||||
|  | func Parse(remote string) (*GitURL, error) { | ||||||
|  | 	if strings.Contains(remote, "://") { | ||||||
|  | 		u, err := stdurl.Parse(remote) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		return &GitURL{URL: u}, nil | ||||||
|  | 	} else if strings.Contains(remote, "@") && strings.Contains(remote, ":") { | ||||||
|  | 		url := stdurl.URL{ | ||||||
|  | 			Scheme: "ssh", | ||||||
|  | 		} | ||||||
|  | 		squareBrackets := false | ||||||
|  | 		lastIndex := -1 | ||||||
|  | 	FOR: | ||||||
|  | 		for i := 0; i < len(remote); i++ { | ||||||
|  | 			switch remote[i] { | ||||||
|  | 			case '@': | ||||||
|  | 				url.User = stdurl.User(remote[:i]) | ||||||
|  | 				lastIndex = i + 1 | ||||||
|  | 			case ':': | ||||||
|  | 				if !squareBrackets { | ||||||
|  | 					url.Host = strings.ReplaceAll(remote[lastIndex:i], "%25", "%") | ||||||
|  | 					if len(remote) <= i+1 { | ||||||
|  | 						return nil, ErrWrongURLFormat{URL: remote} | ||||||
|  | 					} | ||||||
|  | 					url.Path = remote[i+1:] | ||||||
|  | 					break FOR | ||||||
|  | 				} | ||||||
|  | 			case '[': | ||||||
|  | 				squareBrackets = true | ||||||
|  | 			case ']': | ||||||
|  | 				squareBrackets = false | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return &GitURL{ | ||||||
|  | 			URL:       &url, | ||||||
|  | 			extraMark: 1, | ||||||
|  | 		}, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return &GitURL{ | ||||||
|  | 		URL: &stdurl.URL{ | ||||||
|  | 			Scheme: "file", | ||||||
|  | 			Path:   remote, | ||||||
|  | 		}, | ||||||
|  | 		extraMark: 2, | ||||||
|  | 	}, nil | ||||||
|  | } | ||||||
							
								
								
									
										167
									
								
								modules/git/url/url_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								modules/git/url/url_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,167 @@ | |||||||
|  | // Copyright 2022 The Gitea Authors. All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT-style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  |  | ||||||
|  | package url | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"net/url" | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestParseGitURLs(t *testing.T) { | ||||||
|  | 	kases := []struct { | ||||||
|  | 		kase     string | ||||||
|  | 		expected *GitURL | ||||||
|  | 	}{ | ||||||
|  | 		{ | ||||||
|  | 			kase: "git@127.0.0.1:go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "ssh", | ||||||
|  | 					User:   url.User("git"), | ||||||
|  | 					Host:   "127.0.0.1", | ||||||
|  | 					Path:   "go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 1, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "git@[fe80:14fc:cec5:c174:d88%2510]:go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "ssh", | ||||||
|  | 					User:   url.User("git"), | ||||||
|  | 					Host:   "[fe80:14fc:cec5:c174:d88%10]", | ||||||
|  | 					Path:   "go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 1, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "git@[::1]:go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "ssh", | ||||||
|  | 					User:   url.User("git"), | ||||||
|  | 					Host:   "[::1]", | ||||||
|  | 					Path:   "go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 1, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "git@github.com:go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "ssh", | ||||||
|  | 					User:   url.User("git"), | ||||||
|  | 					Host:   "github.com", | ||||||
|  | 					Path:   "go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 1, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "ssh://git@github.com/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "ssh", | ||||||
|  | 					User:   url.User("git"), | ||||||
|  | 					Host:   "github.com", | ||||||
|  | 					Path:   "/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 0, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "ssh://git@[::1]/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "ssh", | ||||||
|  | 					User:   url.User("git"), | ||||||
|  | 					Host:   "[::1]", | ||||||
|  | 					Path:   "/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 0, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "/repositories/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "file", | ||||||
|  | 					Path:   "/repositories/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 2, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "file:///repositories/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "file", | ||||||
|  | 					Path:   "/repositories/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 0, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "https://github.com/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "https", | ||||||
|  | 					Host:   "github.com", | ||||||
|  | 					Path:   "/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 0, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "https://git:git@github.com/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "https", | ||||||
|  | 					Host:   "github.com", | ||||||
|  | 					User:   url.UserPassword("git", "git"), | ||||||
|  | 					Path:   "/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 0, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			kase: "https://[fe80:14fc:cec5:c174:d88%2510]:20/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "https", | ||||||
|  | 					Host:   "[fe80:14fc:cec5:c174:d88%10]:20", | ||||||
|  | 					Path:   "/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 0, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  |  | ||||||
|  | 		{ | ||||||
|  | 			kase: "git://github.com/go-gitea/gitea.git", | ||||||
|  | 			expected: &GitURL{ | ||||||
|  | 				URL: &url.URL{ | ||||||
|  | 					Scheme: "git", | ||||||
|  | 					Host:   "github.com", | ||||||
|  | 					Path:   "/go-gitea/gitea.git", | ||||||
|  | 				}, | ||||||
|  | 				extraMark: 0, | ||||||
|  | 			}, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, kase := range kases { | ||||||
|  | 		t.Run(kase.kase, func(t *testing.T) { | ||||||
|  | 			u, err := Parse(kase.kase) | ||||||
|  | 			assert.NoError(t, err) | ||||||
|  | 			assert.EqualValues(t, kase.expected.extraMark, u.extraMark) | ||||||
|  | 			assert.EqualValues(t, *kase.expected, *u) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -32,6 +32,7 @@ import ( | |||||||
| 	"code.gitea.io/gitea/modules/base" | 	"code.gitea.io/gitea/modules/base" | ||||||
| 	"code.gitea.io/gitea/modules/emoji" | 	"code.gitea.io/gitea/modules/emoji" | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	giturl "code.gitea.io/gitea/modules/git/url" | ||||||
| 	"code.gitea.io/gitea/modules/json" | 	"code.gitea.io/gitea/modules/json" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/markup" | 	"code.gitea.io/gitea/modules/markup" | ||||||
| @@ -971,20 +972,35 @@ type remoteAddress struct { | |||||||
| 	Password string | 	Password string | ||||||
| } | } | ||||||
|  |  | ||||||
| func mirrorRemoteAddress(ctx context.Context, m repo_model.RemoteMirrorer) remoteAddress { | func mirrorRemoteAddress(ctx context.Context, m *repo_model.Repository, remoteName string) remoteAddress { | ||||||
| 	a := remoteAddress{} | 	a := remoteAddress{} | ||||||
|  | 	if !m.IsMirror { | ||||||
| 	u, err := git.GetRemoteAddress(ctx, m.GetRepository().RepoPath(), m.GetRemoteName()) |  | ||||||
| 	if err != nil { |  | ||||||
| 		log.Error("GetRemoteAddress %v", err) |  | ||||||
| 		return a | 		return a | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	remoteURL := m.OriginalURL | ||||||
|  | 	if remoteURL == "" { | ||||||
|  | 		var err error | ||||||
|  | 		remoteURL, err = git.GetRemoteAddress(ctx, m.RepoPath(), remoteName) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Error("GetRemoteURL %v", err) | ||||||
|  | 			return a | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	u, err := giturl.Parse(remoteURL) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Error("giturl.Parse %v", err) | ||||||
|  | 		return a | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if u.Scheme != "ssh" && u.Scheme != "file" { | ||||||
| 		if u.User != nil { | 		if u.User != nil { | ||||||
| 			a.Username = u.User.Username() | 			a.Username = u.User.Username() | ||||||
| 			a.Password, _ = u.User.Password() | 			a.Password, _ = u.User.Password() | ||||||
| 		} | 		} | ||||||
| 		u.User = nil | 		u.User = nil | ||||||
|  | 	} | ||||||
| 	a.Address = u.String() | 	a.Address = u.String() | ||||||
|  |  | ||||||
| 	return a | 	return a | ||||||
|   | |||||||
| @@ -215,22 +215,24 @@ func SettingsPost(ctx *context.Context) { | |||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		u, _ := git.GetRemoteAddress(ctx, ctx.Repo.Repository.RepoPath(), ctx.Repo.Mirror.GetRemoteName()) | 		u, err := git.GetRemoteURL(ctx, ctx.Repo.Repository.RepoPath(), ctx.Repo.Mirror.GetRemoteName()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			ctx.Data["Err_MirrorAddress"] = true | ||||||
|  | 			handleSettingRemoteAddrError(ctx, err, form) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
| 		if u.User != nil && form.MirrorPassword == "" && form.MirrorUsername == u.User.Username() { | 		if u.User != nil && form.MirrorPassword == "" && form.MirrorUsername == u.User.Username() { | ||||||
| 			form.MirrorPassword, _ = u.User.Password() | 			form.MirrorPassword, _ = u.User.Password() | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		address, err := forms.ParseRemoteAddr(form.MirrorAddress, form.MirrorUsername, form.MirrorPassword) | 		err = migrations.IsMigrateURLAllowed(u.String(), ctx.Doer) | ||||||
| 		if err == nil { |  | ||||||
| 			err = migrations.IsMigrateURLAllowed(address, ctx.Doer) |  | ||||||
| 		} |  | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			ctx.Data["Err_MirrorAddress"] = true | 			ctx.Data["Err_MirrorAddress"] = true | ||||||
| 			handleSettingRemoteAddrError(ctx, err, form) | 			handleSettingRemoteAddrError(ctx, err, form) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		if err := mirror_service.UpdateAddress(ctx, ctx.Repo.Mirror, address); err != nil { | 		if err := mirror_service.UpdateAddress(ctx, ctx.Repo.Mirror, u.String()); err != nil { | ||||||
| 			ctx.ServerError("UpdateAddress", err) | 			ctx.ServerError("UpdateAddress", err) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -210,9 +210,10 @@ func runSync(ctx context.Context, m *repo_model.Mirror) ([]*mirrorSyncResult, bo | |||||||
| 	} | 	} | ||||||
| 	gitArgs = append(gitArgs, m.GetRemoteName()) | 	gitArgs = append(gitArgs, m.GetRemoteName()) | ||||||
|  |  | ||||||
| 	remoteAddr, remoteErr := git.GetRemoteAddress(ctx, repoPath, m.GetRemoteName()) | 	remoteURL, remoteErr := git.GetRemoteURL(ctx, repoPath, m.GetRemoteName()) | ||||||
| 	if remoteErr != nil { | 	if remoteErr != nil { | ||||||
| 		log.Error("SyncMirrors [repo: %-v]: GetRemoteAddress Error %v", m.Repo, remoteErr) | 		log.Error("SyncMirrors [repo: %-v]: GetRemoteAddress Error %v", m.Repo, remoteErr) | ||||||
|  | 		return nil, false | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	stdoutBuilder := strings.Builder{} | 	stdoutBuilder := strings.Builder{} | ||||||
| @@ -291,7 +292,7 @@ func runSync(ctx context.Context, m *repo_model.Mirror) ([]*mirrorSyncResult, bo | |||||||
|  |  | ||||||
| 	if m.LFS && setting.LFS.StartServer { | 	if m.LFS && setting.LFS.StartServer { | ||||||
| 		log.Trace("SyncMirrors [repo: %-v]: syncing LFS objects...", m.Repo) | 		log.Trace("SyncMirrors [repo: %-v]: syncing LFS objects...", m.Repo) | ||||||
| 		endpoint := lfs.DetermineEndpoint(remoteAddr.String(), m.LFSEndpoint) | 		endpoint := lfs.DetermineEndpoint(remoteURL.String(), m.LFSEndpoint) | ||||||
| 		lfsClient := lfs.NewClient(endpoint, nil) | 		lfsClient := lfs.NewClient(endpoint, nil) | ||||||
| 		if err = repo_module.StoreMissingLfsObjectsInRepository(ctx, m.Repo, gitRepo, lfsClient); err != nil { | 		if err = repo_module.StoreMissingLfsObjectsInRepository(ctx, m.Repo, gitRepo, lfsClient); err != nil { | ||||||
| 			log.Error("SyncMirrors [repo: %-v]: failed to synchronize LFS objects for repository: %v", m.Repo, err) | 			log.Error("SyncMirrors [repo: %-v]: failed to synchronize LFS objects for repository: %v", m.Repo, err) | ||||||
|   | |||||||
| @@ -131,7 +131,7 @@ func runPushSync(ctx context.Context, m *repo_model.PushMirror) error { | |||||||
| 	timeout := time.Duration(setting.Git.Timeout.Mirror) * time.Second | 	timeout := time.Duration(setting.Git.Timeout.Mirror) * time.Second | ||||||
|  |  | ||||||
| 	performPush := func(path string) error { | 	performPush := func(path string) error { | ||||||
| 		remoteAddr, err := git.GetRemoteAddress(ctx, path, m.RemoteName) | 		remoteURL, err := git.GetRemoteURL(ctx, path, m.RemoteName) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Error("GetRemoteAddress(%s) Error %v", path, err) | 			log.Error("GetRemoteAddress(%s) Error %v", path, err) | ||||||
| 			return errors.New("Unexpected error") | 			return errors.New("Unexpected error") | ||||||
| @@ -147,7 +147,7 @@ func runPushSync(ctx context.Context, m *repo_model.PushMirror) error { | |||||||
| 			} | 			} | ||||||
| 			defer gitRepo.Close() | 			defer gitRepo.Close() | ||||||
|  |  | ||||||
| 			endpoint := lfs.DetermineEndpoint(remoteAddr.String(), "") | 			endpoint := lfs.DetermineEndpoint(remoteURL.String(), "") | ||||||
| 			lfsClient := lfs.NewClient(endpoint, nil) | 			lfsClient := lfs.NewClient(endpoint, nil) | ||||||
| 			if err := pushAllLFSObjects(ctx, gitRepo, lfsClient); err != nil { | 			if err := pushAllLFSObjects(ctx, gitRepo, lfsClient); err != nil { | ||||||
| 				return util.SanitizeErrorCredentialURLs(err) | 				return util.SanitizeErrorCredentialURLs(err) | ||||||
|   | |||||||
| @@ -37,7 +37,9 @@ | |||||||
| 						{{end}} | 						{{end}} | ||||||
| 					</div> | 					</div> | ||||||
| 				</div> | 				</div> | ||||||
| 				{{if .IsMirror}}<div class="fork-flag">{{$.i18n.Tr "repo.mirror_from"}} <a target="_blank" rel="noopener noreferrer" href="{{if .SanitizedOriginalURL}}{{.SanitizedOriginalURL}}{{else}}{{(MirrorRemoteAddress $.Context $.Mirror).Address}}{{end}}">{{if .SanitizedOriginalURL}}{{.SanitizedOriginalURL}}{{else}}{{(MirrorRemoteAddress $.Context $.Mirror).Address}}{{end}}</a></div>{{end}} | 				{{if .IsMirror}} | ||||||
|  | 				{{$address := MirrorRemoteAddress $.Context . $.Mirror.GetRemoteName}} | ||||||
|  | 				<div class="fork-flag">{{$.i18n.Tr "repo.mirror_from"}} <a target="_blank" rel="noopener noreferrer" href="{{$address.Address}}">{{$address.Address}}</a></div>{{end}} | ||||||
| 				{{if .IsFork}}<div class="fork-flag">{{$.i18n.Tr "repo.forked_from"}} <a href="{{.BaseRepo.Link}}">{{.BaseRepo.FullName}}</a></div>{{end}} | 				{{if .IsFork}}<div class="fork-flag">{{$.i18n.Tr "repo.forked_from"}} <a href="{{.BaseRepo.Link}}">{{.BaseRepo.FullName}}</a></div>{{end}} | ||||||
| 				{{if .IsGenerated}}<div class="fork-flag">{{$.i18n.Tr "repo.generated_from"}} <a href="{{.TemplateRepo.Link}}">{{.TemplateRepo.FullName}}</a></div>{{end}} | 				{{if .IsGenerated}}<div class="fork-flag">{{$.i18n.Tr "repo.generated_from"}} <a href="{{.TemplateRepo.Link}}">{{.TemplateRepo.FullName}}</a></div>{{end}} | ||||||
| 			</div> | 			</div> | ||||||
|   | |||||||
| @@ -91,7 +91,7 @@ | |||||||
| 					{{if .Repository.IsMirror}} | 					{{if .Repository.IsMirror}} | ||||||
| 					<tbody> | 					<tbody> | ||||||
| 						<tr> | 						<tr> | ||||||
| 							<td>{{(MirrorRemoteAddress $.Context .Mirror).Address}}</td> | 							<td>{{(MirrorRemoteAddress $.Context .Repository .Mirror.GetRemoteName).Address}}</td> | ||||||
| 							<td>{{$.i18n.Tr "repo.settings.mirror_settings.direction.pull"}}</td> | 							<td>{{$.i18n.Tr "repo.settings.mirror_settings.direction.pull"}}</td> | ||||||
| 							<td>{{.Mirror.UpdatedUnix.AsTime}}</td> | 							<td>{{.Mirror.UpdatedUnix.AsTime}}</td> | ||||||
| 							<td class="right aligned"> | 							<td class="right aligned"> | ||||||
| @@ -119,7 +119,7 @@ | |||||||
| 										<label for="interval">{{.i18n.Tr "repo.mirror_interval" .MinimumMirrorInterval}}</label> | 										<label for="interval">{{.i18n.Tr "repo.mirror_interval" .MinimumMirrorInterval}}</label> | ||||||
| 										<input id="interval" name="interval" value="{{.MirrorInterval}}"> | 										<input id="interval" name="interval" value="{{.MirrorInterval}}"> | ||||||
| 									</div> | 									</div> | ||||||
| 									{{$address := MirrorRemoteAddress $.Context .Mirror}} | 									{{$address := MirrorRemoteAddress $.Context .Repository .Mirror.GetRemoteName}} | ||||||
| 									<div class="field {{if .Err_MirrorAddress}}error{{end}}"> | 									<div class="field {{if .Err_MirrorAddress}}error{{end}}"> | ||||||
| 										<label for="mirror_address">{{.i18n.Tr "repo.mirror_address"}}</label> | 										<label for="mirror_address">{{.i18n.Tr "repo.mirror_address"}}</label> | ||||||
| 										<input id="mirror_address" name="mirror_address" value="{{$address.Address}}" required> | 										<input id="mirror_address" name="mirror_address" value="{{$address.Address}}" required> | ||||||
| @@ -168,7 +168,7 @@ | |||||||
| 					<tbody> | 					<tbody> | ||||||
| 						{{range .PushMirrors}} | 						{{range .PushMirrors}} | ||||||
| 						<tr> | 						<tr> | ||||||
| 							{{$address := MirrorRemoteAddress $.Context .}} | 							{{$address := MirrorRemoteAddress $.Context $.Repository .GetRemoteName}} | ||||||
| 							<td>{{$address.Address}}</td> | 							<td>{{$address.Address}}</td> | ||||||
| 							<td>{{$.i18n.Tr "repo.settings.mirror_settings.direction.push"}}</td> | 							<td>{{$.i18n.Tr "repo.settings.mirror_settings.direction.push"}}</td> | ||||||
| 							<td>{{if .LastUpdateUnix}}{{.LastUpdateUnix.AsTime}}{{else}}{{$.i18n.Tr "never"}}{{end}} {{if .LastError}}<div class="ui red label tooltip" data-content="{{.LastError}}">{{$.i18n.Tr "error"}}</div>{{end}}</td> | 							<td>{{if .LastUpdateUnix}}{{.LastUpdateUnix.AsTime}}{{else}}{{$.i18n.Tr "never"}}{{end}} {{if .LastError}}<div class="ui red label tooltip" data-content="{{.LastError}}">{{$.i18n.Tr "error"}}</div>{{end}}</td> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user