471cfdd161
- 1 - Add: Gitea now creates timestamped database backup bundles under `[backup].PATH`, exposes the backup schedule in the installer, and surfaces the `database_backup` cron task in admin monitoring. - 2 - Add: installed instances now use `.gitea-installed` and `.gitea-recovery.ini` to enter email-gated recovery instead of falling back to public install mode when configuration or database access is broken. - 3 - Mod: the installer recovery flow now covers backup-bundle restore, bundled or manual `app.ini` reuse, uploaded SQL/GZ database restores, and repository-filesystem recovery with source-specific validation, confirmations, and preserved launcher state. - 4 - Fix: recovery now restores bundled `app.ini` snapshots when needed, discovers backup bundles from both the active backup path and persisted `.gitea-recovery.ini` path, and preserves SMTP and other rebuilt settings correctly when `app.ini` is missing or incomplete. - 5 - Fix: recovery validation and restore handling now accept either a selected backup bundle or an uploaded SQL/GZ dump, keep sensitive secrets and existing `LFS_JWT_SECRET` when appropriate, clear SQLite restore targets before import, and complete the post-install handoff without redirect loops. - 6 - Mod: fresh installs now default recovery email authorization to enabled with first-admin fallback, and the install/recovery UI, styling, and EN/RO wording were refined to match the final launcher behavior. Co-Authored-By: petru @ codex (GPT-5) <codex@openai.com> (cherry picked from commit 9879caf2292691b0cb521d12e6fee924b066bae2)
127 lines
3.7 KiB
Go
127 lines
3.7 KiB
Go
// Copyright 2014 The Gogs Authors. All rights reserved.
|
|
// Copyright 2019 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package web
|
|
|
|
import (
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"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/log"
|
|
"code.gitea.io/gitea/modules/optional"
|
|
"code.gitea.io/gitea/modules/setting"
|
|
"code.gitea.io/gitea/modules/sitemap"
|
|
"code.gitea.io/gitea/modules/structs"
|
|
"code.gitea.io/gitea/modules/templates"
|
|
"code.gitea.io/gitea/modules/web/middleware"
|
|
"code.gitea.io/gitea/routers/web/auth"
|
|
"code.gitea.io/gitea/routers/web/user"
|
|
"code.gitea.io/gitea/services/context"
|
|
)
|
|
|
|
const (
|
|
// tplHome home page template
|
|
tplHome templates.TplName = "home"
|
|
)
|
|
|
|
// Home render home page
|
|
func Home(ctx *context.Context) {
|
|
if ctx.IsSigned {
|
|
if !ctx.Doer.IsActive {
|
|
inactiveUser := ctx.Doer
|
|
auth.HandleSignOut(ctx)
|
|
auth.RenderInactiveAccountPrompt(ctx, inactiveUser)
|
|
} else if ctx.Doer.ProhibitLogin {
|
|
log.Info("Failed authentication attempt for %s from %s", ctx.Doer.Name, ctx.RemoteAddr())
|
|
ctx.Data["Title"] = ctx.Tr("auth.prohibit_login")
|
|
ctx.HTML(http.StatusOK, "user/auth/prohibit_login")
|
|
} else if ctx.Doer.MustChangePassword {
|
|
ctx.Data["Title"] = ctx.Tr("auth.must_change_password")
|
|
ctx.Data["ChangePasscodeLink"] = setting.AppSubURL + "/user/change_password"
|
|
middleware.SetRedirectToCookie(ctx.Resp, setting.AppSubURL+ctx.Req.URL.RequestURI())
|
|
ctx.Redirect(setting.AppSubURL + "/user/settings/change_password")
|
|
} else {
|
|
user.Dashboard(ctx)
|
|
}
|
|
return
|
|
// Check non-logged users landing page.
|
|
} else if setting.LandingPageURL != setting.LandingPageHome {
|
|
ctx.Redirect(setting.AppSubURL + string(setting.LandingPageURL))
|
|
return
|
|
}
|
|
|
|
// Check auto-login.
|
|
if ctx.GetSiteCookie(setting.CookieRememberName) != "" {
|
|
ctx.Redirect(setting.AppSubURL + "/user/login")
|
|
return
|
|
}
|
|
|
|
ctx.Data["PageIsHome"] = true
|
|
ctx.Data["IsRepoIndexerEnabled"] = setting.Indexer.RepoIndexerEnabled
|
|
ctx.HTML(http.StatusOK, tplHome)
|
|
}
|
|
|
|
// start edit/add - by petru @ codex
|
|
// PostInstallRedirect absorbs the temporary installer handoff URL after the normal web server is up.
|
|
func PostInstallRedirect(ctx *context.Context) {
|
|
ctx.Redirect(setting.AppSubURL + "/")
|
|
}
|
|
|
|
// end edit/add - by petru @ codex
|
|
|
|
// HomeSitemap renders the main sitemap
|
|
func HomeSitemap(ctx *context.Context) {
|
|
m := sitemap.NewSitemapIndex()
|
|
if !setting.Service.Explore.DisableUsersPage {
|
|
_, cnt, err := user_model.SearchUsers(ctx, user_model.SearchUserOptions{
|
|
Types: []user_model.UserType{user_model.UserTypeIndividual},
|
|
ListOptions: db.ListOptions{PageSize: 1},
|
|
IsActive: optional.Some(true),
|
|
Visible: []structs.VisibleType{structs.VisibleTypePublic},
|
|
})
|
|
if err != nil {
|
|
ctx.ServerError("SearchUsers", err)
|
|
return
|
|
}
|
|
count := int(cnt)
|
|
idx := 1
|
|
for i := 0; i < count; i += setting.UI.SitemapPagingNum {
|
|
m.Add(sitemap.URL{URL: setting.AppURL + "explore/users/sitemap-" + strconv.Itoa(idx) + ".xml"})
|
|
idx++
|
|
}
|
|
}
|
|
|
|
_, cnt, err := repo_model.SearchRepository(ctx, repo_model.SearchRepoOptions{
|
|
ListOptions: db.ListOptions{
|
|
PageSize: 1,
|
|
},
|
|
Actor: ctx.Doer,
|
|
AllPublic: true,
|
|
})
|
|
if err != nil {
|
|
ctx.ServerError("SearchRepository", err)
|
|
return
|
|
}
|
|
count := int(cnt)
|
|
idx := 1
|
|
for i := 0; i < count; i += setting.UI.SitemapPagingNum {
|
|
m.Add(sitemap.URL{URL: setting.AppURL + "explore/repos/sitemap-" + strconv.Itoa(idx) + ".xml"})
|
|
idx++
|
|
}
|
|
|
|
ctx.Resp.Header().Set("Content-Type", "text/xml")
|
|
if _, err := m.WriteTo(ctx.Resp); err != nil {
|
|
log.Error("Failed writing sitemap: %v", err)
|
|
}
|
|
}
|
|
|
|
// NotFound render 404 page
|
|
func NotFound(ctx *context.Context) {
|
|
ctx.Data["Title"] = "Page Not Found"
|
|
ctx.NotFound(nil)
|
|
}
|