Compare commits

2 Commits

Author SHA1 Message Date
c787fa929b fix: config for backend;
refactor: docker compose env variables
2025-06-18 21:27:19 +03:00
7a79028aa7 refactor: config 2025-06-18 20:17:04 +03:00
3 changed files with 48 additions and 60 deletions

View File

@@ -14,7 +14,7 @@ import (
func main() {
if err := config.LoadConfig(); err != nil {
if _, err := config.Load(); err != nil {
panic(err)
}

View File

@@ -1,9 +1,8 @@
package config
import (
"log"
"reflect"
"fmt"
"strings"
"github.com/spf13/viper"
)
@@ -11,81 +10,66 @@ import (
type Config struct {
Hostname string `mapstructure:"HOSTNAME"`
Port string `mapstructure:"PORT"`
DatabaseUrl string `mapstructure:"DATABASE_URL"`
DatabaseUrl string `mapstructure:"POSTGRES_URL"`
RedisUrl string `mapstructure:"REDIS_URL"`
MinioUrl string `mapstructure:"MINIO_URL"`
JwtAlgorithm string `mapstructure:"JWT_ALGORITHM"`
JwtSecret string `mapstructure:"JWT_SECRET"`
JwtIssuer string `mapstructure:"JWT_ISSUER"`
JwtAudience string `mapstructure:"JWT_AUDIENCE"`
JwtExpAccess string `mapstructure:"JWT_EXP_ACCESS"`
JwtExpRefresh string `mapstructure:"JWT_EXP_REFRESH"`
Environment string `mapstructure:"ENVIRONMENT"`
}
var config *Config
func Load() (*Config, error) {
func LoadConfig() error {
// Load .env file
if err := viper.ReadInConfig(); err != nil {
log.Printf("Error reading config file, proceeding with environment variables. %s", err)
}
// Set default parameters
viper.SetDefault("HOSTNAME", "localhost")
viper.SetDefault("PORT", "8080")
viper.SetDefault("DATABASE_URL", "mydb")
viper.SetDefault("REDIS_URL", "myredis")
viper.SetDefault("MINIO_URL", "myminio")
viper.SetDefault("JWT_ALGORITHM", "HS256")
viper.SetDefault("JWT_SECRET", "default_jwt_secret_please_change") // TODO: remove and randomly generate
viper.SetDefault("JWT_SECRET", "default_jwt_secret_please_change")
viper.SetDefault("JWT_EXP_ACCESS", "5m")
viper.SetDefault("JWT_EXP_REFRESH", "1w")
viper.SetDefault("JWT_AUDIENCE", "easywish")
viper.SetDefault("JWT_ISSUER", "easywish")
viper.SetDefault("ENVIRONMENT", "production")
// Set the file name and type for Viper
viper.SetConfigName(".env") // name of config file (without extension)
viper.SetConfigType("env") // REQUIRED if the config file does not have the extension
viper.AddConfigPath(".") // optionally look for config in the working directory
viper.AutomaticEnv()
// Unmarshal the configuration into the Config struct
if err := viper.Unmarshal(&config); err != nil {
log.Fatalf("Unable to decode into struct, %v", err)
required := []string{
"POSTGRES_URL",
"REDIS_URL",
"MINIO_URL",
}
// Perform validation
if err := validateConfig(); err != nil {
return err
}
return nil
}
func validateConfig() error {
v := reflect.ValueOf(*config)
t := v.Type()
for i := range v.NumField() {
field := v.Field(i)
fieldType := t.Field(i)
// Check if the field is a string and is empty
if field.Kind() == reflect.String && field.String() == "" {
return fmt.Errorf("Missing required configuration: %s", fieldType.Name)
var missing []string
for _, key := range required {
if !viper.IsSet(key) {
missing = append(missing, key)
}
}
if len(missing) > 0 {
return nil, fmt.Errorf("missing required environment variables: %s", strings.Join(missing, ", "))
}
return nil
var cfg Config
if err := viper.Unmarshal(&cfg); err != nil {
return nil, fmt.Errorf("failed to unmarshal config: %w", err)
}
config = &cfg
return &cfg, nil
}
func GetConfig() *Config {
if config == nil {
if _, err := Load(); err != nil {
panic(err)
}
}
return config
}
var config *Config

View File

@@ -15,6 +15,10 @@ services:
interval: 5s
timeout: 10s
retries: 3
environment:
POSTGRES_URL: ${POSTGRES_URL}
REDIS_URL: ${REDIS_URL}
MINIO_URL: ${MINIO_URL}
ports:
- "8080:8080"
networks: