From d3a9bf0d78b8ac6b043c9b114f494e4c30386049 Mon Sep 17 00:00:00 2001 From: Alyshan Jahani Date: Tue, 4 Apr 2023 14:03:40 -0400 Subject: [PATCH] pgwire: add server.cockroach_cloud.max_client_connections_per_gateway_reason cluster setting This setting can be used to customize the error message returned when connections are denied due to a limit specified via server.cockroach_cloud.max_client_connections_per_gateway. This functionality is required for serverless. Being able to indicate to the user why a limit is placed on their number of connections will support a better UX. Specifically, indicating to the user which resource limits they have hit. Part of: https://cockroachlabs.atlassian.net/browse/CC-9288 Release note: None --- pkg/sql/pgwire/conn.go | 9 ++++++--- pkg/sql/pgwire/conn_test.go | 20 ++++++++++++++++++++ pkg/sql/pgwire/server.go | 13 +++++++++++++ 3 files changed, 39 insertions(+), 3 deletions(-) diff --git a/pkg/sql/pgwire/conn.go b/pkg/sql/pgwire/conn.go index 426e9190b98f..06a93639e9bc 100644 --- a/pkg/sql/pgwire/conn.go +++ b/pkg/sql/pgwire/conn.go @@ -243,10 +243,13 @@ func (c *conn) checkMaxConnections(ctx context.Context, sqlServer *sql.Server) e // serverless clusters. maxNonRootConnectionsValue := maxNumNonRootConnections.Get(&sqlServer.GetExecutorConfig().Settings.SV) if maxNonRootConnectionsValue >= 0 && sqlServer.GetNonRootConnectionCount() >= maxNonRootConnectionsValue { - // TODO(alyshan): Add another cluster setting that supplements the error message, so that - // clients can know *why* this ReadOnly setting is set and limiting their connections. + // Check if there is a reason to use in the error message. + msg := "cluster connections are limited" + if reason := maxNumNonRootConnectionsReason.Get(&sqlServer.GetExecutorConfig().Settings.SV); reason != "" { + msg = reason + } return c.sendError(ctx, sqlServer.GetExecutorConfig(), errors.WithHintf( - pgerror.New(pgcode.TooManyConnections, "cluster connections are limited"), + pgerror.Newf(pgcode.TooManyConnections, "%s", msg), "the maximum number of allowed connections is %d", maxNonRootConnectionsValue, )) diff --git a/pkg/sql/pgwire/conn_test.go b/pkg/sql/pgwire/conn_test.go index 73bc61134518..b7ba76d7e266 100644 --- a/pkg/sql/pgwire/conn_test.go +++ b/pkg/sql/pgwire/conn_test.go @@ -1946,6 +1946,13 @@ func TestPGWireRejectsNewConnIfTooManyConns(t *testing.T) { require.NoError(t, err) } + setMaxNonRootConnectionsReason := func(reason string) { + conn, cleanup := openConnWithUserSuccess(rootUser) + defer cleanup() + _, err := conn.Exec(ctx, "SET CLUSTER SETTING server.cockroach_cloud.max_client_connections_per_gateway_reason = $1", reason) + require.NoError(t, err) + } + createUser := func(user string, isAdmin bool) { conn, cleanup := openConnWithUserSuccess(rootUser) defer cleanup() @@ -2053,6 +2060,19 @@ func TestPGWireRejectsNewConnIfTooManyConns(t *testing.T) { nonAdminCleanup() requireConnectionCount(t, 0) }) + + t.Run("max_non_root_connections_reason", func(t *testing.T) { + setMaxNonRootConnections(0) + setMaxNonRootConnectionsReason("foobar") + requireConnectionCount(t, 0) + _, cleanup, err := openConnWithUser(admin) + defer cleanup() + require.Error(t, err) + var pgErr *pgconn.PgError + require.ErrorAs(t, err, &pgErr) + require.Equal(t, pgcode.TooManyConnections.String(), pgErr.Code) + require.Regexp(t, "foobar", pgErr.Error()) + }) } func TestConnCloseReleasesReservedMem(t *testing.T) { diff --git a/pkg/sql/pgwire/server.go b/pkg/sql/pgwire/server.go index 682c93f16f73..ffd6ba3ce1ca 100644 --- a/pkg/sql/pgwire/server.go +++ b/pkg/sql/pgwire/server.go @@ -110,6 +110,19 @@ var maxNumNonRootConnections = settings.RegisterIntSetting( -1, ) +// maxNumNonRootConnectionsReason is used to supplement the error message for connections that denied due to +// server.max_non_root_connections_per_gateway. +// Note(alyshan): This setting is not public. It is intended to be used by Cockroach Cloud when limiting +// connections to serverless clusters. +// This setting may be removed one day. +var maxNumNonRootConnectionsReason = settings.RegisterStringSetting( + settings.TenantWritable, + "server.cockroach_cloud.max_client_connections_per_gateway_reason", + "a reason to provide in the error message for connections that are denied due to "+ + "server.cockroach_cloud.max_client_connections_per_gateway", + "", +) + const ( // ErrSSLRequired is returned when a client attempts to connect to a // secure server in cleartext.