-
Notifications
You must be signed in to change notification settings - Fork 3.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
multitenant: add SQL server startup guardrails
This code change prevents a SQL server from starting if its binary version is less than the tenant's active version. It makes `RowDecoder.DecodeRow` thread safe by making it accept a `*tree.DatumAlloc` and use a new alloc if its nil. The check runs after we run permanent upgrades because that's when the version setting is written to the settings table of the system tenant (and we want to apply this check to both system and secondary tenants). To guard against future changes to where we set the version, if GetClusterVersionFromStorage doesn't find a value for the version setting it will return an error because this means the version is not set yet. Release note: None Epic: CRDB-10829
- Loading branch information
healthy-pod
committed
Jan 23, 2023
1 parent
01032c2
commit c282dfe
Showing
9 changed files
with
188 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,117 @@ | ||
// Copyright 2023 The Cockroach Authors. | ||
// | ||
// Licensed as a CockroachDB Enterprise file under the Cockroach Community | ||
// License (the "License"); you may not use this file except in compliance with | ||
// the License. You may obtain a copy of the License at | ||
// | ||
// https://github.com/cockroachdb/cockroach/blob/master/licenses/CCL.txt | ||
|
||
package serverccl | ||
|
||
import ( | ||
"context" | ||
"testing" | ||
|
||
"github.com/cockroachdb/cockroach/pkg/base" | ||
"github.com/cockroachdb/cockroach/pkg/roachpb" | ||
"github.com/cockroachdb/cockroach/pkg/server" | ||
"github.com/cockroachdb/cockroach/pkg/settings/cluster" | ||
"github.com/cockroachdb/cockroach/pkg/testutils" | ||
"github.com/cockroachdb/cockroach/pkg/testutils/serverutils" | ||
"github.com/cockroachdb/cockroach/pkg/util/leaktest" | ||
"github.com/stretchr/testify/assert" | ||
) | ||
|
||
// TestServerStartupGuardrails ensures that a SQL server will fail to start if its binary | ||
// version (TBV) is less than the tenant's logical version (TLV). | ||
func TestServerStartupGuardrails(t *testing.T) { | ||
defer leaktest.AfterTest(t)() | ||
|
||
v := func(major, minor int32) roachpb.Version { | ||
return roachpb.Version{Major: major, Minor: minor} | ||
} | ||
|
||
var tests = []struct { | ||
storageBinaryVersion roachpb.Version | ||
storageBinaryMinSupportedVersion roachpb.Version | ||
tenantBinaryVersion roachpb.Version | ||
tenantBinaryMinSupportedVersion roachpb.Version | ||
expErrMatch string // empty if expecting a nil error | ||
}{ | ||
{ | ||
storageBinaryVersion: v(21, 2), | ||
storageBinaryMinSupportedVersion: v(21, 1), | ||
tenantBinaryVersion: v(21, 2), | ||
tenantBinaryMinSupportedVersion: v(21, 1), | ||
expErrMatch: "", | ||
}, | ||
{ | ||
storageBinaryVersion: v(21, 2), | ||
storageBinaryMinSupportedVersion: v(20, 2), | ||
tenantBinaryVersion: v(21, 1), | ||
tenantBinaryMinSupportedVersion: v(20, 2), | ||
expErrMatch: "preventing SQL server from starting because its binary version is too low for the tenant active version: " + | ||
"server binary version = 21.1, tenant active version = 21.2", | ||
}, | ||
} | ||
|
||
for i, test := range tests { | ||
storageSettings := cluster.MakeTestingClusterSettingsWithVersions( | ||
test.storageBinaryVersion, | ||
test.storageBinaryMinSupportedVersion, | ||
false, /* initializeVersion */ | ||
) | ||
|
||
// The active version of this server should be equal to its binary version. We ensure this is | ||
// true with an assertion below. This is needed because in some test cases we want to ensure | ||
// the active version of this server is greater than the binary version of the tenant. By knowing | ||
// that the Storage Binary Version is higher than the Tenant Binary Version and the Storage Logical | ||
// Version is equal to the Storage Binary Version we can be sure that the Storage Logical Version | ||
// is higher than the Tenant Binary Version. | ||
s, _, _ := serverutils.StartServer(t, base.TestServerArgs{ | ||
// Disable the default test tenant, since we create one explicitly | ||
// below. | ||
DisableDefaultTestTenant: true, | ||
Settings: storageSettings, | ||
Knobs: base.TestingKnobs{ | ||
Server: &server.TestingKnobs{ | ||
BinaryVersionOverride: test.storageBinaryVersion, | ||
DisableAutomaticVersionUpgrade: make(chan struct{}), | ||
}, | ||
}, | ||
}) | ||
|
||
// Ensure that the Storage Logical Version is equal to the Storage Binary Version. | ||
assert.True(t, s.ClusterSettings().Version.ActiveVersion(context.Background()).Version == test.storageBinaryVersion, | ||
"invalid test state: SLV not equal to SBV") | ||
|
||
tenantSettings := cluster.MakeTestingClusterSettingsWithVersions( | ||
test.tenantBinaryVersion, | ||
test.tenantBinaryMinSupportedVersion, | ||
true, /* initializeVersion */ | ||
) | ||
|
||
// The tenant will be created with an active version equal to the Storage Logical Version. | ||
tenantServer, err := s.StartTenant(context.Background(), | ||
base.TestTenantArgs{ | ||
Settings: tenantSettings, | ||
TenantID: serverutils.TestTenantID(), | ||
TestingKnobs: base.TestingKnobs{ | ||
Server: &server.TestingKnobs{ | ||
BinaryVersionOverride: test.tenantBinaryVersion, | ||
DisableAutomaticVersionUpgrade: make(chan struct{}), | ||
}, | ||
}, | ||
}) | ||
|
||
if !testutils.IsError(err, test.expErrMatch) { | ||
t.Fatalf("test %d: got error %s, wanted error matching '%s'", i, err, test.expErrMatch) | ||
} | ||
|
||
// Only attempt to stop the tenant if it was started successfully. | ||
if err == nil { | ||
tenantServer.Stopper().Stop(context.Background()) | ||
} | ||
s.Stopper().Stop(context.Background()) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters