polycule-connect/polyculeconnect/main.go

186 lines
4.9 KiB
Go
Raw Normal View History

2023-10-12 18:36:34 +00:00
package main
import (
"context"
2023-10-14 16:06:02 +00:00
"encoding/json"
2023-10-12 18:36:34 +00:00
"flag"
"fmt"
2023-10-12 18:36:34 +00:00
"os"
"os/signal"
"time"
2023-10-14 16:06:02 +00:00
"git.faercol.me/faercol/polyculeconnect/polyculeconnect/config"
2023-10-21 19:36:50 +00:00
"git.faercol.me/faercol/polyculeconnect/polyculeconnect/connector"
2023-10-14 16:06:02 +00:00
"git.faercol.me/faercol/polyculeconnect/polyculeconnect/logger"
"git.faercol.me/faercol/polyculeconnect/polyculeconnect/server"
2023-10-22 11:22:54 +00:00
"github.com/dexidp/dex/connector/oidc"
2023-10-14 16:06:02 +00:00
dex_server "github.com/dexidp/dex/server"
"github.com/dexidp/dex/storage"
"github.com/dexidp/dex/storage/memory"
2023-10-22 11:22:54 +00:00
"github.com/dexidp/dex/storage/sql"
2023-10-14 16:06:02 +00:00
"github.com/prometheus/client_golang/prometheus"
2023-10-12 18:36:34 +00:00
)
const stopTimeout = 10 * time.Second
type cliArgs struct {
configPath string
}
func parseArgs() *cliArgs {
configPath := flag.String("config", "", "Path to the JSON configuration file")
flag.Parse()
return &cliArgs{
configPath: *configPath,
}
}
func initStorage(conf *config.AppConfig) (storage.Storage, error) {
var storageType storage.Storage
2023-10-22 11:22:54 +00:00
var err error
switch conf.StorageType {
case "memory":
storageType = memory.New(logger.L)
2023-10-22 11:22:54 +00:00
case "sqlite":
sqlconfig := sql.SQLite3{
File: conf.StorageConfig.File,
}
storageType, err = sqlconfig.Open(logger.L)
if err != nil {
logger.L.Fatalf("Failed to initialize sqlite backend: %s", err.Error())
}
default:
return storageType, fmt.Errorf("unsupported storage backend type: %s", conf.StorageType)
}
return storageType, nil
}
2023-10-22 11:22:54 +00:00
func createConnector(backend *config.BackendConfig, dexConf *dex_server.Config, connectorIDs []string) error {
for _, id := range connectorIDs {
if id == backend.ID {
return nil
}
}
backendConfJson, err := json.Marshal(backend.Config)
if err != nil {
return fmt.Errorf("failed to serialize oidc config for backend %q: %s", backend.Name, err.Error())
}
return dexConf.Storage.CreateConnector(storage.Connector{
ID: backend.ID,
Name: backend.Name,
Type: string(backend.Type),
Config: backendConfJson,
})
}
2023-10-12 18:36:34 +00:00
func main() {
args := parseArgs()
mainCtx, cancel := context.WithCancel(context.Background())
conf, err := config.New(args.configPath)
if err != nil {
panic(err)
}
logger.Init(conf.LogLevel)
logger.L.Infof("Initialized logger with level %v", conf.LogLevel)
storageType, err := initStorage(conf)
if err != nil {
logger.L.Fatalf("Failed to initialize storage backend: %s", err.Error())
}
logger.L.Infof("Initialized storage backend %q", conf.StorageType)
2023-10-14 16:06:02 +00:00
dexConf := dex_server.Config{
Web: dex_server.WebConfig{
Dir: "./",
Theme: "default",
},
Storage: storageType,
Issuer: conf.OpenConnectConfig.Issuer,
2023-10-14 16:06:02 +00:00
SupportedResponseTypes: []string{"code"},
2023-10-17 15:42:22 +00:00
SkipApprovalScreen: false,
2023-10-14 16:06:02 +00:00
AllowedOrigins: []string{"*"},
Logger: logger.L,
PrometheusRegistry: prometheus.NewRegistry(),
}
logger.L.Info("Initializing authentication backends")
2023-10-21 19:36:50 +00:00
dex_server.ConnectorsConfig["refuseAll"] = func() dex_server.ConnectorConfig { return new(connector.RefuseAllConfig) }
2023-10-22 11:22:54 +00:00
connectors, err := dexConf.Storage.ListConnectors()
if err != nil {
logger.L.Fatalf("Failed to get existing connectors: %s", err.Error())
}
var connectorIDs []string
for _, conn := range connectors {
connectorIDs = append(connectorIDs, conn.ID)
}
backend := config.BackendConfig{
Config: &oidc.Config{},
2023-10-21 19:36:50 +00:00
Name: "RefuseAll",
2023-10-22 11:22:54 +00:00
ID: "null",
2023-10-21 19:36:50 +00:00
Type: "refuseAll",
2023-10-22 11:22:54 +00:00
}
if err := createConnector(&backend, &dexConf, connectorIDs); err != nil {
2023-10-21 19:36:50 +00:00
logger.L.Errorf("Failed to add connector for backend RefuseAll to stage: %s", err.Error())
}
2023-10-14 16:06:02 +00:00
for _, backend := range conf.OpenConnectConfig.BackendConfigs {
2023-10-22 11:22:54 +00:00
if err := createConnector(backend, &dexConf, connectorIDs); err != nil {
2023-10-21 19:36:50 +00:00
logger.L.Errorf("Failed to add connector for backend %q to stage: %s", backend.Name, err.Error())
2023-10-14 16:06:02 +00:00
continue
}
}
logger.L.Info("Initializing clients")
for _, client := range conf.OpenConnectConfig.ClientConfigs {
if err := dexConf.Storage.CreateClient(*client); err != nil {
logger.L.Errorf("Failed to add client to storage: %s", err.Error())
}
}
dexSrv, err := dex_server.NewServer(mainCtx, dexConf)
if err != nil {
logger.L.Fatalf("Failed to init dex server: %s", err.Error())
}
2023-10-12 18:36:34 +00:00
logger.L.Info("Initializing server")
2023-10-14 16:06:02 +00:00
s, err := server.New(conf, dexSrv, logger.L)
2023-10-12 18:36:34 +00:00
if err != nil {
logger.L.Fatalf("Failed to initialize server: %s", err.Error())
}
go s.Run(mainCtx)
c := make(chan os.Signal, 1)
signal.Notify(c, os.Interrupt)
logger.L.Info("Application successfully started")
logger.L.Debug("Waiting for stop signal")
select {
case <-s.Done():
logger.L.Fatal("Unexpected exit from server")
case <-c:
logger.L.Info("Stopping main application")
cancel()
}
logger.L.Debugf("Waiting %v for all daemons to stop", stopTimeout)
select {
case <-time.After(stopTimeout):
logger.L.Fatalf("Failed to stop all daemons in the expected time")
case <-s.Done():
logger.L.Info("web server successfully stopped")
}
logger.L.Info("Application successfully stopped")
os.Exit(0)
}