| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593 |
- // Copyright 2019 Frédéric Guillot. All rights reserved.
- // Use of this source code is governed by the Apache 2.0
- // license that can be found in the LICENSE file.
- package config // import "miniflux.app/config"
- import (
- "fmt"
- "sort"
- "strings"
- "time"
- "miniflux.app/version"
- )
- const (
- defaultHTTPS = false
- defaultLogDateTime = false
- defaultHSTS = true
- defaultHTTPService = true
- defaultSchedulerService = true
- defaultDebug = false
- defaultTiming = false
- defaultBaseURL = "http://localhost"
- defaultRootURL = "http://localhost"
- defaultBasePath = ""
- defaultWorkerPoolSize = 5
- defaultPollingFrequency = 60
- defaultBatchSize = 100
- defaultPollingScheduler = "round_robin"
- defaultSchedulerEntryFrequencyMinInterval = 5
- defaultSchedulerEntryFrequencyMaxInterval = 24 * 60
- defaultPollingParsingErrorLimit = 3
- defaultRunMigrations = false
- defaultDatabaseURL = "user=postgres password=postgres dbname=miniflux2 sslmode=disable"
- defaultDatabaseMaxConns = 20
- defaultDatabaseMinConns = 1
- defaultDatabaseConnectionLifetime = 5
- defaultListenAddr = "127.0.0.1:8080"
- defaultCertFile = ""
- defaultKeyFile = ""
- defaultCertDomain = ""
- defaultCleanupFrequencyHours = 24
- defaultCleanupArchiveReadDays = 60
- defaultCleanupArchiveUnreadDays = 180
- defaultCleanupArchiveBatchSize = 10000
- defaultCleanupRemoveSessionsDays = 30
- defaultProxyImages = "http-only"
- defaultProxyImageUrl = ""
- defaultFetchYouTubeWatchTime = false
- defaultCreateAdmin = false
- defaultAdminUsername = ""
- defaultAdminPassword = ""
- defaultOAuth2UserCreation = false
- defaultOAuth2ClientID = ""
- defaultOAuth2ClientSecret = ""
- defaultOAuth2RedirectURL = ""
- defaultOAuth2OidcDiscoveryEndpoint = ""
- defaultOAuth2Provider = ""
- defaultPocketConsumerKey = ""
- defaultHTTPClientTimeout = 20
- defaultHTTPClientMaxBodySize = 15
- defaultHTTPClientProxy = ""
- defaultAuthProxyHeader = ""
- defaultAuthProxyUserCreation = false
- defaultMaintenanceMode = false
- defaultMaintenanceMessage = "Miniflux is currently under maintenance"
- defaultMetricsCollector = false
- defaultMetricsRefreshInterval = 60
- defaultMetricsAllowedNetworks = "127.0.0.1/8"
- defaultWatchdog = true
- defaultInvidiousInstance = "yewtu.be"
- )
- var defaultHTTPClientUserAgent = "Mozilla/5.0 (compatible; Miniflux/" + version.Version + "; +https://miniflux.app)"
- // Option contains a key to value map of a single option. It may be used to output debug strings.
- type Option struct {
- Key string
- Value interface{}
- }
- // Options contains configuration options.
- type Options struct {
- HTTPS bool
- logDateTime bool
- hsts bool
- httpService bool
- schedulerService bool
- debug bool
- serverTimingHeader bool
- baseURL string
- rootURL string
- basePath string
- databaseURL string
- databaseMaxConns int
- databaseMinConns int
- databaseConnectionLifetime int
- runMigrations bool
- listenAddr string
- certFile string
- certDomain string
- certKeyFile string
- cleanupFrequencyHours int
- cleanupArchiveReadDays int
- cleanupArchiveUnreadDays int
- cleanupArchiveBatchSize int
- cleanupRemoveSessionsDays int
- pollingFrequency int
- batchSize int
- pollingScheduler string
- schedulerEntryFrequencyMinInterval int
- schedulerEntryFrequencyMaxInterval int
- pollingParsingErrorLimit int
- workerPoolSize int
- createAdmin bool
- adminUsername string
- adminPassword string
- proxyImages string
- proxyImageUrl string
- fetchYouTubeWatchTime bool
- oauth2UserCreationAllowed bool
- oauth2ClientID string
- oauth2ClientSecret string
- oauth2RedirectURL string
- oauth2OidcDiscoveryEndpoint string
- oauth2Provider string
- pocketConsumerKey string
- httpClientTimeout int
- httpClientMaxBodySize int64
- httpClientProxy string
- httpClientUserAgent string
- authProxyHeader string
- authProxyUserCreation bool
- maintenanceMode bool
- maintenanceMessage string
- metricsCollector bool
- metricsRefreshInterval int
- metricsAllowedNetworks []string
- watchdog bool
- invidiousInstance string
- }
- // NewOptions returns Options with default values.
- func NewOptions() *Options {
- return &Options{
- HTTPS: defaultHTTPS,
- logDateTime: defaultLogDateTime,
- hsts: defaultHSTS,
- httpService: defaultHTTPService,
- schedulerService: defaultSchedulerService,
- debug: defaultDebug,
- serverTimingHeader: defaultTiming,
- baseURL: defaultBaseURL,
- rootURL: defaultRootURL,
- basePath: defaultBasePath,
- databaseURL: defaultDatabaseURL,
- databaseMaxConns: defaultDatabaseMaxConns,
- databaseMinConns: defaultDatabaseMinConns,
- databaseConnectionLifetime: defaultDatabaseConnectionLifetime,
- runMigrations: defaultRunMigrations,
- listenAddr: defaultListenAddr,
- certFile: defaultCertFile,
- certDomain: defaultCertDomain,
- certKeyFile: defaultKeyFile,
- cleanupFrequencyHours: defaultCleanupFrequencyHours,
- cleanupArchiveReadDays: defaultCleanupArchiveReadDays,
- cleanupArchiveUnreadDays: defaultCleanupArchiveUnreadDays,
- cleanupArchiveBatchSize: defaultCleanupArchiveBatchSize,
- cleanupRemoveSessionsDays: defaultCleanupRemoveSessionsDays,
- pollingFrequency: defaultPollingFrequency,
- batchSize: defaultBatchSize,
- pollingScheduler: defaultPollingScheduler,
- schedulerEntryFrequencyMinInterval: defaultSchedulerEntryFrequencyMinInterval,
- schedulerEntryFrequencyMaxInterval: defaultSchedulerEntryFrequencyMaxInterval,
- pollingParsingErrorLimit: defaultPollingParsingErrorLimit,
- workerPoolSize: defaultWorkerPoolSize,
- createAdmin: defaultCreateAdmin,
- proxyImages: defaultProxyImages,
- proxyImageUrl: defaultProxyImageUrl,
- fetchYouTubeWatchTime: defaultFetchYouTubeWatchTime,
- oauth2UserCreationAllowed: defaultOAuth2UserCreation,
- oauth2ClientID: defaultOAuth2ClientID,
- oauth2ClientSecret: defaultOAuth2ClientSecret,
- oauth2RedirectURL: defaultOAuth2RedirectURL,
- oauth2OidcDiscoveryEndpoint: defaultOAuth2OidcDiscoveryEndpoint,
- oauth2Provider: defaultOAuth2Provider,
- pocketConsumerKey: defaultPocketConsumerKey,
- httpClientTimeout: defaultHTTPClientTimeout,
- httpClientMaxBodySize: defaultHTTPClientMaxBodySize * 1024 * 1024,
- httpClientProxy: defaultHTTPClientProxy,
- httpClientUserAgent: defaultHTTPClientUserAgent,
- authProxyHeader: defaultAuthProxyHeader,
- authProxyUserCreation: defaultAuthProxyUserCreation,
- maintenanceMode: defaultMaintenanceMode,
- maintenanceMessage: defaultMaintenanceMessage,
- metricsCollector: defaultMetricsCollector,
- metricsRefreshInterval: defaultMetricsRefreshInterval,
- metricsAllowedNetworks: []string{defaultMetricsAllowedNetworks},
- watchdog: defaultWatchdog,
- invidiousInstance: defaultInvidiousInstance,
- }
- }
- // LogDateTime returns true if the date/time should be displayed in log messages.
- func (o *Options) LogDateTime() bool {
- return o.logDateTime
- }
- // HasMaintenanceMode returns true if maintenance mode is enabled.
- func (o *Options) HasMaintenanceMode() bool {
- return o.maintenanceMode
- }
- // MaintenanceMessage returns maintenance message.
- func (o *Options) MaintenanceMessage() string {
- return o.maintenanceMessage
- }
- // HasDebugMode returns true if debug mode is enabled.
- func (o *Options) HasDebugMode() bool {
- return o.debug
- }
- // HasServerTimingHeader returns true if server-timing headers enabled.
- func (o *Options) HasServerTimingHeader() bool {
- return o.serverTimingHeader
- }
- // BaseURL returns the application base URL with path.
- func (o *Options) BaseURL() string {
- return o.baseURL
- }
- // RootURL returns the base URL without path.
- func (o *Options) RootURL() string {
- return o.rootURL
- }
- // BasePath returns the application base path according to the base URL.
- func (o *Options) BasePath() string {
- return o.basePath
- }
- // IsDefaultDatabaseURL returns true if the default database URL is used.
- func (o *Options) IsDefaultDatabaseURL() bool {
- return o.databaseURL == defaultDatabaseURL
- }
- // DatabaseURL returns the database URL.
- func (o *Options) DatabaseURL() string {
- return o.databaseURL
- }
- // DatabaseMaxConns returns the maximum number of database connections.
- func (o *Options) DatabaseMaxConns() int {
- return o.databaseMaxConns
- }
- // DatabaseMinConns returns the minimum number of database connections.
- func (o *Options) DatabaseMinConns() int {
- return o.databaseMinConns
- }
- // DatabaseConnectionLifetime returns the maximum amount of time a connection may be reused.
- func (o *Options) DatabaseConnectionLifetime() time.Duration {
- return time.Duration(o.databaseConnectionLifetime) * time.Minute
- }
- // ListenAddr returns the listen address for the HTTP server.
- func (o *Options) ListenAddr() string {
- return o.listenAddr
- }
- // CertFile returns the SSL certificate filename if any.
- func (o *Options) CertFile() string {
- return o.certFile
- }
- // CertKeyFile returns the private key filename for custom SSL certificate.
- func (o *Options) CertKeyFile() string {
- return o.certKeyFile
- }
- // CertDomain returns the domain to use for Let's Encrypt certificate.
- func (o *Options) CertDomain() string {
- return o.certDomain
- }
- // CleanupFrequencyHours returns the interval in hours for cleanup jobs.
- func (o *Options) CleanupFrequencyHours() int {
- return o.cleanupFrequencyHours
- }
- // CleanupArchiveReadDays returns the number of days after which marking read items as removed.
- func (o *Options) CleanupArchiveReadDays() int {
- return o.cleanupArchiveReadDays
- }
- // CleanupArchiveUnreadDays returns the number of days after which marking unread items as removed.
- func (o *Options) CleanupArchiveUnreadDays() int {
- return o.cleanupArchiveUnreadDays
- }
- // CleanupArchiveBatchSize returns the number of entries to archive for each interval.
- func (o *Options) CleanupArchiveBatchSize() int {
- return o.cleanupArchiveBatchSize
- }
- // CleanupRemoveSessionsDays returns the number of days after which to remove sessions.
- func (o *Options) CleanupRemoveSessionsDays() int {
- return o.cleanupRemoveSessionsDays
- }
- // WorkerPoolSize returns the number of background worker.
- func (o *Options) WorkerPoolSize() int {
- return o.workerPoolSize
- }
- // PollingFrequency returns the interval to refresh feeds in the background.
- func (o *Options) PollingFrequency() int {
- return o.pollingFrequency
- }
- // BatchSize returns the number of feeds to send for background processing.
- func (o *Options) BatchSize() int {
- return o.batchSize
- }
- // PollingScheduler returns the scheduler used for polling feeds.
- func (o *Options) PollingScheduler() string {
- return o.pollingScheduler
- }
- // SchedulerEntryFrequencyMaxInterval returns the maximum interval in minutes for the entry frequency scheduler.
- func (o *Options) SchedulerEntryFrequencyMaxInterval() int {
- return o.schedulerEntryFrequencyMaxInterval
- }
- // SchedulerEntryFrequencyMinInterval returns the minimum interval in minutes for the entry frequency scheduler.
- func (o *Options) SchedulerEntryFrequencyMinInterval() int {
- return o.schedulerEntryFrequencyMinInterval
- }
- // PollingParsingErrorLimit returns the limit of errors when to stop polling.
- func (o *Options) PollingParsingErrorLimit() int {
- return o.pollingParsingErrorLimit
- }
- // IsOAuth2UserCreationAllowed returns true if user creation is allowed for OAuth2 users.
- func (o *Options) IsOAuth2UserCreationAllowed() bool {
- return o.oauth2UserCreationAllowed
- }
- // OAuth2ClientID returns the OAuth2 Client ID.
- func (o *Options) OAuth2ClientID() string {
- return o.oauth2ClientID
- }
- // OAuth2ClientSecret returns the OAuth2 client secret.
- func (o *Options) OAuth2ClientSecret() string {
- return o.oauth2ClientSecret
- }
- // OAuth2RedirectURL returns the OAuth2 redirect URL.
- func (o *Options) OAuth2RedirectURL() string {
- return o.oauth2RedirectURL
- }
- // OAuth2OidcDiscoveryEndpoint returns the OAuth2 OIDC discovery endpoint.
- func (o *Options) OAuth2OidcDiscoveryEndpoint() string {
- return o.oauth2OidcDiscoveryEndpoint
- }
- // OAuth2Provider returns the name of the OAuth2 provider configured.
- func (o *Options) OAuth2Provider() string {
- return o.oauth2Provider
- }
- // HasHSTS returns true if HTTP Strict Transport Security is enabled.
- func (o *Options) HasHSTS() bool {
- return o.hsts
- }
- // RunMigrations returns true if the environment variable RUN_MIGRATIONS is not empty.
- func (o *Options) RunMigrations() bool {
- return o.runMigrations
- }
- // CreateAdmin returns true if the environment variable CREATE_ADMIN is not empty.
- func (o *Options) CreateAdmin() bool {
- return o.createAdmin
- }
- // AdminUsername returns the admin username if defined.
- func (o *Options) AdminUsername() string {
- return o.adminUsername
- }
- // AdminPassword returns the admin password if defined.
- func (o *Options) AdminPassword() string {
- return o.adminPassword
- }
- // FetchYouTubeWatchTime returns true if the YouTube video duration
- // should be fetched and used as a reading time.
- func (o *Options) FetchYouTubeWatchTime() bool {
- return o.fetchYouTubeWatchTime
- }
- // ProxyImages returns "none" to never proxy, "http-only" to proxy non-HTTPS, "all" to always proxy.
- func (o *Options) ProxyImages() string {
- return o.proxyImages
- }
- // ProxyImageUrl returns a string of a URL to use to proxy image requests
- func (o *Options) ProxyImageUrl() string {
- return o.proxyImageUrl
- }
- // HasHTTPService returns true if the HTTP service is enabled.
- func (o *Options) HasHTTPService() bool {
- return o.httpService
- }
- // HasSchedulerService returns true if the scheduler service is enabled.
- func (o *Options) HasSchedulerService() bool {
- return o.schedulerService
- }
- // PocketConsumerKey returns the Pocket Consumer Key if configured.
- func (o *Options) PocketConsumerKey(defaultValue string) string {
- if o.pocketConsumerKey != "" {
- return o.pocketConsumerKey
- }
- return defaultValue
- }
- // HTTPClientTimeout returns the time limit in seconds before the HTTP client cancel the request.
- func (o *Options) HTTPClientTimeout() int {
- return o.httpClientTimeout
- }
- // HTTPClientMaxBodySize returns the number of bytes allowed for the HTTP client to transfer.
- func (o *Options) HTTPClientMaxBodySize() int64 {
- return o.httpClientMaxBodySize
- }
- // HTTPClientProxy returns the proxy URL for HTTP client.
- func (o *Options) HTTPClientProxy() string {
- return o.httpClientProxy
- }
- // HasHTTPClientProxyConfigured returns true if the HTTP proxy is configured.
- func (o *Options) HasHTTPClientProxyConfigured() bool {
- return o.httpClientProxy != ""
- }
- // AuthProxyHeader returns an HTTP header name that contains username for
- // authentication using auth proxy.
- func (o *Options) AuthProxyHeader() string {
- return o.authProxyHeader
- }
- // IsAuthProxyUserCreationAllowed returns true if user creation is allowed for
- // users authenticated using auth proxy.
- func (o *Options) IsAuthProxyUserCreationAllowed() bool {
- return o.authProxyUserCreation
- }
- // HasMetricsCollector returns true if metrics collection is enabled.
- func (o *Options) HasMetricsCollector() bool {
- return o.metricsCollector
- }
- // MetricsRefreshInterval returns the refresh interval in seconds.
- func (o *Options) MetricsRefreshInterval() int {
- return o.metricsRefreshInterval
- }
- // MetricsAllowedNetworks returns the list of networks allowed to connect to the metrics endpoint.
- func (o *Options) MetricsAllowedNetworks() []string {
- return o.metricsAllowedNetworks
- }
- // HTTPClientUserAgent returns the global User-Agent header for miniflux.
- func (o *Options) HTTPClientUserAgent() string {
- return o.httpClientUserAgent
- }
- // HasWatchdog returns true if the systemd watchdog is enabled.
- func (o *Options) HasWatchdog() bool {
- return o.watchdog
- }
- // InvidiousInstance returns the invidious instance used by miniflux
- func (o *Options) InvidiousInstance() string {
- return o.invidiousInstance
- }
- // SortedOptions returns options as a list of key value pairs, sorted by keys.
- func (o *Options) SortedOptions(redactSecret bool) []*Option {
- var keyValues = map[string]interface{}{
- "ADMIN_PASSWORD": redactSecretValue(o.adminPassword, redactSecret),
- "ADMIN_USERNAME": o.adminUsername,
- "AUTH_PROXY_HEADER": o.authProxyHeader,
- "AUTH_PROXY_USER_CREATION": o.authProxyUserCreation,
- "BASE_PATH": o.basePath,
- "BASE_URL": o.baseURL,
- "BATCH_SIZE": o.batchSize,
- "CERT_DOMAIN": o.certDomain,
- "CERT_FILE": o.certFile,
- "CLEANUP_ARCHIVE_READ_DAYS": o.cleanupArchiveReadDays,
- "CLEANUP_ARCHIVE_UNREAD_DAYS": o.cleanupArchiveUnreadDays,
- "CLEANUP_ARCHIVE_BATCH_SIZE": o.cleanupArchiveBatchSize,
- "CLEANUP_FREQUENCY_HOURS": o.cleanupFrequencyHours,
- "CLEANUP_REMOVE_SESSIONS_DAYS": o.cleanupRemoveSessionsDays,
- "CREATE_ADMIN": o.createAdmin,
- "DATABASE_MAX_CONNS": o.databaseMaxConns,
- "DATABASE_MIN_CONNS": o.databaseMinConns,
- "DATABASE_CONNECTION_LIFETIME": o.databaseConnectionLifetime,
- "DATABASE_URL": redactSecretValue(o.databaseURL, redactSecret),
- "DEBUG": o.debug,
- "DISABLE_HSTS": !o.hsts,
- "DISABLE_SCHEDULER_SERVICE": !o.schedulerService,
- "DISABLE_HTTP_SERVICE": !o.httpService,
- "FETCH_YOUTUBE_WATCH_TIME": o.fetchYouTubeWatchTime,
- "HTTPS": o.HTTPS,
- "HTTP_CLIENT_MAX_BODY_SIZE": o.httpClientMaxBodySize,
- "HTTP_CLIENT_PROXY": o.httpClientProxy,
- "HTTP_CLIENT_TIMEOUT": o.httpClientTimeout,
- "HTTP_CLIENT_USER_AGENT": o.httpClientUserAgent,
- "HTTP_SERVICE": o.httpService,
- "KEY_FILE": o.certKeyFile,
- "INVIDIOUS_INSTANCE": o.invidiousInstance,
- "LISTEN_ADDR": o.listenAddr,
- "LOG_DATE_TIME": o.logDateTime,
- "MAINTENANCE_MESSAGE": o.maintenanceMessage,
- "MAINTENANCE_MODE": o.maintenanceMode,
- "METRICS_ALLOWED_NETWORKS": strings.Join(o.metricsAllowedNetworks, ","),
- "METRICS_COLLECTOR": o.metricsCollector,
- "METRICS_REFRESH_INTERVAL": o.metricsRefreshInterval,
- "OAUTH2_CLIENT_ID": o.oauth2ClientID,
- "OAUTH2_CLIENT_SECRET": redactSecretValue(o.oauth2ClientSecret, redactSecret),
- "OAUTH2_OIDC_DISCOVERY_ENDPOINT": o.oauth2OidcDiscoveryEndpoint,
- "OAUTH2_PROVIDER": o.oauth2Provider,
- "OAUTH2_REDIRECT_URL": o.oauth2RedirectURL,
- "OAUTH2_USER_CREATION": o.oauth2UserCreationAllowed,
- "POCKET_CONSUMER_KEY": redactSecretValue(o.pocketConsumerKey, redactSecret),
- "POLLING_FREQUENCY": o.pollingFrequency,
- "POLLING_PARSING_ERROR_LIMIT": o.pollingParsingErrorLimit,
- "POLLING_SCHEDULER": o.pollingScheduler,
- "PROXY_IMAGES": o.proxyImages,
- "PROXY_IMAGE_URL": o.proxyImageUrl,
- "ROOT_URL": o.rootURL,
- "RUN_MIGRATIONS": o.runMigrations,
- "SCHEDULER_ENTRY_FREQUENCY_MAX_INTERVAL": o.schedulerEntryFrequencyMaxInterval,
- "SCHEDULER_ENTRY_FREQUENCY_MIN_INTERVAL": o.schedulerEntryFrequencyMinInterval,
- "SCHEDULER_SERVICE": o.schedulerService,
- "SERVER_TIMING_HEADER": o.serverTimingHeader,
- "WORKER_POOL_SIZE": o.workerPoolSize,
- "WATCHDOG": o.watchdog,
- }
- keys := make([]string, 0, len(keyValues))
- for key := range keyValues {
- keys = append(keys, key)
- }
- sort.Strings(keys)
- var sortedOptions []*Option
- for _, key := range keys {
- sortedOptions = append(sortedOptions, &Option{Key: key, Value: keyValues[key]})
- }
- return sortedOptions
- }
- func (o *Options) String() string {
- var builder strings.Builder
- for _, option := range o.SortedOptions(false) {
- fmt.Fprintf(&builder, "%s=%v\n", option.Key, option.Value)
- }
- return builder.String()
- }
- func redactSecretValue(value string, redactSecret bool) string {
- if redactSecret && value != "" {
- return "******"
- }
- return value
- }
|