2022-11-13 19:17:07 +01:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2022-11-21 22:52:44 +01:00
|
|
|
"fmt"
|
2022-12-07 22:11:44 +01:00
|
|
|
"github.com/rs/zerolog"
|
2022-11-13 19:17:07 +01:00
|
|
|
"github.com/rs/zerolog/log"
|
2022-12-07 23:32:58 +01:00
|
|
|
"gogs.mikescher.com/BlackForestBytes/goext/timeext"
|
2022-11-13 19:17:07 +01:00
|
|
|
"os"
|
2022-12-07 23:32:58 +01:00
|
|
|
"reflect"
|
|
|
|
"strconv"
|
2022-11-18 21:25:40 +01:00
|
|
|
"time"
|
2022-11-13 19:17:07 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type Config struct {
|
2022-11-25 22:42:21 +01:00
|
|
|
Namespace string
|
2022-12-07 23:32:58 +01:00
|
|
|
BaseURL string `env:"SCN_URL"`
|
|
|
|
GinDebug bool `env:"SCN_GINDEBUG"`
|
|
|
|
LogLevel zerolog.Level `env:"SCN_LOGLEVEL"`
|
|
|
|
ServerIP string `env:"SCN_IP"`
|
|
|
|
ServerPort string `env:"SCN_PORT"`
|
|
|
|
DBFile string `env:"SCN_DB_FILE"`
|
|
|
|
DBJournal string `env:"SCN_DB_JOURNAL"`
|
|
|
|
DBTimeout time.Duration `env:"SCN_DB_TIMEOUT"`
|
|
|
|
DBMaxOpenConns int `env:"SCN_DB_MAXOPENCONNECTIONS"`
|
|
|
|
DBMaxIdleConns int `env:"SCN_DB_MAXIDLECONNECTIONS"`
|
|
|
|
DBConnMaxLifetime time.Duration `env:"SCN_DB_CONNEXTIONMAXLIFETIME"`
|
|
|
|
DBConnMaxIdleTime time.Duration `env:"SCN_DB_CONNEXTIONMAXIDLETIME"`
|
|
|
|
DBCheckForeignKeys bool `env:"SCN_DB_CHECKFOREIGNKEYS"`
|
|
|
|
RequestTimeout time.Duration `env:"SCN_REQUEST_TIMEOUT"`
|
|
|
|
ReturnRawErrors bool `env:"SCN_ERROR_RETURN"`
|
|
|
|
DummyFirebase bool `env:"SCN_DUMMY_FB"`
|
|
|
|
DummyGoogleAPI bool `env:"SCN_DUMMY_GOOG"`
|
2022-12-14 18:46:26 +01:00
|
|
|
FirebaseTokenURI string `env:"SCN_FB_TOKENURI"`
|
|
|
|
FirebaseProjectID string `env:"SCN_FB_PROJECTID"`
|
|
|
|
FirebasePrivKeyID string `env:"SCN_FB_PRIVATEKEYID"`
|
|
|
|
FirebaseClientMail string `env:"SCN_FB_CLIENTEMAIL"`
|
|
|
|
FirebasePrivateKey string `env:"SCN_FB_PRIVATEKEY"`
|
|
|
|
GoogleAPITokenURI string `env:"SCN_GOOG_TOKENURI"`
|
|
|
|
GoogleAPIPrivKeyID string `env:"SCN_GOOG_PRIVATEKEYID"`
|
|
|
|
GoogleAPIClientMail string `env:"SCN_GOOG_CLIENTEMAIL"`
|
|
|
|
GoogleAPIPrivateKey string `env:"SCN_GOOG_PRIVATEKEY"`
|
|
|
|
GooglePackageName string `env:"SCN_GOOG_PACKAGENAME"`
|
|
|
|
GoogleProProductID string `env:"SCN_GOOG_PROPRODUCTID"`
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var Conf Config
|
|
|
|
|
2022-11-23 19:32:23 +01:00
|
|
|
var configLocHost = func() Config {
|
|
|
|
return Config{
|
2022-11-25 22:42:21 +01:00
|
|
|
Namespace: "local-host",
|
|
|
|
BaseURL: "http://localhost:8080",
|
|
|
|
GinDebug: true,
|
2022-12-07 22:11:44 +01:00
|
|
|
LogLevel: zerolog.DebugLevel,
|
2022-11-25 22:42:21 +01:00
|
|
|
ServerIP: "0.0.0.0",
|
|
|
|
ServerPort: "8080",
|
|
|
|
DBFile: ".run-data/db.sqlite3",
|
2022-12-07 22:11:44 +01:00
|
|
|
DBJournal: "WAL",
|
|
|
|
DBTimeout: 5 * time.Second,
|
2022-12-09 00:40:50 +01:00
|
|
|
DBCheckForeignKeys: false,
|
2022-12-07 22:11:44 +01:00
|
|
|
DBMaxOpenConns: 5,
|
|
|
|
DBMaxIdleConns: 5,
|
|
|
|
DBConnMaxLifetime: 60 * time.Minute,
|
|
|
|
DBConnMaxIdleTime: 60 * time.Minute,
|
2022-11-25 22:42:21 +01:00
|
|
|
RequestTimeout: 16 * time.Second,
|
|
|
|
ReturnRawErrors: true,
|
|
|
|
DummyFirebase: true,
|
|
|
|
FirebaseTokenURI: "",
|
|
|
|
FirebaseProjectID: "",
|
|
|
|
FirebasePrivKeyID: "",
|
|
|
|
FirebaseClientMail: "",
|
|
|
|
FirebasePrivateKey: "",
|
|
|
|
DummyGoogleAPI: true,
|
|
|
|
GoogleAPITokenURI: "",
|
|
|
|
GoogleAPIPrivKeyID: "",
|
|
|
|
GoogleAPIClientMail: "",
|
|
|
|
GoogleAPIPrivateKey: "",
|
|
|
|
GooglePackageName: "",
|
|
|
|
GoogleProProductID: "",
|
2022-11-23 19:32:23 +01:00
|
|
|
}
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
|
|
|
|
2022-11-23 19:32:23 +01:00
|
|
|
var configLocDocker = func() Config {
|
|
|
|
return Config{
|
2022-11-25 22:42:21 +01:00
|
|
|
Namespace: "local-docker",
|
|
|
|
BaseURL: "http://localhost:8080",
|
|
|
|
GinDebug: true,
|
2022-12-07 22:11:44 +01:00
|
|
|
LogLevel: zerolog.DebugLevel,
|
2022-11-25 22:42:21 +01:00
|
|
|
ServerIP: "0.0.0.0",
|
|
|
|
ServerPort: "80",
|
|
|
|
DBFile: "/data/scn_docker.sqlite3",
|
2022-12-07 22:11:44 +01:00
|
|
|
DBJournal: "WAL",
|
|
|
|
DBTimeout: 5 * time.Second,
|
2022-12-09 00:40:50 +01:00
|
|
|
DBCheckForeignKeys: false,
|
2022-12-07 22:11:44 +01:00
|
|
|
DBMaxOpenConns: 5,
|
|
|
|
DBMaxIdleConns: 5,
|
|
|
|
DBConnMaxLifetime: 60 * time.Minute,
|
|
|
|
DBConnMaxIdleTime: 60 * time.Minute,
|
2022-11-25 22:42:21 +01:00
|
|
|
RequestTimeout: 16 * time.Second,
|
|
|
|
ReturnRawErrors: true,
|
|
|
|
DummyFirebase: true,
|
|
|
|
FirebaseTokenURI: "",
|
|
|
|
FirebaseProjectID: "",
|
|
|
|
FirebasePrivKeyID: "",
|
|
|
|
FirebaseClientMail: "",
|
|
|
|
FirebasePrivateKey: "",
|
|
|
|
DummyGoogleAPI: true,
|
|
|
|
GoogleAPITokenURI: "",
|
|
|
|
GoogleAPIPrivKeyID: "",
|
|
|
|
GoogleAPIClientMail: "",
|
|
|
|
GoogleAPIPrivateKey: "",
|
|
|
|
GooglePackageName: "",
|
|
|
|
GoogleProProductID: "",
|
2022-11-23 19:32:23 +01:00
|
|
|
}
|
2022-11-20 03:41:38 +01:00
|
|
|
}
|
|
|
|
|
2022-11-23 19:32:23 +01:00
|
|
|
var configDev = func() Config {
|
|
|
|
return Config{
|
2022-11-25 22:42:21 +01:00
|
|
|
Namespace: "develop",
|
2022-12-14 18:43:32 +01:00
|
|
|
BaseURL: confEnv("SCN_URL"),
|
2022-11-25 22:42:21 +01:00
|
|
|
GinDebug: true,
|
2022-12-07 22:11:44 +01:00
|
|
|
LogLevel: zerolog.DebugLevel,
|
2022-11-25 22:42:21 +01:00
|
|
|
ServerIP: "0.0.0.0",
|
|
|
|
ServerPort: "80",
|
|
|
|
DBFile: "/data/scn.sqlite3",
|
2022-12-07 22:11:44 +01:00
|
|
|
DBJournal: "WAL",
|
|
|
|
DBTimeout: 5 * time.Second,
|
2022-12-09 00:40:50 +01:00
|
|
|
DBCheckForeignKeys: false,
|
2022-12-07 22:11:44 +01:00
|
|
|
DBMaxOpenConns: 5,
|
|
|
|
DBMaxIdleConns: 5,
|
|
|
|
DBConnMaxLifetime: 60 * time.Minute,
|
|
|
|
DBConnMaxIdleTime: 60 * time.Minute,
|
2022-11-25 22:42:21 +01:00
|
|
|
RequestTimeout: 16 * time.Second,
|
|
|
|
ReturnRawErrors: true,
|
|
|
|
DummyFirebase: false,
|
|
|
|
FirebaseTokenURI: "https://oauth2.googleapis.com/token",
|
2022-12-14 18:46:26 +01:00
|
|
|
FirebaseProjectID: confEnv("SCN_FB_PROJECTID"),
|
|
|
|
FirebasePrivKeyID: confEnv("SCN_FB_PRIVATEKEYID"),
|
|
|
|
FirebaseClientMail: confEnv("SCN_FB_CLIENTEMAIL"),
|
|
|
|
FirebasePrivateKey: confEnv("SCN_FB_PRIVATEKEY"),
|
2022-11-25 22:42:21 +01:00
|
|
|
DummyGoogleAPI: false,
|
|
|
|
GoogleAPITokenURI: "https://oauth2.googleapis.com/token",
|
2022-12-14 18:46:26 +01:00
|
|
|
GoogleAPIPrivKeyID: confEnv("SCN_GOOG_PRIVATEKEYID"),
|
|
|
|
GoogleAPIClientMail: confEnv("SCN_GOOG_CLIENTEMAIL"),
|
|
|
|
GoogleAPIPrivateKey: confEnv("SCN_GOOG_PRIVATEKEY"),
|
|
|
|
GooglePackageName: confEnv("SCN_GOOG_PACKAGENAME"),
|
|
|
|
GoogleProProductID: confEnv("SCN_GOOG_PROPRODUCTID"),
|
2022-11-23 19:32:23 +01:00
|
|
|
}
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
|
|
|
|
2022-11-23 19:32:23 +01:00
|
|
|
var configStag = func() Config {
|
|
|
|
return Config{
|
2022-11-25 22:42:21 +01:00
|
|
|
Namespace: "staging",
|
2022-12-14 18:43:32 +01:00
|
|
|
BaseURL: confEnv("SCN_URL"),
|
2022-11-25 22:42:21 +01:00
|
|
|
GinDebug: true,
|
2022-12-07 22:11:44 +01:00
|
|
|
LogLevel: zerolog.DebugLevel,
|
2022-11-25 22:42:21 +01:00
|
|
|
ServerIP: "0.0.0.0",
|
|
|
|
ServerPort: "80",
|
|
|
|
DBFile: "/data/scn.sqlite3",
|
2022-12-07 22:11:44 +01:00
|
|
|
DBJournal: "WAL",
|
|
|
|
DBTimeout: 5 * time.Second,
|
2022-12-09 00:40:50 +01:00
|
|
|
DBCheckForeignKeys: false,
|
2022-12-07 22:11:44 +01:00
|
|
|
DBMaxOpenConns: 5,
|
|
|
|
DBMaxIdleConns: 5,
|
|
|
|
DBConnMaxLifetime: 60 * time.Minute,
|
|
|
|
DBConnMaxIdleTime: 60 * time.Minute,
|
2022-11-25 22:42:21 +01:00
|
|
|
RequestTimeout: 16 * time.Second,
|
|
|
|
ReturnRawErrors: true,
|
|
|
|
DummyFirebase: false,
|
|
|
|
FirebaseTokenURI: "https://oauth2.googleapis.com/token",
|
2022-12-14 18:46:26 +01:00
|
|
|
FirebaseProjectID: confEnv("SCN_FB_PROJECTID"),
|
|
|
|
FirebasePrivKeyID: confEnv("SCN_FB_PRIVATEKEYID"),
|
|
|
|
FirebaseClientMail: confEnv("SCN_FB_CLIENTEMAIL"),
|
|
|
|
FirebasePrivateKey: confEnv("SCN_FB_PRIVATEKEY"),
|
2022-11-25 22:42:21 +01:00
|
|
|
DummyGoogleAPI: false,
|
|
|
|
GoogleAPITokenURI: "https://oauth2.googleapis.com/token",
|
2022-12-14 18:46:26 +01:00
|
|
|
GoogleAPIPrivKeyID: confEnv("SCN_GOOG_PRIVATEKEYID"),
|
|
|
|
GoogleAPIClientMail: confEnv("SCN_GOOG_CLIENTEMAIL"),
|
|
|
|
GoogleAPIPrivateKey: confEnv("SCN_GOOG_PRIVATEKEY"),
|
|
|
|
GooglePackageName: confEnv("SCN_GOOG_PACKAGENAME"),
|
|
|
|
GoogleProProductID: confEnv("SCN_GOOG_PROPRODUCTID"),
|
2022-11-23 19:32:23 +01:00
|
|
|
}
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
|
|
|
|
2022-11-23 19:32:23 +01:00
|
|
|
var configProd = func() Config {
|
|
|
|
return Config{
|
2022-11-25 22:42:21 +01:00
|
|
|
Namespace: "production",
|
2022-12-07 23:32:58 +01:00
|
|
|
BaseURL: confEnv("SCN_URL"),
|
2022-11-25 22:42:21 +01:00
|
|
|
GinDebug: false,
|
2022-12-07 22:11:44 +01:00
|
|
|
LogLevel: zerolog.InfoLevel,
|
2022-11-25 22:42:21 +01:00
|
|
|
ServerIP: "0.0.0.0",
|
|
|
|
ServerPort: "80",
|
|
|
|
DBFile: "/data/scn.sqlite3",
|
2022-12-07 22:11:44 +01:00
|
|
|
DBJournal: "WAL",
|
|
|
|
DBTimeout: 5 * time.Second,
|
2022-12-09 00:40:50 +01:00
|
|
|
DBCheckForeignKeys: false,
|
2022-12-07 22:11:44 +01:00
|
|
|
DBMaxOpenConns: 5,
|
|
|
|
DBMaxIdleConns: 5,
|
|
|
|
DBConnMaxLifetime: 60 * time.Minute,
|
|
|
|
DBConnMaxIdleTime: 60 * time.Minute,
|
2022-11-25 22:42:21 +01:00
|
|
|
RequestTimeout: 16 * time.Second,
|
|
|
|
ReturnRawErrors: false,
|
|
|
|
DummyFirebase: false,
|
|
|
|
FirebaseTokenURI: "https://oauth2.googleapis.com/token",
|
2022-12-14 18:46:26 +01:00
|
|
|
FirebaseProjectID: confEnv("SCN_SCN_FB_PROJECTID"),
|
|
|
|
FirebasePrivKeyID: confEnv("SCN_SCN_FB_PRIVATEKEYID"),
|
|
|
|
FirebaseClientMail: confEnv("SCN_SCN_FB_CLIENTEMAIL"),
|
|
|
|
FirebasePrivateKey: confEnv("SCN_SCN_FB_PRIVATEKEY"),
|
2022-11-25 22:42:21 +01:00
|
|
|
DummyGoogleAPI: false,
|
|
|
|
GoogleAPITokenURI: "https://oauth2.googleapis.com/token",
|
2022-12-14 18:46:26 +01:00
|
|
|
GoogleAPIPrivKeyID: confEnv("SCN_SCN_GOOG_PRIVATEKEYID"),
|
|
|
|
GoogleAPIClientMail: confEnv("SCN_SCN_GOOG_CLIENTEMAIL"),
|
|
|
|
GoogleAPIPrivateKey: confEnv("SCN_SCN_GOOG_PRIVATEKEY"),
|
|
|
|
GooglePackageName: confEnv("SCN_SCN_GOOG_PACKAGENAME"),
|
|
|
|
GoogleProProductID: confEnv("SCN_SCN_GOOG_PROPRODUCTID"),
|
2022-11-23 19:32:23 +01:00
|
|
|
}
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
|
|
|
|
2022-11-23 19:32:23 +01:00
|
|
|
var allConfig = map[string]func() Config{
|
|
|
|
"local-host": configLocHost,
|
|
|
|
"local-docker": configLocDocker,
|
|
|
|
"develop": configDev,
|
|
|
|
"staging": configStag,
|
|
|
|
"production": configProd,
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func getConfig(ns string) (Config, bool) {
|
|
|
|
if ns == "" {
|
2022-12-07 23:32:58 +01:00
|
|
|
ns = "local-host"
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
2022-12-07 23:32:58 +01:00
|
|
|
if cfn, ok := allConfig[ns]; ok {
|
|
|
|
c := cfn()
|
|
|
|
parseConfOverride(&c)
|
|
|
|
return c, true
|
2022-11-13 19:17:07 +01:00
|
|
|
}
|
|
|
|
return Config{}, false
|
|
|
|
}
|
|
|
|
|
2022-11-21 22:52:44 +01:00
|
|
|
func confEnv(key string) string {
|
|
|
|
if v, ok := os.LookupEnv(key); ok {
|
|
|
|
return v
|
|
|
|
} else {
|
2022-11-26 17:03:26 +01:00
|
|
|
log.Fatal().Msg(fmt.Sprintf("Missing required environment variable '%s'", key))
|
|
|
|
return ""
|
2022-11-21 22:52:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-13 19:17:07 +01:00
|
|
|
func init() {
|
2022-12-14 18:46:26 +01:00
|
|
|
ns := os.Getenv("SCN_NAMESPACE")
|
2022-11-13 19:17:07 +01:00
|
|
|
|
|
|
|
cfg, ok := getConfig(ns)
|
|
|
|
if !ok {
|
|
|
|
log.Fatal().Str("ns", ns).Msg("Unknown config-namespace")
|
|
|
|
}
|
|
|
|
|
|
|
|
Conf = cfg
|
|
|
|
}
|
2022-12-07 23:32:58 +01:00
|
|
|
|
|
|
|
func parseConfOverride(c *Config) {
|
|
|
|
|
|
|
|
rval := reflect.ValueOf(c).Elem()
|
|
|
|
rtyp := rval.Type()
|
|
|
|
|
|
|
|
for i := 0; i < rtyp.NumField(); i++ {
|
|
|
|
|
|
|
|
rsfield := rtyp.Field(i)
|
|
|
|
rvfield := rval.Field(i)
|
|
|
|
|
|
|
|
envkey := rsfield.Tag.Get("env")
|
|
|
|
if envkey == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
envval, efound := os.LookupEnv(envkey)
|
|
|
|
if !efound {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if rvfield.Kind() == reflect.String {
|
|
|
|
|
|
|
|
rvfield.Set(reflect.ValueOf(envval))
|
|
|
|
|
|
|
|
fmt.Printf("[CONF] Overwrite config '%s' with '%s'\n", envkey, envval)
|
|
|
|
|
|
|
|
} else if rvfield.Type() == reflect.TypeOf(zerolog.Level(0)) {
|
|
|
|
|
|
|
|
envint, err := strconv.ParseInt(envval, 10, 8)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("Failed to parse env-config variable '%s' to int (value := '%s')", envkey, envval))
|
|
|
|
}
|
|
|
|
if envint < -1 || envint > 7 {
|
|
|
|
panic(fmt.Sprintf("Failed to parse zerolog-level (invalid number: %d)", envint))
|
|
|
|
}
|
|
|
|
|
|
|
|
lvl := zerolog.Level(envint)
|
|
|
|
|
|
|
|
rvfield.Set(reflect.ValueOf(lvl))
|
|
|
|
|
|
|
|
fmt.Printf("[CONF] Overwrite config '%s' with '%s'\n", envkey, lvl.String())
|
|
|
|
|
|
|
|
} else if rvfield.Type() == reflect.TypeOf(time.Duration(0)) {
|
|
|
|
|
|
|
|
dur, err := timeext.ParseDurationShortString(envval)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("Failed to parse env-config variable '%s' to duration (value := '%s')", envkey, envval))
|
|
|
|
}
|
|
|
|
|
|
|
|
rvfield.Set(reflect.ValueOf(dur))
|
|
|
|
|
|
|
|
fmt.Printf("[CONF] Overwrite config '%s' with '%s'\n", envkey, dur.String())
|
|
|
|
|
|
|
|
} else {
|
|
|
|
panic(fmt.Sprintf("Unknown kind/type in config: [ %s | %s ]", rvfield.Kind().String(), rvfield.Type().String()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|