2012-12-10 20:59:23 -05:00
package main
import (
2021-01-07 11:52:50 +00:00
"context"
2012-12-17 13:38:33 -05:00
"fmt"
2019-03-20 06:44:51 -07:00
"math/rand"
2021-01-07 11:52:50 +00:00
"net/http"
2014-08-07 16:16:39 -04:00
"os"
2020-04-05 00:12:38 +09:00
"os/signal"
2015-03-19 23:03:00 -04:00
"runtime"
2020-04-05 00:12:38 +09:00
"syscall"
2014-11-08 13:26:55 -05:00
"time"
2012-12-10 20:59:23 -05:00
2020-11-09 20:34:55 +00:00
"github.com/ghodss/yaml"
2020-09-30 01:44:42 +09:00
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/options"
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/logger"
2021-01-07 11:52:50 +00:00
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/middleware"
2020-09-30 01:44:42 +09:00
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/validation"
2020-11-09 20:17:43 +00:00
"github.com/spf13/pflag"
2012-12-10 20:59:23 -05:00
)
func main ( ) {
2019-02-10 08:37:45 -08:00
logger . SetFlags ( logger . Lshortfile )
2014-11-09 14:51:10 -05:00
2020-11-09 20:17:43 +00:00
configFlagSet := pflag . NewFlagSet ( "oauth2-proxy" , pflag . ContinueOnError )
2021-02-06 15:21:12 +00:00
// Because we parse early to determine alpha vs legacy config, we have to
// ignore any unknown flags for now
configFlagSet . ParseErrorsWhitelist . UnknownFlags = true
2020-11-09 20:17:43 +00:00
config := configFlagSet . String ( "config" , "" , "path to config file" )
alphaConfig := configFlagSet . String ( "alpha-config" , "" , "path to alpha config file (use at your own risk - the structure in this config file may change between minor releases)" )
2020-11-09 20:34:55 +00:00
convertConfig := configFlagSet . Bool ( "convert-config-to-alpha" , false , "if true, the proxy will load configuration as normal and convert existing configuration to the alpha config structure, and print it to stdout" )
2020-11-09 20:17:43 +00:00
showVersion := configFlagSet . Bool ( "version" , false , "print version string" )
configFlagSet . Parse ( os . Args [ 1 : ] )
2014-11-09 14:51:10 -05:00
2014-11-09 21:07:02 -05:00
if * showVersion {
2020-03-29 14:54:36 +01:00
fmt . Printf ( "oauth2-proxy %s (built with %s)\n" , VERSION , runtime . Version ( ) )
2014-11-09 21:07:02 -05:00
return
}
2020-11-09 20:34:55 +00:00
if * convertConfig && * alphaConfig != "" {
logger . Fatal ( "cannot use alpha-config and conver-config-to-alpha together" )
}
2020-11-09 20:17:43 +00:00
opts , err := loadConfiguration ( * config , * alphaConfig , configFlagSet , os . Args [ 1 : ] )
2020-05-26 20:06:27 +01:00
if err != nil {
2021-01-07 11:52:50 +00:00
logger . Fatalf ( "ERROR: %v" , err )
2020-05-26 20:06:27 +01:00
}
2020-11-09 20:34:55 +00:00
if * convertConfig {
if err := printConvertedConfig ( opts ) ; err != nil {
logger . Fatalf ( "ERROR: could not convert config: %v" , err )
}
return
}
2021-01-07 11:52:50 +00:00
if err = validation . Validate ( opts ) ; err != nil {
logger . Fatalf ( "%s" , err )
2012-12-10 20:59:23 -05:00
}
2019-02-10 08:37:45 -08:00
2015-06-06 14:37:54 -04:00
validator := NewValidator ( opts . EmailDomains , opts . AuthenticatedEmailsFile )
2020-05-25 14:00:49 +01:00
oauthproxy , err := NewOAuthProxy ( opts , validator )
if err != nil {
2021-01-07 11:52:50 +00:00
logger . Fatalf ( "ERROR: Failed to initialise OAuth2 Proxy: %v" , err )
2020-05-25 14:00:49 +01:00
}
2014-11-09 14:51:10 -05:00
2019-03-20 06:44:51 -07:00
rand . Seed ( time . Now ( ) . UnixNano ( ) )
2021-01-07 11:52:50 +00:00
oauthProxyStop := make ( chan struct { } , 1 )
metricsStop := startMetricsServer ( opts . MetricsAddress , oauthProxyStop )
2015-06-07 21:51:47 -04:00
s := & Server {
2020-07-29 20:10:14 +01:00
Handler : oauthproxy ,
2015-06-07 21:51:47 -04:00
Opts : opts ,
2021-01-07 11:52:50 +00:00
stop : oauthProxyStop ,
2015-02-11 12:07:40 +11:00
}
2020-04-05 00:12:38 +09:00
// Observe signals in background goroutine.
go func ( ) {
sigint := make ( chan os . Signal , 1 )
signal . Notify ( sigint , os . Interrupt , syscall . SIGTERM )
<- sigint
2021-01-07 11:52:50 +00:00
s . stop <- struct { } { } // notify having caught signal stop oauthproxy
close ( metricsStop ) // and the metrics endpoint
2020-04-05 00:12:38 +09:00
} ( )
2015-06-07 21:51:47 -04:00
s . ListenAndServe ( )
2012-12-10 20:59:23 -05:00
}
2020-11-09 20:17:43 +00:00
2021-01-07 11:52:50 +00:00
// startMetricsServer will start the metrics server on the specified address.
// It always return a channel to signal stop even when it does not run.
func startMetricsServer ( address string , oauthProxyStop chan struct { } ) chan struct { } {
stop := make ( chan struct { } , 1 )
// Attempt to setup the metrics endpoint if we have an address
if address != "" {
s := & http . Server { Addr : address , Handler : middleware . DefaultMetricsHandler }
go func ( ) {
// ListenAndServe always returns a non-nil error. After Shutdown or
// Close, the returned error is ErrServerClosed
if err := s . ListenAndServe ( ) ; err != http . ErrServerClosed {
logger . Println ( err )
// Stop the metrics shutdown go routine
close ( stop )
// Stop the oauthproxy server, we have encounter an unexpected error
close ( oauthProxyStop )
}
} ( )
go func ( ) {
<- stop
if err := s . Shutdown ( context . Background ( ) ) ; err != nil {
logger . Print ( err )
}
} ( )
}
return stop
}
2020-11-09 20:17:43 +00:00
// loadConfiguration will load in the user's configuration.
// It will either load the alpha configuration (if alphaConfig is given)
// or the legacy configuration.
func loadConfiguration ( config , alphaConfig string , extraFlags * pflag . FlagSet , args [ ] string ) ( * options . Options , error ) {
if alphaConfig != "" {
logger . Printf ( "WARNING: You are using alpha configuration. The structure in this configuration file may change without notice. You MUST remove conflicting options from your existing configuration." )
return loadAlphaOptions ( config , alphaConfig , extraFlags , args )
}
return loadLegacyOptions ( config , extraFlags , args )
}
// loadLegacyOptions loads the old toml options using the legacy flagset
// and legacy options struct.
func loadLegacyOptions ( config string , extraFlags * pflag . FlagSet , args [ ] string ) ( * options . Options , error ) {
optionsFlagSet := options . NewLegacyFlagSet ( )
optionsFlagSet . AddFlagSet ( extraFlags )
if err := optionsFlagSet . Parse ( args ) ; err != nil {
return nil , fmt . Errorf ( "failed to parse flags: %v" , err )
}
legacyOpts := options . NewLegacyOptions ( )
if err := options . Load ( config , optionsFlagSet , legacyOpts ) ; err != nil {
return nil , fmt . Errorf ( "failed to load config: %v" , err )
}
opts , err := legacyOpts . ToOptions ( )
if err != nil {
return nil , fmt . Errorf ( "failed to convert config: %v" , err )
}
return opts , nil
}
// loadAlphaOptions loads the old style config excluding options converted to
// the new alpha format, then merges the alpha options, loaded from YAML,
// into the core configuration.
func loadAlphaOptions ( config , alphaConfig string , extraFlags * pflag . FlagSet , args [ ] string ) ( * options . Options , error ) {
opts , err := loadOptions ( config , extraFlags , args )
if err != nil {
return nil , fmt . Errorf ( "failed to load core options: %v" , err )
}
alphaOpts := & options . AlphaOptions { }
if err := options . LoadYAML ( alphaConfig , alphaOpts ) ; err != nil {
return nil , fmt . Errorf ( "failed to load alpha options: %v" , err )
}
alphaOpts . MergeInto ( opts )
return opts , nil
}
// loadOptions loads the configuration using the old style format into the
// core options.Options struct.
// This means that none of the options that have been converted to alpha config
// will be loaded using this method.
func loadOptions ( config string , extraFlags * pflag . FlagSet , args [ ] string ) ( * options . Options , error ) {
optionsFlagSet := options . NewFlagSet ( )
optionsFlagSet . AddFlagSet ( extraFlags )
if err := optionsFlagSet . Parse ( args ) ; err != nil {
return nil , fmt . Errorf ( "failed to parse flags: %v" , err )
}
opts := options . NewOptions ( )
if err := options . Load ( config , optionsFlagSet , opts ) ; err != nil {
return nil , fmt . Errorf ( "failed to load config: %v" , err )
}
return opts , nil
}
2020-11-09 20:34:55 +00:00
// printConvertedConfig extracts alpha options from the loaded configuration
// and renders these to stdout in YAML format.
func printConvertedConfig ( opts * options . Options ) error {
alphaConfig := & options . AlphaOptions { }
alphaConfig . ExtractFrom ( opts )
data , err := yaml . Marshal ( alphaConfig )
if err != nil {
return fmt . Errorf ( "unable to marshal config: %v" , err )
}
if _ , err := os . Stdout . Write ( data ) ; err != nil {
return fmt . Errorf ( "unable to write output: %v" , err )
}
return nil
}