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:
parent
88f2e457d8
commit
ce3dd04c63
10 changed files with 316 additions and 34 deletions
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
if u.User != nil {
|
remoteURL := m.OriginalURL
|
||||||
a.Username = u.User.Username()
|
if remoteURL == "" {
|
||||||
a.Password, _ = u.User.Password()
|
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 {
|
||||||
|
a.Username = u.User.Username()
|
||||||
|
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>
|
||||||
|
|
Loading…
Reference in a new issue