From 6312afd2eb0d1dc808d600a902eb1e16b07fd9cb Mon Sep 17 00:00:00 2001 From: aeneasr <3372410+aeneasr@users.noreply.github.com> Date: Fri, 4 Mar 2022 09:17:51 +0100 Subject: [PATCH] fix(session): properly declare session secrets Previously, a misconfiguration of Gorilla's session store caused incorrect handling of the configured secrets. From now on, cookies will also be properly encrypted at all times. BREAKING CHANGE: If you are using two or more secrets for the `secrets.session`, this patch might break existing Ory Session Cookies. This has the effect that users will need to re-authenticate when visiting your app. Closes #2272 --- driver/registry_default.go | 9 ++++++++- session/manager_http_test.go | 25 +++++++++++++++++++++++++ 2 files changed, 33 insertions(+), 1 deletion(-) diff --git a/driver/registry_default.go b/driver/registry_default.go index 62a089edb233..247e59d8d4bf 100644 --- a/driver/registry_default.go +++ b/driver/registry_default.go @@ -2,6 +2,7 @@ package driver import ( "context" + "crypto/sha256" "net/http" "strings" "sync" @@ -426,7 +427,13 @@ func (m *RegistryDefault) SelfServiceErrorHandler() *errorx.Handler { } func (m *RegistryDefault) CookieManager(ctx context.Context) sessions.StoreExact { - cs := sessions.NewCookieStore(m.Config(ctx).SecretsSession()...) + var keys [][]byte + for _, k := range m.Config(ctx).SecretsSession() { + encrypt := sha256.Sum256(k) + keys = append(keys, k, encrypt[:]) + } + + cs := sessions.NewCookieStore(keys...) cs.Options.Secure = !m.Config(ctx).IsInsecureDevMode() cs.Options.HttpOnly = true diff --git a/session/manager_http_test.go b/session/manager_http_test.go index 5641534feb69..f754ee0c55ba 100644 --- a/session/manager_http_test.go +++ b/session/manager_http_test.go @@ -205,6 +205,31 @@ func TestManagerHTTP(t *testing.T) { assert.EqualValues(t, http.StatusOK, res.StatusCode) }) + t.Run("case=key rotation", func(t *testing.T) { + original := conf.Source().Strings(config.ViperKeySecretsCookie) + t.Cleanup(func() { + conf.MustSet(config.ViperKeySecretsCookie, original) + }) + conf.MustSet(config.ViperKeySessionLifespan, "1m") + conf.MustSet(config.ViperKeySecretsCookie, []string{"foo"}) + + i := identity.Identity{Traits: []byte("{}")} + require.NoError(t, reg.PrivilegedIdentityPool().CreateIdentity(context.Background(), &i)) + s, _ = session.NewActiveSession(&i, conf, time.Now(), identity.CredentialsTypePassword) + + c := testhelpers.NewClientWithCookies(t) + testhelpers.MockHydrateCookieClient(t, c, pts.URL+"/session/set") + + res, err := c.Get(pts.URL + "/session/get") + require.NoError(t, err) + assert.EqualValues(t, http.StatusOK, res.StatusCode) + + conf.MustSet(config.ViperKeySecretsCookie, []string{"bar", "foo"}) + res, err = c.Get(pts.URL + "/session/get") + require.NoError(t, err) + assert.EqualValues(t, http.StatusOK, res.StatusCode) + }) + t.Run("case=no panic on invalid cookie name", func(t *testing.T) { conf.MustSet(config.ViperKeySessionLifespan, "1m") conf.MustSet(config.ViperKeySessionName, "$%˜\"")