/* GoToSocial Copyright (C) 2021 GoToSocial Authors admin@gotosocial.org This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . */ package main import ( "fmt" "os" "github.com/sirupsen/logrus" "github.com/superseriousbusiness/gotosocial/internal/action" "github.com/superseriousbusiness/gotosocial/internal/config" "github.com/superseriousbusiness/gotosocial/internal/db" "github.com/superseriousbusiness/gotosocial/internal/gotosocial" "github.com/superseriousbusiness/gotosocial/internal/log" "github.com/superseriousbusiness/gotosocial/testrig" "github.com/urfave/cli/v2" ) func main() { flagNames := config.GetFlagNames() envNames := config.GetEnvNames() defaults := config.GetDefaults() app := &cli.App{ Usage: "a fediverse social media server", Flags: []cli.Flag{ // GENERAL FLAGS &cli.StringFlag{ Name: flagNames.LogLevel, Usage: "Log level to run at: debug, info, warn, fatal", Value: defaults.LogLevel, EnvVars: []string{envNames.LogLevel}, }, &cli.StringFlag{ Name: flagNames.ApplicationName, Usage: "Name of the application, used in various places internally", Value: defaults.ApplicationName, EnvVars: []string{envNames.ApplicationName}, Hidden: true, }, &cli.StringFlag{ Name: flagNames.ConfigPath, Usage: "Path to a yaml file containing gotosocial configuration. Values set in this file will be overwritten by values set as env vars or arguments", Value: defaults.ConfigPath, EnvVars: []string{envNames.ConfigPath}, }, &cli.StringFlag{ Name: flagNames.Host, Usage: "Hostname to use for the server (eg., example.org, gotosocial.whatever.com)", Value: defaults.Host, EnvVars: []string{envNames.Host}, }, &cli.StringFlag{ Name: flagNames.Protocol, Usage: "Protocol to use for the REST api of the server (only use http for debugging and tests!)", Value: defaults.Protocol, EnvVars: []string{envNames.Protocol}, }, // DATABASE FLAGS &cli.StringFlag{ Name: flagNames.DbType, Usage: "Database type: eg., postgres", Value: defaults.DbType, EnvVars: []string{envNames.DbType}, }, &cli.StringFlag{ Name: flagNames.DbAddress, Usage: "Database ipv4 address or hostname", Value: defaults.DbAddress, EnvVars: []string{envNames.DbAddress}, }, &cli.IntFlag{ Name: flagNames.DbPort, Usage: "Database port", Value: defaults.DbPort, EnvVars: []string{envNames.DbPort}, }, &cli.StringFlag{ Name: flagNames.DbUser, Usage: "Database username", Value: defaults.DbUser, EnvVars: []string{envNames.DbUser}, }, &cli.StringFlag{ Name: flagNames.DbPassword, Usage: "Database password", Value: defaults.DbPassword, EnvVars: []string{envNames.DbPassword}, }, &cli.StringFlag{ Name: flagNames.DbDatabase, Usage: "Database name", Value: defaults.DbDatabase, EnvVars: []string{envNames.DbDatabase}, }, // TEMPLATE FLAGS &cli.StringFlag{ Name: flagNames.TemplateBaseDir, Usage: "Basedir for html templating files for rendering pages and composing emails.", Value: defaults.TemplateBaseDir, EnvVars: []string{envNames.TemplateBaseDir}, }, // ACCOUNTS FLAGS &cli.BoolFlag{ Name: flagNames.AccountsOpenRegistration, Usage: "Allow anyone to submit an account signup request. If false, server will be invite-only.", Value: defaults.AccountsOpenRegistration, EnvVars: []string{envNames.AccountsOpenRegistration}, }, &cli.BoolFlag{ Name: flagNames.AccountsApprovalRequired, Usage: "Do account signups require approval by an admin or moderator before user can log in? If false, new registrations will be automatically approved.", Value: defaults.AccountsRequireApproval, EnvVars: []string{envNames.AccountsApprovalRequired}, }, &cli.BoolFlag{ Name: flagNames.AccountsReasonRequired, Usage: "Do new account signups require a reason to be submitted on registration?", Value: defaults.AccountsReasonRequired, EnvVars: []string{envNames.AccountsReasonRequired}, }, // MEDIA FLAGS &cli.IntFlag{ Name: flagNames.MediaMaxImageSize, Usage: "Max size of accepted images in bytes", Value: defaults.MediaMaxImageSize, EnvVars: []string{envNames.MediaMaxImageSize}, }, &cli.IntFlag{ Name: flagNames.MediaMaxVideoSize, Usage: "Max size of accepted videos in bytes", Value: defaults.MediaMaxVideoSize, EnvVars: []string{envNames.MediaMaxVideoSize}, }, &cli.IntFlag{ Name: flagNames.MediaMinDescriptionChars, Usage: "Min required chars for an image description", Value: defaults.MediaMinDescriptionChars, EnvVars: []string{envNames.MediaMinDescriptionChars}, }, &cli.IntFlag{ Name: flagNames.MediaMaxDescriptionChars, Usage: "Max permitted chars for an image description", Value: defaults.MediaMaxDescriptionChars, EnvVars: []string{envNames.MediaMaxDescriptionChars}, }, // STORAGE FLAGS &cli.StringFlag{ Name: flagNames.StorageBackend, Usage: "Storage backend to use for media attachments", Value: defaults.StorageBackend, EnvVars: []string{envNames.StorageBackend}, }, &cli.StringFlag{ Name: flagNames.StorageBasePath, Usage: "Full path to an already-created directory where gts should store/retrieve media files. Subfolders will be created within this dir.", Value: defaults.StorageBasePath, EnvVars: []string{envNames.StorageBasePath}, }, &cli.StringFlag{ Name: flagNames.StorageServeProtocol, Usage: "Protocol to use for serving media attachments (use https if storage is local)", Value: defaults.StorageServeProtocol, EnvVars: []string{envNames.StorageServeProtocol}, }, &cli.StringFlag{ Name: flagNames.StorageServeHost, Usage: "Hostname to serve media attachments from (use the same value as host if storage is local)", Value: defaults.StorageServeHost, EnvVars: []string{envNames.StorageServeHost}, }, &cli.StringFlag{ Name: flagNames.StorageServeBasePath, Usage: "Path to append to protocol and hostname to create the base path from which media files will be served (default will mostly be fine)", Value: defaults.StorageServeBasePath, EnvVars: []string{envNames.StorageServeBasePath}, }, // STATUSES FLAGS &cli.IntFlag{ Name: flagNames.StatusesMaxChars, Usage: "Max permitted characters for posted statuses", Value: defaults.StatusesMaxChars, EnvVars: []string{envNames.StatusesMaxChars}, }, &cli.IntFlag{ Name: flagNames.StatusesCWMaxChars, Usage: "Max permitted characters for content/spoiler warnings on statuses", Value: defaults.StatusesCWMaxChars, EnvVars: []string{envNames.StatusesCWMaxChars}, }, &cli.IntFlag{ Name: flagNames.StatusesPollMaxOptions, Usage: "Max amount of options permitted on a poll", Value: defaults.StatusesPollMaxOptions, EnvVars: []string{envNames.StatusesPollMaxOptions}, }, &cli.IntFlag{ Name: flagNames.StatusesPollOptionMaxChars, Usage: "Max amount of characters for a poll option", Value: defaults.StatusesPollOptionMaxChars, EnvVars: []string{envNames.StatusesPollOptionMaxChars}, }, &cli.IntFlag{ Name: flagNames.StatusesMaxMediaFiles, Usage: "Maximum number of media files/attachments per status", Value: defaults.StatusesMaxMediaFiles, EnvVars: []string{envNames.StatusesMaxMediaFiles}, }, }, Commands: []*cli.Command{ { Name: "server", Usage: "gotosocial server-related tasks", Subcommands: []*cli.Command{ { Name: "start", Usage: "start the gotosocial server", Action: func(c *cli.Context) error { return runAction(c, gotosocial.Run) }, }, }, }, { Name: "db", Usage: "database-related tasks and utils", Subcommands: []*cli.Command{ { Name: "init", Usage: "initialize a database with the required schema for gotosocial; has no effect & is safe to run on an already-initialized db", Action: func(c *cli.Context) error { return runAction(c, db.Initialize) }, }, }, }, { Name: "testrig", Usage: "gotosocial testrig tasks", Subcommands: []*cli.Command{ { Name: "start", Usage: "start the gotosocial testrig", Action: func(c *cli.Context) error { return runAction(c, testrig.Run) }, }, }, }, }, } err := app.Run(os.Args) if err != nil { logrus.Fatal(err) } } // runAction builds up the config and logger necessary for any // gotosocial action, and then executes the action. func runAction(c *cli.Context, a action.GTSAction) error { // create a new *config.Config based on the config path provided... conf, err := config.FromFile(c.String(config.GetFlagNames().ConfigPath)) if err != nil { return fmt.Errorf("error creating config: %s", err) } // ... and the flags set on the *cli.Context by urfave conf.ParseCLIFlags(c) // create a logger with the log level, formatting, and output splitter already set log, err := log.New(conf.LogLevel) if err != nil { return fmt.Errorf("error creating logger: %s", err) } return a(c.Context, conf, log) }