Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
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.
223 lines
6.0 KiB
223 lines
6.0 KiB
// Copyright 2020 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 doctor |
|
|
|
import ( |
|
"fmt" |
|
"os" |
|
"os/exec" |
|
"path" |
|
"strings" |
|
|
|
"code.gitea.io/gitea/models" |
|
"code.gitea.io/gitea/models/db" |
|
repo_model "code.gitea.io/gitea/models/repo" |
|
user_model "code.gitea.io/gitea/models/user" |
|
"code.gitea.io/gitea/modules/git" |
|
"code.gitea.io/gitea/modules/log" |
|
"code.gitea.io/gitea/modules/repository" |
|
"code.gitea.io/gitea/modules/setting" |
|
"code.gitea.io/gitea/modules/structs" |
|
"code.gitea.io/gitea/modules/util" |
|
|
|
lru "github.com/hashicorp/golang-lru" |
|
"xorm.io/builder" |
|
) |
|
|
|
func iterateRepositories(each func(*repo_model.Repository) error) error { |
|
err := db.Iterate( |
|
db.DefaultContext, |
|
new(repo_model.Repository), |
|
builder.Gt{"id": 0}, |
|
func(idx int, bean interface{}) error { |
|
return each(bean.(*repo_model.Repository)) |
|
}, |
|
) |
|
return err |
|
} |
|
|
|
func checkScriptType(logger log.Logger, autofix bool) error { |
|
path, err := exec.LookPath(setting.ScriptType) |
|
if err != nil { |
|
logger.Critical("ScriptType \"%q\" is not on the current PATH. Error: %v", setting.ScriptType, err) |
|
return fmt.Errorf("ScriptType \"%q\" is not on the current PATH. Error: %v", setting.ScriptType, err) |
|
} |
|
logger.Info("ScriptType %s is on the current PATH at %s", setting.ScriptType, path) |
|
return nil |
|
} |
|
|
|
func checkHooks(logger log.Logger, autofix bool) error { |
|
if err := iterateRepositories(func(repo *repo_model.Repository) error { |
|
results, err := repository.CheckDelegateHooks(repo.RepoPath()) |
|
if err != nil { |
|
logger.Critical("Unable to check delegate hooks for repo %-v. ERROR: %v", repo, err) |
|
return fmt.Errorf("Unable to check delegate hooks for repo %-v. ERROR: %v", repo, err) |
|
} |
|
if len(results) > 0 && autofix { |
|
logger.Warn("Regenerated hooks for %s", repo.FullName()) |
|
if err := repository.CreateDelegateHooks(repo.RepoPath()); err != nil { |
|
logger.Critical("Unable to recreate delegate hooks for %-v. ERROR: %v", repo, err) |
|
return fmt.Errorf("Unable to recreate delegate hooks for %-v. ERROR: %v", repo, err) |
|
} |
|
} |
|
for _, result := range results { |
|
logger.Warn(result) |
|
} |
|
return nil |
|
}); err != nil { |
|
logger.Critical("Errors noted whilst checking delegate hooks.") |
|
return err |
|
} |
|
return nil |
|
} |
|
|
|
func checkUserStarNum(logger log.Logger, autofix bool) error { |
|
if err := models.DoctorUserStarNum(); err != nil { |
|
logger.Critical("Unable update User Stars numbers") |
|
return err |
|
} |
|
return nil |
|
} |
|
|
|
func checkEnablePushOptions(logger log.Logger, autofix bool) error { |
|
numRepos := 0 |
|
numNeedUpdate := 0 |
|
|
|
if err := iterateRepositories(func(repo *repo_model.Repository) error { |
|
numRepos++ |
|
r, err := git.OpenRepository(repo.RepoPath()) |
|
if err != nil { |
|
return err |
|
} |
|
defer r.Close() |
|
|
|
if autofix { |
|
_, err := git.NewCommand("config", "receive.advertisePushOptions", "true").RunInDir(r.Path) |
|
return err |
|
} |
|
|
|
value, err := git.NewCommand("config", "receive.advertisePushOptions").RunInDir(r.Path) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
result, valid := git.ParseBool(strings.TrimSpace(value)) |
|
if !result || !valid { |
|
numNeedUpdate++ |
|
logger.Info("%s: does not have receive.advertisePushOptions set correctly: %q", repo.FullName(), value) |
|
} |
|
return nil |
|
}); err != nil { |
|
logger.Critical("Unable to EnablePushOptions: %v", err) |
|
return err |
|
} |
|
|
|
if autofix { |
|
logger.Info("Enabled push options for %d repositories.", numRepos) |
|
} else { |
|
logger.Info("Checked %d repositories, %d need updates.", numRepos, numNeedUpdate) |
|
|
|
} |
|
|
|
return nil |
|
} |
|
|
|
func checkDaemonExport(logger log.Logger, autofix bool) error { |
|
numRepos := 0 |
|
numNeedUpdate := 0 |
|
cache, err := lru.New(512) |
|
if err != nil { |
|
logger.Critical("Unable to create cache: %v", err) |
|
return err |
|
} |
|
if err := iterateRepositories(func(repo *repo_model.Repository) error { |
|
numRepos++ |
|
|
|
if owner, has := cache.Get(repo.OwnerID); has { |
|
repo.Owner = owner.(*user_model.User) |
|
} else { |
|
if err := repo.GetOwner(db.DefaultContext); err != nil { |
|
return err |
|
} |
|
cache.Add(repo.OwnerID, repo.Owner) |
|
} |
|
|
|
// Create/Remove git-daemon-export-ok for git-daemon... |
|
daemonExportFile := path.Join(repo.RepoPath(), `git-daemon-export-ok`) |
|
isExist, err := util.IsExist(daemonExportFile) |
|
if err != nil { |
|
log.Error("Unable to check if %s exists. Error: %v", daemonExportFile, err) |
|
return err |
|
} |
|
isPublic := !repo.IsPrivate && repo.Owner.Visibility == structs.VisibleTypePublic |
|
|
|
if isPublic != isExist { |
|
numNeedUpdate++ |
|
if autofix { |
|
if !isPublic && isExist { |
|
if err = util.Remove(daemonExportFile); err != nil { |
|
log.Error("Failed to remove %s: %v", daemonExportFile, err) |
|
} |
|
} else if isPublic && !isExist { |
|
if f, err := os.Create(daemonExportFile); err != nil { |
|
log.Error("Failed to create %s: %v", daemonExportFile, err) |
|
} else { |
|
f.Close() |
|
} |
|
} |
|
} |
|
} |
|
return nil |
|
}); err != nil { |
|
logger.Critical("Unable to checkDaemonExport: %v", err) |
|
return err |
|
} |
|
|
|
if autofix { |
|
logger.Info("Updated git-daemon-export-ok files for %d of %d repositories.", numNeedUpdate, numRepos) |
|
} else { |
|
logger.Info("Checked %d repositories, %d need updates.", numRepos, numNeedUpdate) |
|
} |
|
|
|
return nil |
|
} |
|
|
|
func init() { |
|
Register(&Check{ |
|
Title: "Check if SCRIPT_TYPE is available", |
|
Name: "script-type", |
|
IsDefault: false, |
|
Run: checkScriptType, |
|
Priority: 5, |
|
}) |
|
Register(&Check{ |
|
Title: "Check if hook files are up-to-date and executable", |
|
Name: "hooks", |
|
IsDefault: false, |
|
Run: checkHooks, |
|
Priority: 6, |
|
}) |
|
Register(&Check{ |
|
Title: "Recalculate Stars number for all user", |
|
Name: "recalculate-stars-number", |
|
IsDefault: false, |
|
Run: checkUserStarNum, |
|
Priority: 6, |
|
}) |
|
Register(&Check{ |
|
Title: "Enable push options", |
|
Name: "enable-push-options", |
|
IsDefault: false, |
|
Run: checkEnablePushOptions, |
|
Priority: 7, |
|
}) |
|
Register(&Check{ |
|
Title: "Check git-daemon-export-ok files", |
|
Name: "check-git-daemon-export-ok", |
|
IsDefault: false, |
|
Run: checkDaemonExport, |
|
Priority: 8, |
|
}) |
|
}
|
|
|