Files
gitea/routers/install/install.go
T
petru 512e577c3f
release-nightly / nightly-binary (push) Has been cancelled
release-nightly / nightly-container (push) Has been cancelled
Added - Added optional sensitive-secret import for installer app.ini uploads.
- 1 - I added an explicit installer checkbox for importing sensitive secrets from `app.ini` in `templates/install.tmpl`.
- 2 - I extended the installer form, submit pipeline, and final config writer so the optional import reuses `LFS_JWT_SECRET`, `INTERNAL_TOKEN`, and `oauth2.JWT_SECRET` from the uploaded `app.ini` instead of generating new values, including a submit-time fallback that re-reads the uploaded file if the checkbox was enabled after the first auto-import.
- 3 - I finalized secret resolution for both direct values and `LFS_JWT_SECRET_URI` / `INTERNAL_TOKEN_URI` / `JWT_SECRET_URI` file-based references, and added regression coverage for direct imports, URI-based imports, the real `POST /import_app_ini` flow, and the persisted `app.ini` output.
2026-05-12 20:35:52 +00:00

1188 lines
44 KiB
Go

// Copyright 2014 The Gogs Authors. All rights reserved.
// Copyright 2021 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package install
import (
"bytes"
"errors"
"image"
_ "image/png"
"io"
"net/http"
"net/mail"
"net/url"
"os"
"os/exec"
"path/filepath"
"runtime"
"slices"
"strconv"
"strings"
"time"
"code.gitea.io/gitea/models/db"
db_install "code.gitea.io/gitea/models/db/install"
user_model "code.gitea.io/gitea/models/user"
"code.gitea.io/gitea/modules/auth/password/hash"
"code.gitea.io/gitea/modules/generate"
"code.gitea.io/gitea/modules/graceful"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/optional"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/templates"
"code.gitea.io/gitea/modules/timeutil"
"code.gitea.io/gitea/modules/typesniffer"
"code.gitea.io/gitea/modules/web"
"code.gitea.io/gitea/modules/web/middleware"
"code.gitea.io/gitea/routers/common"
auth_service "code.gitea.io/gitea/services/auth"
"code.gitea.io/gitea/services/context"
"code.gitea.io/gitea/services/forms"
"code.gitea.io/gitea/services/mailer"
"code.gitea.io/gitea/services/versioned_migration"
)
const (
// tplInstall template for installation page
tplInstall templates.TplName = "install"
tplPostInstall templates.TplName = "post-install"
registrationModeAdminOnly = "admin_only"
registrationModeLocalOnly = "local_only"
registrationModeExternalOnly = "external_only"
registrationModeLocalAndExternal = "local_and_external"
installBrandingMaxFileSize = int64(1 << 20)
installBrandingMinPNGEdge = 64
installAppINIImportMaxSize = int64(1 << 20)
)
type installBrandingAssetSpec struct {
FormField string
TargetName string
LabelKey string
MimeType string
}
var installBrandingAssetSpecs = []installBrandingAssetSpec{
{FormField: "logo_svg", TargetName: "logo.svg", LabelKey: "install.branding.logo_svg", MimeType: typesniffer.MimeTypeImageSvg},
{FormField: "logo_png", TargetName: "logo.png", LabelKey: "install.branding.logo_png", MimeType: "image/png"},
{FormField: "loading_png", TargetName: "loading.png", LabelKey: "install.branding.loading_png", MimeType: "image/png"},
{FormField: "favicon_svg", TargetName: "favicon.svg", LabelKey: "install.branding.favicon_svg", MimeType: typesniffer.MimeTypeImageSvg},
{FormField: "favicon_png", TargetName: "favicon.png", LabelKey: "install.branding.favicon_png", MimeType: "image/png"},
}
// getSupportedDbTypeNames returns a slice for supported database types and names. The slice is used to keep the order
func getSupportedDbTypeNames() (dbTypeNames []map[string]string) {
for _, t := range setting.SupportedDatabaseTypes {
dbTypeNames = append(dbTypeNames, map[string]string{"type": t, "name": setting.DatabaseTypeNames[t]})
}
return dbTypeNames
}
func resolveInstallDefaultLanguage(defaultLanguage string) string {
if slices.Contains(setting.Langs, defaultLanguage) {
return defaultLanguage
}
if len(setting.Langs) > 0 {
return setting.Langs[0]
}
return ""
}
func reorderInstallLanguages(defaultLanguage string) []string {
defaultLanguage = resolveInstallDefaultLanguage(defaultLanguage)
if defaultLanguage == "" {
return append([]string(nil), setting.Langs...)
}
langs := []string{defaultLanguage}
for _, lang := range setting.Langs {
if lang != defaultLanguage {
langs = append(langs, lang)
}
}
return langs
}
func resolveInstallRegistrationMode(disableRegistration, allowOnlyInternalRegistration, allowOnlyExternalRegistration bool) string {
switch {
case disableRegistration:
return registrationModeAdminOnly
case allowOnlyExternalRegistration:
return registrationModeExternalOnly
case allowOnlyInternalRegistration:
return registrationModeLocalOnly
default:
return registrationModeLocalAndExternal
}
}
func isLocalRegistrationMode(mode string) bool {
return mode == registrationModeLocalOnly || mode == registrationModeLocalAndExternal
}
func isExternalRegistrationMode(mode string) bool {
return mode == registrationModeExternalOnly || mode == registrationModeLocalAndExternal
}
func normalizeInstallRegistrationOptions(form *forms.InstallForm) {
if form.RegistrationMode == "" {
form.RegistrationMode = resolveInstallRegistrationMode(form.DisableRegistration, form.AllowOnlyInternalRegistration, form.AllowOnlyExternalRegistration)
}
switch form.RegistrationMode {
case registrationModeAdminOnly:
form.DisableRegistration = true
form.AllowOnlyInternalRegistration = false
form.AllowOnlyExternalRegistration = false
case registrationModeLocalOnly:
form.DisableRegistration = false
form.AllowOnlyInternalRegistration = true
form.AllowOnlyExternalRegistration = false
case registrationModeExternalOnly:
form.DisableRegistration = false
form.AllowOnlyInternalRegistration = false
form.AllowOnlyExternalRegistration = true
default:
form.RegistrationMode = registrationModeLocalAndExternal
form.DisableRegistration = false
form.AllowOnlyInternalRegistration = false
form.AllowOnlyExternalRegistration = false
}
if form.AdminCreatedAccountMode == "" {
if form.RegistrationMode == registrationModeAdminOnly {
form.AdminCreatedAccountMode = setting.AdminCreatedAccountModeInvite
} else {
form.AdminCreatedAccountMode = setting.AdminCreatedAccountModeLocal
}
}
if form.AdminCreatedAccountMode != setting.AdminCreatedAccountModeInvite {
form.AdminCreatedAccountMode = setting.AdminCreatedAccountModeLocal
}
if !isLocalRegistrationMode(form.RegistrationMode) {
form.RegisterConfirm = false
form.RegisterManualConfirm = false
form.EnableCaptcha = false
}
if form.RegisterManualConfirm {
form.RegisterConfirm = true
}
if !isExternalRegistrationMode(form.RegistrationMode) {
form.EnableOpenIDSignIn = false
form.EnableOpenIDSignUp = false
}
if form.EnableOpenIDSignUp {
form.EnableOpenIDSignIn = true
}
}
func collectInstallBrandingAssets(ctx *context.Context, useSharedAssets bool) (map[string][]byte, error) {
uploads := make(map[string][]byte)
for _, spec := range installBrandingAssetSpecs {
file, header, err := ctx.Req.FormFile(spec.FormField)
if errors.Is(err, http.ErrMissingFile) {
continue
}
if err != nil {
return nil, errors.New(string(ctx.Tr("install.branding.upload_read_failed", ctx.Tr(spec.LabelKey), err)))
}
data, err := func() ([]byte, error) {
defer file.Close()
if header.Size > installBrandingMaxFileSize {
return nil, errors.New(string(ctx.Tr("install.branding.upload_too_big", ctx.Tr(spec.LabelKey), installBrandingMaxFileSize/1024)))
}
data, err := io.ReadAll(io.LimitReader(file, installBrandingMaxFileSize+1))
if err != nil {
return nil, errors.New(string(ctx.Tr("install.branding.upload_read_failed", ctx.Tr(spec.LabelKey), err)))
}
if int64(len(data)) > installBrandingMaxFileSize {
return nil, errors.New(string(ctx.Tr("install.branding.upload_too_big", ctx.Tr(spec.LabelKey), installBrandingMaxFileSize/1024)))
}
return data, nil
}()
if err != nil {
return nil, err
}
sniffedType := typesniffer.DetectContentType(data)
switch spec.MimeType {
case typesniffer.MimeTypeImageSvg:
if !sniffedType.IsSvgImage() {
return nil, errors.New(string(ctx.Tr("install.branding.upload_invalid_type", ctx.Tr(spec.LabelKey), "SVG")))
}
case "image/png":
if sniffedType.GetMimeType() != "image/png" {
return nil, errors.New(string(ctx.Tr("install.branding.upload_invalid_type", ctx.Tr(spec.LabelKey), "PNG")))
}
cfg, format, err := image.DecodeConfig(bytes.NewReader(data))
if err != nil || format != "png" {
return nil, errors.New(string(ctx.Tr("install.branding.upload_invalid_type", ctx.Tr(spec.LabelKey), "PNG")))
}
if cfg.Width != cfg.Height {
return nil, errors.New(string(ctx.Tr("install.branding.upload_png_square", ctx.Tr(spec.LabelKey))))
}
if cfg.Width < installBrandingMinPNGEdge || cfg.Height < installBrandingMinPNGEdge {
return nil, errors.New(string(ctx.Tr("install.branding.upload_png_too_small", ctx.Tr(spec.LabelKey), installBrandingMinPNGEdge, installBrandingMinPNGEdge)))
}
}
uploads[spec.TargetName] = data
}
if useSharedAssets {
if logoSVG, ok := uploads["logo.svg"]; ok {
uploads["favicon.svg"] = logoSVG
}
if logoPNG, ok := uploads["logo.png"]; ok {
uploads["favicon.png"] = logoPNG
}
}
return uploads, nil
}
func saveInstallBrandingAssets(uploads map[string][]byte) error {
if len(uploads) == 0 {
return nil
}
customImgPath := filepath.Join(setting.CustomPath, "public", "assets", "img")
if err := os.MkdirAll(customImgPath, os.ModePerm); err != nil {
return err
}
for targetName, data := range uploads {
if err := os.WriteFile(filepath.Join(customImgPath, targetName), data, 0o644); err != nil {
return err
}
}
if logoSVG, ok := uploads["logo.svg"]; ok {
if err := os.WriteFile(filepath.Join(customImgPath, "gitea.svg"), logoSVG, 0o644); err != nil {
return err
}
}
return nil
}
func getInstallProgressLogo() string {
customImgPath := filepath.Join(setting.CustomPath, "public", "assets", "img")
if _, err := os.Stat(filepath.Join(customImgPath, "loading.png")); err == nil {
return "loading.png"
}
if _, err := os.Stat(filepath.Join(customImgPath, "logo.svg")); err == nil {
return "logo.svg"
}
if _, err := os.Stat(filepath.Join(customImgPath, "logo.png")); err == nil {
return "logo.png"
}
return "loading.png"
}
func installContexter() func(next http.Handler) http.Handler {
return context.ContexterInstallPage(map[string]any{
"DbTypeNames": getSupportedDbTypeNames(),
"AppINIImportMaxSizeKB": installAppINIImportMaxSize / 1024,
"BrandingMaxFileSizeKB": installBrandingMaxFileSize / 1024,
"BrandingMinPNGEdge": installBrandingMinPNGEdge,
"EnvConfigKeys": setting.CollectEnvConfigKeys(),
"CustomConfFile": setting.CustomConf,
"PasswordHashAlgorithms": hash.RecommendedHashAlgorithms,
})
}
func newInstallFormFromSettings() (forms.InstallForm, string) {
form := forms.InstallForm{}
form.BrandingUseSharedAssets = true
// Database settings
form.DbHost = setting.Database.Host
form.DbUser = setting.Database.User
form.DbPasswd = setting.Database.Passwd
form.DbName = setting.Database.Name
form.DbPath = setting.Database.Path
form.DbSchema = setting.Database.Schema
form.SSLMode = setting.Database.SSLMode
curDBType := setting.Database.Type.String()
if !slices.Contains(setting.SupportedDatabaseTypes, curDBType) {
curDBType = "mysql"
}
// Application general settings
form.AppName = setting.AppName
form.DefaultLanguage = resolveInstallDefaultLanguage("")
form.RepoRootPath = setting.RepoRootPath
form.LFSRootPath = setting.LFS.Storage.Path
form.RunUser = setting.RunUser
form.Domain = setting.Domain
form.SSHPort = setting.SSH.Port
form.HTTPPort = setting.HTTPPort
form.AppURL = setting.AppURL
form.LogRootPath = setting.Log.RootPath
// E-mail service settings
if setting.MailService != nil {
form.SMTPAddr = setting.MailService.SMTPAddr
form.SMTPPort = setting.MailService.SMTPPort
form.SMTPFrom = setting.MailService.From
form.SMTPUser = setting.MailService.User
form.SMTPPasswd = setting.MailService.Passwd
}
form.RegisterConfirm = setting.Service.RegisterEmailConfirm
form.RegisterManualConfirm = setting.Service.RegisterManualConfirm
form.MailNotify = setting.Service.EnableNotifyMail
if setting.MailService == nil && !setting.Service.EnableNotifyMail {
form.MailNotify = true
}
form.RegistrationMode = resolveInstallRegistrationMode(setting.Service.DisableRegistration, setting.Service.AllowOnlyInternalRegistration, setting.Service.AllowOnlyExternalRegistration)
form.AdminCreatedAccountMode = setting.Service.AdminCreatedAccountMode
form.EnableOpenIDSignIn = setting.Service.EnableOpenIDSignIn
form.EnableOpenIDSignUp = setting.Service.EnableOpenIDSignUp
form.DisableRegistration = setting.Service.DisableRegistration
form.AllowOnlyInternalRegistration = setting.Service.AllowOnlyInternalRegistration
form.AllowOnlyExternalRegistration = setting.Service.AllowOnlyExternalRegistration
form.EnableCaptcha = setting.Service.EnableCaptcha
form.RequireSignInView = setting.Service.RequireSignInViewStrict
form.DefaultKeepEmailPrivate = setting.Service.DefaultKeepEmailPrivate
form.DefaultAllowCreateOrganization = setting.Service.DefaultAllowCreateOrganization
form.DefaultEnableTimetracking = setting.Service.DefaultEnableTimetracking
form.NoReplyAddress = setting.Service.NoReplyAddress
form.PasswordAlgorithm = hash.ConfigHashAlgorithm(setting.PasswordHashAlgo)
form.AdminManagementPolicy = setting.Admin.AdminManagementPolicy
normalizeInstallRegistrationOptions(&form)
return form, curDBType
}
func renderInstallPage(ctx *context.Context, form *forms.InstallForm, curDBType string) {
if form.DefaultLanguage == "" {
form.DefaultLanguage = resolveInstallDefaultLanguage("")
}
normalizeInstallRegistrationOptions(form)
ctx.Data["CurDbType"] = curDBType
middleware.AssignForm(form, ctx.Data)
ctx.HTML(http.StatusOK, tplInstall)
}
func importedDBType(dbType string, fallback string) string {
dbType = strings.TrimSpace(strings.ToLower(dbType))
if slices.Contains(setting.SupportedDatabaseTypes, dbType) {
return dbType
}
return fallback
}
func importedInt(sec setting.ConfigSection, key string, fallback int) int {
cfgKey := setting.ConfigSectionKey(sec, key)
if cfgKey == nil {
return fallback
}
return cfgKey.MustInt(fallback)
}
func importedFirstLang(value string, fallback string) string {
for _, lang := range strings.Split(value, ",") {
lang = strings.TrimSpace(lang)
if lang != "" {
return resolveInstallDefaultLanguage(lang)
}
}
return resolveInstallDefaultLanguage(fallback)
}
func populateInstallFormFromConfig(form *forms.InstallForm, cfg setting.ConfigProvider, fallbackDBType string) string {
rootSec := cfg.Section("")
dbSec := cfg.Section("database")
repoSec := cfg.Section("repository")
serverSec := cfg.Section("server")
lfsSec := cfg.Section("lfs")
logSec := cfg.Section("log")
mailerSec := cfg.Section("mailer")
serviceSec := cfg.Section("service")
openIDSec := cfg.Section("openid")
securitySec := cfg.Section("security")
adminSec := cfg.Section("admin")
i18nSec := cfg.Section("i18n")
form.AppName = setting.ConfigSectionKeyString(rootSec, "APP_NAME", form.AppName)
form.RunUser = setting.ConfigSectionKeyString(rootSec, "RUN_USER", form.RunUser)
form.DbType = importedDBType(setting.ConfigSectionKeyString(dbSec, "DB_TYPE", fallbackDBType), fallbackDBType)
form.DbHost = setting.ConfigSectionKeyString(dbSec, "HOST", form.DbHost)
form.DbUser = setting.ConfigSectionKeyString(dbSec, "USER", form.DbUser)
form.DbPasswd = setting.ConfigSectionKeyString(dbSec, "PASSWD", form.DbPasswd)
form.DbName = setting.ConfigSectionKeyString(dbSec, "NAME", form.DbName)
form.DbSchema = setting.ConfigSectionKeyString(dbSec, "SCHEMA", form.DbSchema)
form.SSLMode = setting.ConfigSectionKeyString(dbSec, "SSL_MODE", form.SSLMode)
form.DbPath = setting.ConfigSectionKeyString(dbSec, "PATH", form.DbPath)
form.DefaultLanguage = importedFirstLang(setting.ConfigSectionKeyString(i18nSec, "LANGS"), form.DefaultLanguage)
form.RepoRootPath = setting.ConfigSectionKeyString(repoSec, "ROOT", form.RepoRootPath)
form.LFSRootPath = setting.ConfigSectionKeyString(lfsSec, "PATH", form.LFSRootPath)
form.Domain = setting.ConfigSectionKeyString(serverSec, "DOMAIN", setting.ConfigSectionKeyString(serverSec, "SSH_DOMAIN", form.Domain))
form.SSHPort = importedInt(serverSec, "SSH_PORT", form.SSHPort)
if setting.ConfigSectionKeyBool(serverSec, "DISABLE_SSH") {
form.SSHPort = 0
}
form.HTTPPort = setting.ConfigSectionKeyString(serverSec, "HTTP_PORT", form.HTTPPort)
form.AppURL = setting.ConfigSectionKeyString(serverSec, "ROOT_URL", form.AppURL)
form.LogRootPath = setting.ConfigSectionKeyString(logSec, "ROOT_PATH", form.LogRootPath)
if setting.ConfigSectionKeyBool(mailerSec, "ENABLED") {
form.SMTPAddr = setting.ConfigSectionKeyString(mailerSec, "SMTP_ADDR", form.SMTPAddr)
form.SMTPPort = setting.ConfigSectionKeyString(mailerSec, "SMTP_PORT", form.SMTPPort)
form.SMTPFrom = setting.ConfigSectionKeyString(mailerSec, "FROM", form.SMTPFrom)
form.SMTPUser = setting.ConfigSectionKeyString(mailerSec, "USER", form.SMTPUser)
form.SMTPPasswd = setting.ConfigSectionKeyString(mailerSec, "PASSWD", form.SMTPPasswd)
}
form.RegisterConfirm = setting.ConfigSectionKeyBool(serviceSec, "REGISTER_EMAIL_CONFIRM", form.RegisterConfirm)
form.RegisterManualConfirm = setting.ConfigSectionKeyBool(serviceSec, "REGISTER_MANUAL_CONFIRM", form.RegisterManualConfirm)
form.MailNotify = setting.ConfigSectionKeyBool(serviceSec, "ENABLE_NOTIFY_MAIL", form.MailNotify)
form.AdminCreatedAccountMode = setting.ConfigSectionKeyString(serviceSec, "ADMIN_CREATED_ACCOUNT_MODE", form.AdminCreatedAccountMode)
form.DisableRegistration = setting.ConfigSectionKeyBool(serviceSec, "DISABLE_REGISTRATION", form.DisableRegistration)
form.AllowOnlyInternalRegistration = setting.ConfigSectionKeyBool(serviceSec, "ALLOW_ONLY_INTERNAL_REGISTRATION", form.AllowOnlyInternalRegistration)
form.AllowOnlyExternalRegistration = setting.ConfigSectionKeyBool(serviceSec, "ALLOW_ONLY_EXTERNAL_REGISTRATION", form.AllowOnlyExternalRegistration)
form.RegistrationMode = ""
form.EnableCaptcha = setting.ConfigSectionKeyBool(serviceSec, "ENABLE_CAPTCHA", form.EnableCaptcha)
form.RequireSignInView = setting.ConfigSectionKeyBool(serviceSec, "REQUIRE_SIGNIN_VIEW", form.RequireSignInView)
form.DefaultKeepEmailPrivate = setting.ConfigSectionKeyBool(serviceSec, "DEFAULT_KEEP_EMAIL_PRIVATE", form.DefaultKeepEmailPrivate)
form.DefaultAllowCreateOrganization = setting.ConfigSectionKeyBool(serviceSec, "DEFAULT_ALLOW_CREATE_ORGANIZATION", form.DefaultAllowCreateOrganization)
form.DefaultEnableTimetracking = setting.ConfigSectionKeyBool(serviceSec, "DEFAULT_ENABLE_TIMETRACKING", form.DefaultEnableTimetracking)
form.NoReplyAddress = setting.ConfigSectionKeyString(serviceSec, "NO_REPLY_ADDRESS", form.NoReplyAddress)
form.EnableOpenIDSignIn = setting.ConfigSectionKeyBool(openIDSec, "ENABLE_OPENID_SIGNIN", form.EnableOpenIDSignIn)
form.EnableOpenIDSignUp = setting.ConfigSectionKeyBool(openIDSec, "ENABLE_OPENID_SIGNUP", form.EnableOpenIDSignUp)
form.EnableUpdateChecker = setting.ConfigSectionKeyBool(securitySec, "ENABLE_UPDATE_CHECKER", form.EnableUpdateChecker)
form.PasswordAlgorithm = setting.ConfigSectionKeyString(securitySec, "PASSWORD_HASH_ALGO", form.PasswordAlgorithm)
form.AdminManagementPolicy = setting.ConfigSectionKeyString(adminSec, "ADMIN_MANAGEMENT_POLICY", form.AdminManagementPolicy)
if form.ImportSensitiveSecrets {
form.ImportedLFSJWTSecret = readImportedSecretValue(serverSec, "LFS_JWT_SECRET_URI", "LFS_JWT_SECRET")
form.ImportedInternalToken = readImportedSecretValue(securitySec, "INTERNAL_TOKEN_URI", "INTERNAL_TOKEN")
form.ImportedOAuth2JWTSecret = readImportedSecretValue(cfg.Section("oauth2"), "JWT_SECRET_URI", "JWT_SECRET")
}
normalizeInstallRegistrationOptions(form)
return form.DbType
}
func readImportedSecretValue(sec setting.ConfigSection, uriKey, verbatimKey string) string {
verbatim := setting.ConfigSectionKeyString(sec, verbatimKey)
if verbatim != "" {
return verbatim
}
uriValue := setting.ConfigSectionKeyString(sec, uriKey)
if uriValue == "" {
return ""
}
parsed, err := url.Parse(uriValue)
if err != nil || parsed.Scheme != "file" {
return ""
}
path := parsed.Path
if parsed.Opaque != "" {
path = parsed.Opaque
}
if parsed.Host != "" {
switch {
case len(parsed.Host) == 2 && parsed.Host[1] == ':':
path = parsed.Host + path
default:
path = "//" + parsed.Host + path
}
}
path, err = url.PathUnescape(path)
if err != nil || path == "" {
return ""
}
if runtime.GOOS == "windows" && strings.HasPrefix(path, "/") && len(path) > 2 && path[2] == ':' {
path = path[1:]
}
data, err := os.ReadFile(path)
if err != nil {
return ""
}
return strings.TrimSpace(string(data))
}
func readImportedInstallConfig(ctx *context.Context) (setting.ConfigProvider, error) {
file, header, err := ctx.Req.FormFile("app_ini_file")
if errors.Is(err, http.ErrMissingFile) {
return nil, errors.New(string(ctx.Tr("install.import_app_ini_missing")))
}
if err != nil {
return nil, errors.New(string(ctx.Tr("install.import_app_ini_read_failed", err)))
}
defer file.Close()
if header.Size > installAppINIImportMaxSize {
return nil, errors.New(string(ctx.Tr("install.import_app_ini_too_big", installAppINIImportMaxSize/1024)))
}
data, err := io.ReadAll(io.LimitReader(file, installAppINIImportMaxSize+1))
if err != nil {
return nil, errors.New(string(ctx.Tr("install.import_app_ini_read_failed", err)))
}
if int64(len(data)) > installAppINIImportMaxSize {
return nil, errors.New(string(ctx.Tr("install.import_app_ini_too_big", installAppINIImportMaxSize/1024)))
}
cfg, err := setting.NewConfigProviderFromData(string(data))
if err != nil {
return nil, errors.New(string(ctx.Tr("install.import_app_ini_invalid", err)))
}
return cfg, nil
}
func populateInstallSensitiveSecretsFromConfig(form *forms.InstallForm, cfg setting.ConfigProvider) {
if !form.ImportSensitiveSecrets {
return
}
if form.ImportedLFSJWTSecret == "" {
form.ImportedLFSJWTSecret = readImportedSecretValue(cfg.Section("server"), "LFS_JWT_SECRET_URI", "LFS_JWT_SECRET")
}
if form.ImportedInternalToken == "" {
form.ImportedInternalToken = readImportedSecretValue(cfg.Section("security"), "INTERNAL_TOKEN_URI", "INTERNAL_TOKEN")
}
if form.ImportedOAuth2JWTSecret == "" {
form.ImportedOAuth2JWTSecret = readImportedSecretValue(cfg.Section("oauth2"), "JWT_SECRET_URI", "JWT_SECRET")
}
}
func applyInstallSensitiveSecretsToConfig(cfg setting.ConfigProvider, form *forms.InstallForm) error {
if form.LFSRootPath != "" {
if form.ImportSensitiveSecrets && form.ImportedLFSJWTSecret != "" {
cfg.Section("server").DeleteKey("LFS_JWT_SECRET_URI")
cfg.Section("server").Key("LFS_JWT_SECRET").SetValue(form.ImportedLFSJWTSecret)
} else if !cfg.Section("server").HasKey("LFS_JWT_SECRET_URI") {
_, lfsJwtSecret := generate.NewJwtSecretWithBase64()
cfg.Section("server").Key("LFS_JWT_SECRET").SetValue(lfsJwtSecret)
}
}
// the internal token could be read from INTERNAL_TOKEN or INTERNAL_TOKEN_URI (the file is guaranteed to be non-empty)
// if there is no InternalToken, generate one and save to security.INTERNAL_TOKEN
if form.ImportSensitiveSecrets && form.ImportedInternalToken != "" {
cfg.Section("security").DeleteKey("INTERNAL_TOKEN_URI")
cfg.Section("security").Key("INTERNAL_TOKEN").SetValue(form.ImportedInternalToken)
} else if setting.InternalToken == "" {
internalToken, err := generate.NewInternalToken()
if err != nil {
return err
}
cfg.Section("security").Key("INTERNAL_TOKEN").SetValue(internalToken)
}
// FIXME: at the moment, no matter oauth2 is enabled or not, it must generate a "oauth2 JWT_SECRET"
// see the "loadOAuth2From" in "setting/oauth2.go"
if form.ImportSensitiveSecrets && form.ImportedOAuth2JWTSecret != "" {
cfg.Section("oauth2").DeleteKey("JWT_SECRET_URI")
cfg.Section("oauth2").Key("JWT_SECRET").SetValue(form.ImportedOAuth2JWTSecret)
} else if !cfg.Section("oauth2").HasKey("JWT_SECRET") && !cfg.Section("oauth2").HasKey("JWT_SECRET_URI") {
_, jwtSecretBase64 := generate.NewJwtSecretWithBase64()
cfg.Section("oauth2").Key("JWT_SECRET").SetValue(jwtSecretBase64)
}
return nil
}
// Install render installation page
func Install(ctx *context.Context) {
if setting.InstallLock {
InstallDone(ctx)
return
}
form, curDBType := newInstallFormFromSettings()
renderInstallPage(ctx, &form, curDBType)
}
func ImportAppINI(ctx *context.Context) {
if setting.InstallLock {
InstallDone(ctx)
return
}
form, curDBType := newInstallFormFromSettings()
form.ImportSensitiveSecrets = ctx.FormBool("import_sensitive_secrets")
cfg, err := readImportedInstallConfig(ctx)
if err != nil {
ctx.Data["CurDbType"] = curDBType
ctx.RenderWithErrDeprecated(err.Error(), tplInstall, &form)
return
}
curDBType = populateInstallFormFromConfig(&form, cfg, curDBType)
ctx.Flash.Success(ctx.Tr("install.import_app_ini_success"), true)
renderInstallPage(ctx, &form, curDBType)
}
// TestMail checks the mail settings entered on the install page.
func TestMail(ctx *context.Context) {
email := strings.TrimSpace(ctx.FormString("test_mail_email"))
if email == "" {
email = strings.TrimSpace(ctx.FormString("admin_email"))
}
state := "success"
message := string(ctx.Tr("admin.config.test_mail_sent", email))
ok := true
mailService, err := buildInstallTestMailService(ctx)
if err != nil {
state = "failed"
message = err.Error()
ok = false
} else if err := mailer.SendTestMailWith(email, mailService); err != nil {
state = "failed"
message = string(ctx.Tr("admin.config.test_mail_failed", email, mailer.ShortTestMailError(err)))
ok = false
}
ctx.JSON(http.StatusOK, map[string]any{
"ok": ok,
"state": state,
"message": message,
"email": email,
})
}
func buildInstallTestMailService(ctx *context.Context) (*setting.Mailer, error) {
smtpAddr := strings.TrimSpace(ctx.FormString("smtp_addr"))
smtpPort := strings.TrimSpace(ctx.FormString("smtp_port"))
smtpFrom := strings.TrimSpace(ctx.FormString("smtp_from"))
if smtpAddr == "" {
return nil, errors.New(string(ctx.Tr("install.test_mail_missing_host")))
}
email := strings.TrimSpace(ctx.FormString("test_mail_email"))
if email == "" {
email = strings.TrimSpace(ctx.FormString("admin_email"))
}
if email == "" {
return nil, errors.New(string(ctx.Tr("install.test_mail_missing_recipient")))
}
if _, err := mail.ParseAddress(email); err != nil {
return nil, errors.New(string(ctx.Tr("form.email_invalid")))
}
parsedFrom, err := mail.ParseAddress(smtpFrom)
if err != nil {
return nil, errors.New(string(ctx.Tr("install.smtp_from_invalid")))
}
protocol := inferInstallMailProtocol(smtpAddr, smtpPort)
if smtpPort == "" {
switch protocol {
case "smtp":
smtpPort = "25"
case "smtp+starttls":
smtpPort = "587"
default:
smtpPort = "465"
}
}
return &setting.Mailer{
Name: setting.AppName,
From: smtpFrom,
FromName: parsedFrom.Name,
FromEmail: parsedFrom.Address,
Protocol: protocol,
SMTPAddr: smtpAddr,
SMTPPort: smtpPort,
User: strings.TrimSpace(ctx.FormString("smtp_user")),
Passwd: ctx.FormString("smtp_passwd"),
EnableHelo: true,
OverrideHeader: map[string][]string{},
}, nil
}
func inferInstallMailProtocol(smtpAddr, smtpPort string) string {
if strings.ContainsAny(smtpAddr, "/\\") {
return "smtp+unix"
}
switch smtpPort {
case "25":
return "smtp"
case "587":
return "smtp+starttls"
default:
return "smtps"
}
}
func checkDatabase(ctx *context.Context, form *forms.InstallForm) bool {
var err error
if (setting.Database.Type == "sqlite3") &&
len(setting.Database.Path) == 0 {
ctx.Data["Err_DbPath"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.err_empty_db_path"), tplInstall, form)
return false
}
// Check if the user is trying to re-install in an installed database
db.UnsetDefaultEngine()
defer db.UnsetDefaultEngine()
if err = db.InitEngine(ctx); err != nil {
if strings.Contains(err.Error(), `Unknown database type: sqlite3`) {
ctx.Data["Err_DbType"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.sqlite3_not_available", "https://docs.gitea.com/installation/install-from-binary"), tplInstall, form)
} else {
ctx.Data["Err_DbSetting"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_db_setting", err), tplInstall, form)
}
return false
}
err = db_install.CheckDatabaseConnection(ctx)
if err != nil {
ctx.Data["Err_DbSetting"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_db_setting", err), tplInstall, form)
return false
}
hasPostInstallationUser, err := db_install.HasPostInstallationUsers(ctx)
if err != nil {
ctx.Data["Err_DbSetting"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_db_table", "user", err), tplInstall, form)
return false
}
dbMigrationVersion, err := db_install.GetMigrationVersion(ctx)
if err != nil {
ctx.Data["Err_DbSetting"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_db_table", "version", err), tplInstall, form)
return false
}
if hasPostInstallationUser && dbMigrationVersion > 0 {
log.Error("The database is likely to have been used by Gitea before, database migration version=%d", dbMigrationVersion)
confirmed := form.ReinstallConfirmFirst && form.ReinstallConfirmSecond && form.ReinstallConfirmThird
if !confirmed {
ctx.Data["Err_DbInstalledBefore"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.reinstall_error"), tplInstall, form)
return false
}
log.Info("User confirmed re-installation of Gitea into a pre-existing database")
}
if hasPostInstallationUser || dbMigrationVersion > 0 {
log.Info("Gitea will be installed in a database with: hasPostInstallationUser=%v, dbMigrationVersion=%v", hasPostInstallationUser, dbMigrationVersion)
}
return true
}
// SubmitInstall response for submit install items
func SubmitInstall(ctx *context.Context) {
if setting.InstallLock {
InstallDone(ctx)
return
}
var err error
form := *web.GetForm(ctx).(*forms.InstallForm)
form.ImportSensitiveSecrets = ctx.FormBool("import_sensitive_secrets")
form.ImportedLFSJWTSecret = ctx.FormString("imported_lfs_jwt_secret")
form.ImportedInternalToken = ctx.FormString("imported_internal_token")
form.ImportedOAuth2JWTSecret = ctx.FormString("imported_o_auth2_jwt_secret")
if form.ImportSensitiveSecrets && (form.ImportedLFSJWTSecret == "" || form.ImportedInternalToken == "" || form.ImportedOAuth2JWTSecret == "") {
if importedCfg, err := readImportedInstallConfig(ctx); err == nil {
populateInstallSensitiveSecretsFromConfig(&form, importedCfg)
}
}
// fix form values
if form.AppURL != "" && form.AppURL[len(form.AppURL)-1] != '/' {
form.AppURL += "/"
}
if form.AdminManagementPolicy == "" {
form.AdminManagementPolicy = setting.AdminManagementPolicyGrantorOnly
}
form.DefaultLanguage = strings.TrimSpace(form.DefaultLanguage)
normalizeInstallRegistrationOptions(&form)
if form.DefaultLanguage == "" {
form.DefaultLanguage = resolveInstallDefaultLanguage("")
} else if !slices.Contains(setting.Langs, form.DefaultLanguage) {
ctx.Data["Err_DefaultLanguage"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("form.lang_select_error"), tplInstall, &form)
return
}
ctx.Data["CurDbType"] = form.DbType
if ctx.HasError() {
ctx.Data["Err_SMTP"] = ctx.Data["Err_SMTPUser"] != nil
ctx.Data["Err_Registration"] = ctx.Data["Err_RegistrationMode"] != nil
ctx.Data["Err_Admin"] = ctx.Data["Err_AdminName"] != nil || ctx.Data["Err_AdminPasswd"] != nil || ctx.Data["Err_AdminEmail"] != nil
ctx.HTML(http.StatusOK, tplInstall)
return
}
if _, err = exec.LookPath("git"); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.test_git_failed", err), tplInstall, &form)
return
}
// ---- Basic checks are passed, now test configuration.
// Test database setting.
setting.Database.Type = setting.DatabaseType(form.DbType)
setting.Database.Host = form.DbHost
setting.Database.User = form.DbUser
setting.Database.Passwd = form.DbPasswd
setting.Database.Name = form.DbName
setting.Database.Schema = form.DbSchema
setting.Database.SSLMode = form.SSLMode
setting.Database.Path = form.DbPath
setting.Database.LogSQL = !setting.IsProd
if !checkDatabase(ctx, &form) {
return
}
// Prepare AppDataPath, it is very important for Gitea
if err = setting.PrepareAppDataPath(); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_app_data_path", err), tplInstall, &form)
return
}
// Test repository root path.
form.RepoRootPath = strings.ReplaceAll(form.RepoRootPath, "\\", "/")
if err = os.MkdirAll(form.RepoRootPath, os.ModePerm); err != nil {
ctx.Data["Err_RepoRootPath"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_repo_path", err), tplInstall, &form)
return
}
// Test LFS root path if not empty, empty meaning disable LFS
if form.LFSRootPath != "" {
form.LFSRootPath = strings.ReplaceAll(form.LFSRootPath, "\\", "/")
if err := os.MkdirAll(form.LFSRootPath, os.ModePerm); err != nil {
ctx.Data["Err_LFSRootPath"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_lfs_path", err), tplInstall, &form)
return
}
}
// Test log root path.
form.LogRootPath = strings.ReplaceAll(form.LogRootPath, "\\", "/")
if err = os.MkdirAll(form.LogRootPath, os.ModePerm); err != nil {
ctx.Data["Err_LogRootPath"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_log_root_path", err), tplInstall, &form)
return
}
// Check logic loophole between disable self-registration and no admin account.
if form.DisableRegistration && len(form.AdminName) == 0 {
ctx.Data["Err_Registration"] = true
ctx.Data["Err_Admin"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.no_admin_and_disable_registration"), tplInstall, form)
return
}
// Check admin user creation
if len(form.AdminName) > 0 {
// Ensure AdminName is valid
if err := user_model.IsUsableUsername(form.AdminName); err != nil {
ctx.Data["Err_Admin"] = true
ctx.Data["Err_AdminName"] = true
if db.IsErrNameReserved(err) {
ctx.RenderWithErrDeprecated(ctx.Tr("install.err_admin_name_is_reserved"), tplInstall, form)
return
} else if db.IsErrNamePatternNotAllowed(err) {
ctx.RenderWithErrDeprecated(ctx.Tr("install.err_admin_name_pattern_not_allowed"), tplInstall, form)
return
}
ctx.RenderWithErrDeprecated(ctx.Tr("install.err_admin_name_is_invalid"), tplInstall, form)
return
}
// Check Admin email
if len(form.AdminEmail) == 0 {
ctx.Data["Err_Admin"] = true
ctx.Data["Err_AdminEmail"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.err_empty_admin_email"), tplInstall, form)
return
}
// Check admin password.
if len(form.AdminPasswd) == 0 {
ctx.Data["Err_Admin"] = true
ctx.Data["Err_AdminPasswd"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.err_empty_admin_password"), tplInstall, form)
return
}
if form.AdminPasswd != form.AdminConfirmPasswd {
ctx.Data["Err_Admin"] = true
ctx.Data["Err_AdminPasswd"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("form.password_not_match"), tplInstall, form)
return
}
}
brandingUploads, err := collectInstallBrandingAssets(ctx, form.BrandingUseSharedAssets)
if err != nil {
ctx.Data["Err_Branding"] = true
ctx.RenderWithErrDeprecated(err.Error(), tplInstall, form)
return
}
// Init the engine with migration
if err = db.InitEngineWithMigration(ctx, versioned_migration.Migrate); err != nil {
db.UnsetDefaultEngine()
ctx.Data["Err_DbSetting"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_db_setting", err), tplInstall, &form)
return
}
// Save settings.
cfg, err := setting.NewConfigProviderFromFile(setting.CustomConf)
if err != nil {
log.Error("Failed to load custom conf '%s': %v", setting.CustomConf, err)
}
cfg.Section("").Key("APP_NAME").SetValue(form.AppName)
cfg.Section("").Key("RUN_USER").SetValue(form.RunUser)
cfg.Section("").Key("WORK_PATH").SetValue(setting.AppWorkPath)
cfg.Section("").Key("RUN_MODE").SetValue("prod")
cfg.Section("database").Key("DB_TYPE").SetValue(setting.Database.Type.String())
cfg.Section("database").Key("HOST").SetValue(setting.Database.Host)
cfg.Section("database").Key("NAME").SetValue(setting.Database.Name)
cfg.Section("database").Key("USER").SetValue(setting.Database.User)
cfg.Section("database").Key("PASSWD").SetValue(setting.Database.Passwd)
cfg.Section("database").Key("SCHEMA").SetValue(setting.Database.Schema)
cfg.Section("database").Key("SSL_MODE").SetValue(setting.Database.SSLMode)
cfg.Section("database").Key("PATH").SetValue(setting.Database.Path)
cfg.Section("database").Key("LOG_SQL").SetValue("false") // LOG_SQL is rarely helpful
cfg.Section("repository").Key("ROOT").SetValue(form.RepoRootPath)
cfg.Section("server").Key("SSH_DOMAIN").SetValue(form.Domain)
cfg.Section("server").Key("DOMAIN").SetValue(form.Domain)
cfg.Section("server").Key("HTTP_PORT").SetValue(form.HTTPPort)
cfg.Section("server").Key("ROOT_URL").SetValue(form.AppURL)
cfg.Section("server").Key("APP_DATA_PATH").SetValue(setting.AppDataPath)
if form.SSHPort == 0 {
cfg.Section("server").Key("DISABLE_SSH").SetValue("true")
} else {
cfg.Section("server").Key("DISABLE_SSH").SetValue("false")
cfg.Section("server").Key("SSH_PORT").SetValue(strconv.Itoa(form.SSHPort))
}
if form.LFSRootPath != "" {
cfg.Section("server").Key("LFS_START_SERVER").SetValue("true")
cfg.Section("lfs").Key("PATH").SetValue(form.LFSRootPath)
} else {
cfg.Section("server").Key("LFS_START_SERVER").SetValue("false")
}
if len(strings.TrimSpace(form.SMTPAddr)) > 0 {
if _, err := mail.ParseAddress(form.SMTPFrom); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.smtp_from_invalid"), tplInstall, &form)
return
}
cfg.Section("mailer").Key("ENABLED").SetValue("true")
cfg.Section("mailer").Key("SMTP_ADDR").SetValue(form.SMTPAddr)
cfg.Section("mailer").Key("SMTP_PORT").SetValue(form.SMTPPort)
cfg.Section("mailer").Key("FROM").SetValue(form.SMTPFrom)
cfg.Section("mailer").Key("USER").SetValue(form.SMTPUser)
cfg.Section("mailer").Key("PASSWD").SetValue(form.SMTPPasswd)
} else {
cfg.Section("mailer").Key("ENABLED").SetValue("false")
}
registerEmailConfirm := form.RegisterConfirm && !form.RegisterManualConfirm && isLocalRegistrationMode(form.RegistrationMode)
registerManualConfirm := form.RegisterManualConfirm && isLocalRegistrationMode(form.RegistrationMode)
cfg.Section("service").Key("REGISTER_EMAIL_CONFIRM").SetValue(strconv.FormatBool(registerEmailConfirm))
cfg.Section("service").Key("REGISTER_MANUAL_CONFIRM").SetValue(strconv.FormatBool(registerManualConfirm))
cfg.Section("service").Key("ENABLE_NOTIFY_MAIL").SetValue(strconv.FormatBool(form.MailNotify))
cfg.Section("service").Key("ADMIN_CREATED_ACCOUNT_MODE").SetValue(form.AdminCreatedAccountMode)
cfg.Section("openid").Key("ENABLE_OPENID_SIGNIN").SetValue(strconv.FormatBool(form.EnableOpenIDSignIn))
cfg.Section("openid").Key("ENABLE_OPENID_SIGNUP").SetValue(strconv.FormatBool(form.EnableOpenIDSignUp))
cfg.Section("service").Key("DISABLE_REGISTRATION").SetValue(strconv.FormatBool(form.DisableRegistration))
cfg.Section("service").Key("ALLOW_ONLY_INTERNAL_REGISTRATION").SetValue(strconv.FormatBool(form.AllowOnlyInternalRegistration))
cfg.Section("service").Key("ALLOW_ONLY_EXTERNAL_REGISTRATION").SetValue(strconv.FormatBool(form.AllowOnlyExternalRegistration))
cfg.Section("service").Key("ENABLE_CAPTCHA").SetValue(strconv.FormatBool(form.EnableCaptcha))
cfg.Section("service").Key("REQUIRE_SIGNIN_VIEW").SetValue(strconv.FormatBool(form.RequireSignInView))
cfg.Section("service").Key("DEFAULT_KEEP_EMAIL_PRIVATE").SetValue(strconv.FormatBool(form.DefaultKeepEmailPrivate))
cfg.Section("service").Key("DEFAULT_ALLOW_CREATE_ORGANIZATION").SetValue(strconv.FormatBool(form.DefaultAllowCreateOrganization))
cfg.Section("service").Key("DEFAULT_ENABLE_TIMETRACKING").SetValue(strconv.FormatBool(form.DefaultEnableTimetracking))
cfg.Section("service").Key("NO_REPLY_ADDRESS").SetValue(form.NoReplyAddress)
cfg.Section("cron.update_checker").Key("ENABLED").SetValue(strconv.FormatBool(form.EnableUpdateChecker))
cfg.Section("admin").Key("SUPER_ADMIN_ENABLED").SetValue("true")
cfg.Section("admin").Key("ADMIN_MANAGEMENT_POLICY").SetValue(form.AdminManagementPolicy)
cfg.Section("i18n").Key("LANGS").SetValue(strings.Join(reorderInstallLanguages(form.DefaultLanguage), ","))
cfg.Section("session").Key("PROVIDER").SetValue("file")
cfg.Section("log").Key("MODE").MustString("console")
cfg.Section("log").Key("LEVEL").SetValue(setting.Log.Level.String())
cfg.Section("log").Key("ROOT_PATH").SetValue(form.LogRootPath)
cfg.Section("repository.pull-request").Key("DEFAULT_MERGE_STYLE").SetValue("merge")
cfg.Section("repository.signing").Key("DEFAULT_TRUST_MODEL").SetValue("committer")
cfg.Section("security").Key("INSTALL_LOCK").SetValue("true")
if err = applyInstallSensitiveSecretsToConfig(cfg, &form); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.internal_token_failed", err), tplInstall, &form)
return
}
// if there is already a SECRET_KEY, we should not overwrite it, otherwise the encrypted data will not be able to be decrypted
if setting.SecretKey == "" {
var secretKey string
if secretKey, err = generate.NewSecretKey(); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.secret_key_failed", err), tplInstall, &form)
return
}
cfg.Section("security").Key("SECRET_KEY").SetValue(secretKey)
}
if len(form.PasswordAlgorithm) > 0 {
var algorithm *hash.PasswordHashAlgorithm
setting.PasswordHashAlgo, algorithm = hash.SetDefaultPasswordHashAlgorithm(form.PasswordAlgorithm)
if algorithm == nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_password_algorithm"), tplInstall, &form)
return
}
cfg.Section("security").Key("PASSWORD_HASH_ALGO").SetValue(form.PasswordAlgorithm)
}
log.Info("Save settings to custom config file %s", setting.CustomConf)
err = os.MkdirAll(filepath.Dir(setting.CustomConf), os.ModePerm)
if err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.save_config_failed", err), tplInstall, &form)
return
}
setting.EnvironmentToConfig(cfg, os.Environ())
if err = cfg.SaveTo(setting.CustomConf); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.save_config_failed", err), tplInstall, &form)
return
}
if err = saveInstallBrandingAssets(brandingUploads); err != nil {
ctx.Data["Err_Branding"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.branding.upload_save_failed", err), tplInstall, &form)
return
}
// unset default engine before reload database setting
db.UnsetDefaultEngine()
// ---- All checks are passed
// Reload settings (and re-initialize database connection)
setting.InitCfgProvider(setting.CustomConf)
setting.LoadCommonSettings()
setting.MustInstalled()
setting.LoadDBSetting()
if err := common.InitDBEngine(ctx); err != nil {
log.Fatal("ORM engine initialization failed: %v", err)
}
// Create admin account
if len(form.AdminName) > 0 {
u := &user_model.User{
Name: form.AdminName,
Email: form.AdminEmail,
Passwd: form.AdminPasswd,
IsAdmin: true,
}
overwriteDefault := &user_model.CreateUserOverwriteOptions{
IsRestricted: optional.Some(false),
IsActive: optional.Some(true),
}
if err = user_model.CreateUser(ctx, u, &user_model.Meta{}, overwriteDefault); err != nil {
if !user_model.IsErrUserAlreadyExist(err) {
setting.InstallLock = false
ctx.Data["Err_AdminName"] = true
ctx.Data["Err_AdminEmail"] = true
ctx.RenderWithErrDeprecated(ctx.Tr("install.invalid_admin_setting", err), tplInstall, &form)
return
}
log.Info("Admin account already exist")
u, _ = user_model.GetUserByName(ctx, u.Name)
}
nt, token, err := auth_service.CreateAuthTokenForUserID(ctx, u.ID)
if err != nil {
ctx.ServerError("CreateAuthTokenForUserID", err)
return
}
ctx.SetSiteCookie(setting.CookieRememberName, nt.ID+":"+token, setting.LogInRememberDays*timeutil.Day)
// Auto-login for admin
if err = ctx.Session.Set("uid", u.ID); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.save_config_failed", err), tplInstall, &form)
return
}
if err = ctx.Session.Set("uname", u.Name); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.save_config_failed", err), tplInstall, &form)
return
}
if err = ctx.Session.Release(); err != nil {
ctx.RenderWithErrDeprecated(ctx.Tr("install.save_config_failed", err), tplInstall, &form)
return
}
}
setting.ClearEnvConfigKeys()
log.Info("First-time run install finished!")
InstallDone(ctx)
go func() {
// Sleep for a while to make sure the user's browser has loaded the post-install page and its assets (images, css, js)
// What if this duration is not long enough? That's impossible -- if the user can't load the simple page in time, how could they install or use Gitea in the future ....
time.Sleep(3 * time.Second)
// Now get the http.Server from this request and shut it down
// NB: This is not our hammerable graceful shutdown this is http.Server.Shutdown
srv := ctx.Value(http.ServerContextKey).(*http.Server)
if err := srv.Shutdown(graceful.GetManager().HammerContext()); err != nil {
log.Error("Unable to shutdown the install server! Error: %v", err)
}
// After the HTTP server for "install" shuts down, the `runWeb()` will continue to run the "normal" server
}()
}
// InstallDone shows the "post-install" page, makes it easier to develop the page.
// The name is not called as "PostInstall" to avoid misinterpretation as a handler for "POST /install"
func InstallDone(ctx *context.Context) { //nolint:revive // export stutter
hasUsers, _ := user_model.HasUsers(ctx)
ctx.Data["IsAccountCreated"] = hasUsers.HasAnyUser
ctx.Data["InstallProgressLogo"] = getInstallProgressLogo()
ctx.HTML(http.StatusOK, tplPostInstall)
}