mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-26 20:11:02 +00:00 
			
		
		
		
	FORGEJO_* environment variables are set to the corresponding GITEA_* variable when the cli starts. This approach is intended to minimize the conflicts on rebase. All occurences of GITEA_* are left untouched in the codebase and they are only changed to FORGEJO_* if exposed to the user. (cherry picked from commite466f9d10e) (cherry picked from commite33e95931b) (cherry picked from commit2cfc6519b7) (cherry picked from commitaf8864373a) (cherry picked from commita0550ff339) (cherry picked from commit24dc0a5191) (cherry picked from commite255eea2b4) (cherry picked from commit0c4f5afa7a) (cherry picked from commit42fce708d0) (cherry picked from commite7278c3c22) (cherry picked from commit0fb9ed7e0e) (cherry picked from commita98308aa4d) (cherry picked from commitb8695fcbe0) (cherry picked from commit4aee8719f5) (cherry picked from commit1c503c1ba7) (cherry picked from commitcf1ed8551e) (cherry picked from commitc52459b088) (cherry picked from commit92cac277b2) (cherry picked from commit2c744f1118) (cherry picked from commit989a98a8c5) (cherry picked from commit9cf7052bd4) (cherry picked from commita13c9667e0) Conflicts: main.go https://codeberg.org/forgejo/forgejo/pulls/1216 (cherry picked from commita9c3cf060d) (cherry picked from commiteb6d904b23) (cherry picked from commitfd4878e487) (cherry picked from commitcc2cab9241) (cherry picked from commitbe523c63db) (cherry picked from commit254d1a2f9a) (cherry picked from commit1eecf4200e)
		
			
				
	
	
		
			222 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			222 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2023 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package cmd
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"strings"
 | |
| 
 | |
| 	"code.gitea.io/gitea/cmd/forgejo"
 | |
| 	"code.gitea.io/gitea/modules/log"
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| 
 | |
| 	"github.com/urfave/cli/v2"
 | |
| )
 | |
| 
 | |
| // cmdHelp is our own help subcommand with more information
 | |
| func cmdHelp() *cli.Command {
 | |
| 	c := &cli.Command{
 | |
| 		Name:      "help",
 | |
| 		Aliases:   []string{"h"},
 | |
| 		Usage:     "Shows a list of commands or help for one command",
 | |
| 		ArgsUsage: "[command]",
 | |
| 		Action: func(c *cli.Context) (err error) {
 | |
| 			lineage := c.Lineage() // The order is from child to parent: help, doctor, Gitea, {Command:nil}
 | |
| 			targetCmdIdx := 0
 | |
| 			if c.Command.Name == "help" {
 | |
| 				targetCmdIdx = 1
 | |
| 			}
 | |
| 			if lineage[targetCmdIdx+1].Command != nil {
 | |
| 				err = cli.ShowCommandHelp(lineage[targetCmdIdx+1], lineage[targetCmdIdx].Command.Name)
 | |
| 			} else {
 | |
| 				err = cli.ShowAppHelp(c)
 | |
| 			}
 | |
| 			_, _ = fmt.Fprintf(c.App.Writer, `
 | |
| DEFAULT CONFIGURATION:
 | |
|    AppPath:    %s
 | |
|    WorkPath:   %s
 | |
|    CustomPath: %s
 | |
|    ConfigFile: %s
 | |
| 
 | |
| `, setting.AppPath, setting.AppWorkPath, setting.CustomPath, setting.CustomConf)
 | |
| 			return err
 | |
| 		},
 | |
| 	}
 | |
| 	return c
 | |
| }
 | |
| 
 | |
| var helpFlag = cli.HelpFlag
 | |
| 
 | |
| func init() {
 | |
| 	// cli.HelpFlag = nil TODO: after https://github.com/urfave/cli/issues/1794 we can use this
 | |
| }
 | |
| 
 | |
| func appGlobalFlags() []cli.Flag {
 | |
| 	return []cli.Flag{
 | |
| 		// make the builtin flags at the top
 | |
| 		helpFlag,
 | |
| 
 | |
| 		// shared configuration flags, they are for global and for each sub-command at the same time
 | |
| 		// eg: such command is valid: "./gitea --config /tmp/app.ini web --config /tmp/app.ini", while it's discouraged indeed
 | |
| 		// keep in mind that the short flags like "-C", "-c" and "-w" are globally polluted, they can't be used for sub-commands anymore.
 | |
| 		&cli.StringFlag{
 | |
| 			Name:    "custom-path",
 | |
| 			Aliases: []string{"C"},
 | |
| 			Usage:   "Set custom path (defaults to '{WorkPath}/custom')",
 | |
| 		},
 | |
| 		&cli.StringFlag{
 | |
| 			Name:    "config",
 | |
| 			Aliases: []string{"c"},
 | |
| 			Value:   setting.CustomConf,
 | |
| 			Usage:   "Set custom config file (defaults to '{WorkPath}/custom/conf/app.ini')",
 | |
| 		},
 | |
| 		&cli.StringFlag{
 | |
| 			Name:    "work-path",
 | |
| 			Aliases: []string{"w"},
 | |
| 			Usage:   "Set Forgejo's working path (defaults to the directory of the Forgejo binary)",
 | |
| 		},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func prepareSubcommandWithConfig(command *cli.Command, globalFlags []cli.Flag) {
 | |
| 	command.Flags = append(append([]cli.Flag{}, globalFlags...), command.Flags...)
 | |
| 	command.Action = prepareWorkPathAndCustomConf(command.Action)
 | |
| 	command.HideHelp = true
 | |
| 	if command.Name != "help" {
 | |
| 		command.Subcommands = append(command.Subcommands, cmdHelp())
 | |
| 	}
 | |
| 	for i := range command.Subcommands {
 | |
| 		prepareSubcommandWithConfig(command.Subcommands[i], globalFlags)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // prepareWorkPathAndCustomConf wraps the Action to prepare the work path and custom config
 | |
| // It can't use "Before", because each level's sub-command's Before will be called one by one, so the "init" would be done multiple times
 | |
| func prepareWorkPathAndCustomConf(action cli.ActionFunc) func(ctx *cli.Context) error {
 | |
| 	return func(ctx *cli.Context) error {
 | |
| 		var args setting.ArgWorkPathAndCustomConf
 | |
| 		// from children to parent, check the global flags
 | |
| 		for _, curCtx := range ctx.Lineage() {
 | |
| 			if curCtx.IsSet("work-path") && args.WorkPath == "" {
 | |
| 				args.WorkPath = curCtx.String("work-path")
 | |
| 			}
 | |
| 			if curCtx.IsSet("custom-path") && args.CustomPath == "" {
 | |
| 				args.CustomPath = curCtx.String("custom-path")
 | |
| 			}
 | |
| 			if curCtx.IsSet("config") && args.CustomConf == "" {
 | |
| 				args.CustomConf = curCtx.String("config")
 | |
| 			}
 | |
| 		}
 | |
| 		setting.InitWorkPathAndCommonConfig(os.Getenv, args)
 | |
| 		if ctx.Bool("help") || action == nil {
 | |
| 			// the default behavior of "urfave/cli": "nil action" means "show help"
 | |
| 			return cmdHelp().Action(ctx)
 | |
| 		}
 | |
| 		return action(ctx)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func NewMainApp(version, versionExtra string) *cli.App {
 | |
| 	path, err := os.Executable()
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 	executable := filepath.Base(path)
 | |
| 
 | |
| 	var subCmdsStandalone []*cli.Command = make([]*cli.Command, 0, 10)
 | |
| 	var subCmdWithConfig []*cli.Command = make([]*cli.Command, 0, 10)
 | |
| 
 | |
| 	//
 | |
| 	// If the executable is forgejo-cli, provide a Forgejo specific CLI
 | |
| 	// that is NOT compatible with Gitea.
 | |
| 	//
 | |
| 	if executable == "forgejo-cli" {
 | |
| 		subCmdsStandalone = append(subCmdsStandalone, forgejo.CmdActions(context.Background()))
 | |
| 	} else {
 | |
| 		//
 | |
| 		// Otherwise provide a Gitea compatible CLI which includes Forgejo
 | |
| 		// specific additions under the forgejo-cli subcommand. It allows
 | |
| 		// admins to migration from Gitea to Forgejo by replacing the gitea
 | |
| 		// binary and rename it to forgejo if they want.
 | |
| 		//
 | |
| 		subCmdsStandalone = append(subCmdsStandalone, forgejo.CmdForgejo(context.Background()))
 | |
| 		subCmdWithConfig = append(subCmdWithConfig, CmdActions)
 | |
| 	}
 | |
| 
 | |
| 	return innerNewMainApp(version, versionExtra, subCmdsStandalone, subCmdWithConfig)
 | |
| }
 | |
| 
 | |
| func innerNewMainApp(version, versionExtra string, subCmdsStandaloneArgs, subCmdWithConfigArgs []*cli.Command) *cli.App {
 | |
| 	app := cli.NewApp()
 | |
| 	app.Name = "Forgejo"
 | |
| 	app.Usage = "Beyond coding. We forge."
 | |
| 	app.Description = `By default, forgejo will start serving using the web-server with no argument, which can alternatively be run by running the subcommand "web".`
 | |
| 	app.Version = version + versionExtra
 | |
| 	app.EnableBashCompletion = true
 | |
| 
 | |
| 	// these sub-commands need to use config file
 | |
| 	subCmdWithConfig := []*cli.Command{
 | |
| 		CmdWeb,
 | |
| 		CmdServ,
 | |
| 		CmdHook,
 | |
| 		CmdDump,
 | |
| 		CmdAdmin,
 | |
| 		CmdMigrate,
 | |
| 		CmdKeys,
 | |
| 		CmdDoctor,
 | |
| 		CmdManager,
 | |
| 		CmdEmbedded,
 | |
| 		CmdMigrateStorage,
 | |
| 		CmdDumpRepository,
 | |
| 		CmdRestoreRepository,
 | |
| 		cmdHelp(), // the "help" sub-command was used to show the more information for "work path" and "custom config"
 | |
| 	}
 | |
| 
 | |
| 	cmdConvert := util.ToPointer(*cmdDoctorConvert)
 | |
| 	cmdConvert.Hidden = true // still support the legacy "./gitea doctor" by the hidden sub-command, remove it in next release
 | |
| 	subCmdWithConfig = append(subCmdWithConfig, cmdConvert)
 | |
| 	subCmdWithConfig = append(subCmdWithConfig, subCmdWithConfigArgs...)
 | |
| 
 | |
| 	// these sub-commands do not need the config file, and they do not depend on any path or environment variable.
 | |
| 	subCmdStandalone := []*cli.Command{
 | |
| 		CmdCert,
 | |
| 		CmdGenerate,
 | |
| 		CmdDocs,
 | |
| 	}
 | |
| 	subCmdStandalone = append(subCmdStandalone, subCmdsStandaloneArgs...)
 | |
| 
 | |
| 	app.DefaultCommand = CmdWeb.Name
 | |
| 
 | |
| 	globalFlags := appGlobalFlags()
 | |
| 	app.Flags = append(app.Flags, cli.VersionFlag)
 | |
| 	app.Flags = append(app.Flags, globalFlags...)
 | |
| 	app.HideHelp = true // use our own help action to show helps (with more information like default config)
 | |
| 	app.Before = PrepareConsoleLoggerLevel(log.INFO)
 | |
| 	for i := range subCmdWithConfig {
 | |
| 		prepareSubcommandWithConfig(subCmdWithConfig[i], globalFlags)
 | |
| 	}
 | |
| 	app.Commands = append(app.Commands, subCmdWithConfig...)
 | |
| 	app.Commands = append(app.Commands, subCmdStandalone...)
 | |
| 
 | |
| 	return app
 | |
| }
 | |
| 
 | |
| func RunMainApp(app *cli.App, args ...string) error {
 | |
| 	err := app.Run(args)
 | |
| 	if err == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 	if strings.HasPrefix(err.Error(), "flag provided but not defined:") {
 | |
| 		// the cli package should already have output the error message, so just exit
 | |
| 		cli.OsExiter(1)
 | |
| 		return err
 | |
| 	}
 | |
| 	_, _ = fmt.Fprintf(app.ErrWriter, "Command error: %v\n", err)
 | |
| 	cli.OsExiter(1)
 | |
| 	return err
 | |
| }
 |