mirror of
https://github.com/oauth2-proxy/oauth2-proxy.git
synced 2025-01-10 04:18:14 +02:00
7eeaea0b3f
* Set and verify a nonce with OIDC * Create a CSRF object to manage nonces & cookies * Add missing generic cookie unit tests * Add config flag to control OIDC SkipNonce * Send hashed nonces in authentication requests * Encrypt the CSRF cookie * Add clarity to naming & add more helper methods * Make CSRF an interface and keep underlying nonces private * Add ReverseProxy scope to cookie tests * Align to new 1.16 SameSite cookie default * Perform SecretBytes conversion on CSRF cookie crypto * Make state encoding signatures consistent * Mock time in CSRF struct via Clock * Improve InsecureSkipNonce docstring
89 lines
2.6 KiB
Go
89 lines
2.6 KiB
Go
package validation
|
|
|
|
import (
|
|
"context"
|
|
"encoding/base64"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/apis/options"
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/encryption"
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/sessions/redis"
|
|
)
|
|
|
|
func validateSessionCookieMinimal(o *options.Options) []string {
|
|
if !o.Session.Cookie.Minimal {
|
|
return []string{}
|
|
}
|
|
|
|
msgs := []string{}
|
|
for _, header := range append(o.InjectRequestHeaders, o.InjectResponseHeaders...) {
|
|
for _, value := range header.Values {
|
|
if value.ClaimSource != nil {
|
|
if value.ClaimSource.Claim == "access_token" {
|
|
msgs = append(msgs,
|
|
fmt.Sprintf("access_token claim for header %q requires oauth tokens in sessions. session_cookie_minimal cannot be set", header.Name))
|
|
}
|
|
if value.ClaimSource.Claim == "id_token" {
|
|
msgs = append(msgs,
|
|
fmt.Sprintf("id_token claim for header %q requires oauth tokens in sessions. session_cookie_minimal cannot be set", header.Name))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if o.Cookie.Refresh != time.Duration(0) {
|
|
msgs = append(msgs,
|
|
"cookie_refresh > 0 requires oauth tokens in sessions. session_cookie_minimal cannot be set")
|
|
}
|
|
return msgs
|
|
}
|
|
|
|
// validateRedisSessionStore builds a Redis Client from the options and
|
|
// attempts to connect, Set, Get and Del a random health check key
|
|
func validateRedisSessionStore(o *options.Options) []string {
|
|
if o.Session.Type != options.RedisSessionStoreType {
|
|
return []string{}
|
|
}
|
|
|
|
client, err := redis.NewRedisClient(o.Session.Redis)
|
|
if err != nil {
|
|
return []string{fmt.Sprintf("unable to initialize a redis client: %v", err)}
|
|
}
|
|
|
|
n, err := encryption.Nonce()
|
|
if err != nil {
|
|
return []string{fmt.Sprintf("unable to generate a redis initialization test key: %v", err)}
|
|
}
|
|
nonce := base64.RawURLEncoding.EncodeToString(n)
|
|
|
|
key := fmt.Sprintf("%s-healthcheck-%s", o.Cookie.Name, nonce)
|
|
return sendRedisConnectionTest(client, key, nonce)
|
|
}
|
|
|
|
func sendRedisConnectionTest(client redis.Client, key string, val string) []string {
|
|
msgs := []string{}
|
|
ctx := context.Background()
|
|
|
|
err := client.Set(ctx, key, []byte(val), time.Duration(60)*time.Second)
|
|
if err != nil {
|
|
msgs = append(msgs, fmt.Sprintf("unable to set a redis initialization key: %v", err))
|
|
} else {
|
|
gval, err := client.Get(ctx, key)
|
|
if err != nil {
|
|
msgs = append(msgs,
|
|
fmt.Sprintf("unable to retrieve redis initialization key: %v", err))
|
|
}
|
|
if string(gval) != val {
|
|
msgs = append(msgs,
|
|
"the retrieved redis initialization key did not match the value we set")
|
|
}
|
|
}
|
|
|
|
err = client.Del(ctx, key)
|
|
if err != nil {
|
|
msgs = append(msgs, fmt.Sprintf("unable to delete the redis initialization key: %v", err))
|
|
}
|
|
return msgs
|
|
}
|