From 2815b06021ce0bec23f799d170504d2ca3517969 Mon Sep 17 00:00:00 2001 From: Herko Lategan Date: Tue, 5 Dec 2023 12:36:39 +0000 Subject: [PATCH 1/6] roachtest: add sql instance to functional options Epic: None Release Note: None --- pkg/cmd/roachtest/option/connection_options.go | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/pkg/cmd/roachtest/option/connection_options.go b/pkg/cmd/roachtest/option/connection_options.go index 315f0d608086..6528bdd58c0a 100644 --- a/pkg/cmd/roachtest/option/connection_options.go +++ b/pkg/cmd/roachtest/option/connection_options.go @@ -35,6 +35,12 @@ func TenantName(tenantName string) func(*ConnOption) { } } +func SQLInstance(sqlInstance int) func(*ConnOption) { + return func(option *ConnOption) { + option.SQLInstance = sqlInstance + } +} + func ConnectionOption(key, value string) func(*ConnOption) { return func(option *ConnOption) { if len(option.Options) == 0 { From 2c19543d994a41503bf581c769c8b439710b40a6 Mon Sep 17 00:00:00 2001 From: Herko Lategan Date: Tue, 5 Dec 2023 12:56:39 +0000 Subject: [PATCH 2/6] roachtest: add API to start virtual clusters Previously the cluster interface only exposed a method to start storage nodes, but that is insufficient to start virtual clusters that have a separate method on the `roachprod` API (for starting). This change adds a new method `StartServiceForVirtualCluster` to the cluster interface to enable roachtests to start virtual clusters. Some refactoring was required to enable different sets of cluster settings, depending on what service type is going to be started. There are now two sets of cluster settings that can be utilised in `test_runner`. For virtual clusters `virtualClusterSettings` will be used, and for storage clusters `clusterSettings` will be utilised. Epic: None Release Note: None --- pkg/cmd/roachtest/cluster.go | 103 +++++++++++++----- .../roachtest/cluster/cluster_interface.go | 5 + pkg/cmd/roachtest/test_runner.go | 1 + pkg/roachprod/install/cockroach.go | 2 + 4 files changed, 84 insertions(+), 27 deletions(-) diff --git a/pkg/cmd/roachtest/cluster.go b/pkg/cmd/roachtest/cluster.go index 20caeb8283b8..e3ebb2a6a7b9 100644 --- a/pkg/cmd/roachtest/cluster.go +++ b/pkg/cmd/roachtest/cluster.go @@ -639,8 +639,11 @@ type clusterImpl struct { expiration time.Time encAtRest bool // use encryption at rest - // clusterSettings are additional cluster settings set on cluster startup. + // clusterSettings are additional cluster settings set on the storage cluster startup. clusterSettings map[string]string + // virtualClusterSettings are additional cluster settings to set on the + // virtual cluster startup. + virtualClusterSettings map[string]string // goCoverDir is the directory for Go coverage data (if coverage is enabled). // BAZEL_COVER_DIR will be set to this value when starting a node. goCoverDir string @@ -1910,6 +1913,38 @@ func (c *clusterImpl) clearStatusForClusterOpt(worker bool) { } } +func (c *clusterImpl) configureClusterSettingOptions( + defaultClusterSettings install.ClusterSettingsOption, settings install.ClusterSettings, +) []install.ClusterSettingOption { + setUnlessExists := func(name string, value interface{}) { + if !envExists(settings.Env, name) { + settings.Env = append(settings.Env, fmt.Sprintf("%s=%s", name, fmt.Sprint(value))) + } + } + // Set the same seed on every node, to be used by builds with + // runtime assertions enabled. + setUnlessExists("COCKROACH_RANDOM_SEED", c.cockroachRandomSeed()) + + // Panic on span use-after-Finish, so we catch such bugs. + setUnlessExists("COCKROACH_CRASH_ON_SPAN_USE_AFTER_FINISH", true) + + if c.goCoverDir != "" { + settings.Env = append(settings.Env, fmt.Sprintf("BAZEL_COVER_DIR=%s", c.goCoverDir)) + } + + return []install.ClusterSettingOption{ + install.TagOption(settings.Tag), + install.PGUrlCertsDirOption(settings.PGUrlCertsDir), + install.SecureOption(settings.Secure), + install.UseTreeDistOption(settings.UseTreeDist), + install.EnvOption(settings.Env), + install.NumRacksOption(settings.NumRacks), + install.BinaryOption(settings.Binary), + defaultClusterSettings, + install.ClusterSettingsOption(settings.ClusterSettings), + } +} + // StartE starts cockroach nodes on a subset of the cluster. The nodes parameter // can either be a specific node, empty (to indicate all nodes), or a pair of // nodes indicating a range. @@ -1928,17 +1963,6 @@ func (c *clusterImpl) StartE( startOpts.RoachprodOpts.EncryptedStores = c.encAtRest - setUnlessExists := func(name string, value interface{}) { - if !envExists(settings.Env, name) { - settings.Env = append(settings.Env, fmt.Sprintf("%s=%s", name, fmt.Sprint(value))) - } - } - // Panic on span use-after-Finish, so we catch such bugs. - setUnlessExists("COCKROACH_CRASH_ON_SPAN_USE_AFTER_FINISH", true) - // Set the same seed on every node, to be used by builds with - // runtime assertions enabled. - setUnlessExists("COCKROACH_RANDOM_SEED", c.cockroachRandomSeed()) - // Needed for backward-compat on crdb_internal.ranges{_no_leases}. // Remove in v23.2. if !envExists(settings.Env, "COCKROACH_FORCE_DEPRECATED_SHOW_RANGE_BEHAVIOR") { @@ -1947,21 +1971,7 @@ func (c *clusterImpl) StartE( settings.Env = append(settings.Env, "COCKROACH_FORCE_DEPRECATED_SHOW_RANGE_BEHAVIOR=false") } - if c.goCoverDir != "" { - settings.Env = append(settings.Env, fmt.Sprintf("BAZEL_COVER_DIR=%s", c.goCoverDir)) - } - - clusterSettingsOpts := []install.ClusterSettingOption{ - install.TagOption(settings.Tag), - install.PGUrlCertsDirOption(settings.PGUrlCertsDir), - install.SecureOption(settings.Secure), - install.UseTreeDistOption(settings.UseTreeDist), - install.EnvOption(settings.Env), - install.NumRacksOption(settings.NumRacks), - install.BinaryOption(settings.Binary), - install.ClusterSettingsOption(c.clusterSettings), - install.ClusterSettingsOption(settings.ClusterSettings), - } + clusterSettingsOpts := c.configureClusterSettingOptions(c.clusterSettings, settings) if err := roachprod.Start(ctx, l, c.MakeNodes(opts...), startOpts.RoachprodOpts, clusterSettingsOpts...); err != nil { return err @@ -1977,6 +1987,45 @@ func (c *clusterImpl) StartE( return nil } +func (c *clusterImpl) StartServiceForVirtualClusterE( + ctx context.Context, + l *logger.Logger, + externalNodes option.NodeListOption, + startOpts option.StartOpts, + settings install.ClusterSettings, + opts ...option.Option, +) error { + + c.setStatusForClusterOpt("starting virtual cluster", startOpts.RoachtestOpts.Worker, opts...) + defer c.clearStatusForClusterOpt(startOpts.RoachtestOpts.Worker) + + clusterSettingsOpts := c.configureClusterSettingOptions(c.virtualClusterSettings, settings) + + if err := roachprod.StartServiceForVirtualCluster(ctx, l, c.MakeNodes(externalNodes), c.MakeNodes(opts...), startOpts.RoachprodOpts, clusterSettingsOpts...); err != nil { + return err + } + + if settings.Secure { + if err := c.RefetchCertsFromNode(ctx, 1); err != nil { + return err + } + } + return nil +} + +func (c *clusterImpl) StartServiceForVirtualCluster( + ctx context.Context, + l *logger.Logger, + externalNodes option.NodeListOption, + startOpts option.StartOpts, + settings install.ClusterSettings, + opts ...option.Option, +) { + if err := c.StartServiceForVirtualClusterE(ctx, l, externalNodes, startOpts, settings, opts...); err != nil { + c.t.Fatal(err) + } +} + func (c *clusterImpl) RefetchCertsFromNode(ctx context.Context, node int) error { var err error c.localCertsDir, err = os.MkdirTemp("", "roachtest-certs") diff --git a/pkg/cmd/roachtest/cluster/cluster_interface.go b/pkg/cmd/roachtest/cluster/cluster_interface.go index f1337049103d..2e7bce895d78 100644 --- a/pkg/cmd/roachtest/cluster/cluster_interface.go +++ b/pkg/cmd/roachtest/cluster/cluster_interface.go @@ -64,6 +64,11 @@ type Cluster interface { StopCockroachGracefullyOnNode(ctx context.Context, l *logger.Logger, node int) error NewMonitor(context.Context, ...option.Option) Monitor + // Starting virtual clusters. + + StartServiceForVirtualClusterE(ctx context.Context, l *logger.Logger, externalNodes option.NodeListOption, startOpts option.StartOpts, settings install.ClusterSettings, opts ...option.Option) error + StartServiceForVirtualCluster(ctx context.Context, l *logger.Logger, externalNodes option.NodeListOption, startOpts option.StartOpts, settings install.ClusterSettings, opts ...option.Option) + // Hostnames and IP addresses of the nodes. InternalAddr(ctx context.Context, l *logger.Logger, node option.NodeListOption) ([]string, error) diff --git a/pkg/cmd/roachtest/test_runner.go b/pkg/cmd/roachtest/test_runner.go index ec634a641eb6..44da53560a65 100644 --- a/pkg/cmd/roachtest/test_runner.go +++ b/pkg/cmd/roachtest/test_runner.go @@ -787,6 +787,7 @@ func (r *testRunner) runWorker( // Set initial cluster settings for this test. c.clusterSettings = map[string]string{} + c.virtualClusterSettings = map[string]string{} switch testSpec.Leases { case registry.DefaultLeases: diff --git a/pkg/roachprod/install/cockroach.go b/pkg/roachprod/install/cockroach.go index d6fa592bb736..abc4d27ed18c 100644 --- a/pkg/roachprod/install/cockroach.go +++ b/pkg/roachprod/install/cockroach.go @@ -162,7 +162,9 @@ const ( // StartRoutingProxy starts the SQL proxy process to route // connections to multiple virtual clusters. StartRoutingProxy +) +const ( // startSQLTimeout identifies the COCKROACH_CONNECT_TIMEOUT to use (in seconds) // for sql cmds within syncedCluster.Start(). startSQLTimeout = 1200 From 77fee697e17cc286d96ae14f6e9511aa465bbdca Mon Sep 17 00:00:00 2001 From: Herko Lategan Date: Tue, 5 Dec 2023 12:59:16 +0000 Subject: [PATCH 3/6] roachtest: update multitenant/distsql to use new roachprod service APIs Previously the multitenant distsql roachtest relied on an internal util in `roachtest` to start virtual clusters. This change updates the test to use the new official `roachtest` and `roachprod` APIs for starting virtual clusters. Fixes: #116019 Epic: None Release Note: None --- .../roachtest/tests/multitenant_distsql.go | 86 +++++++------------ 1 file changed, 31 insertions(+), 55 deletions(-) diff --git a/pkg/cmd/roachtest/tests/multitenant_distsql.go b/pkg/cmd/roachtest/tests/multitenant_distsql.go index 658607388af7..80c8242b3c98 100644 --- a/pkg/cmd/roachtest/tests/multitenant_distsql.go +++ b/pkg/cmd/roachtest/tests/multitenant_distsql.go @@ -13,7 +13,6 @@ package tests import ( "archive/zip" "context" - gosql "database/sql" "fmt" "io" "strconv" @@ -64,70 +63,46 @@ func runMultiTenantDistSQL( // 1 byte to bypass the guardrails. settings := install.MakeClusterSettings(install.SecureOption(true)) settings.Env = append(settings.Env, "COCKROACH_MIN_RANGE_MAX_BYTES=1") - tenantEnvOpt := createTenantEnvVar(settings.Env[len(settings.Env)-1]) c.Start(ctx, t.L(), option.DefaultStartOpts(), settings, c.Node(1)) c.Start(ctx, t.L(), option.DefaultStartOpts(), settings, c.Node(2)) c.Start(ctx, t.L(), option.DefaultStartOpts(), settings, c.Node(3)) + storageNodes := c.Range(1, 3) - const ( - tenantID = 11 - tenantBaseHTTPPort = 8081 - tenantBaseSQLPort = 26259 - // localPortOffset is used to avoid port conflicts with nodes on a local - // cluster. - localPortOffset = 1000 - ) - - tenantHTTPPort := func(offset int) int { - if c.IsLocal() || numInstances > c.Spec().NodeCount { - return tenantBaseHTTPPort + localPortOffset + offset - } - return tenantBaseHTTPPort - } - tenantSQLPort := func(offset int) int { - if c.IsLocal() || numInstances > c.Spec().NodeCount { - return tenantBaseSQLPort + localPortOffset + offset - } - return tenantBaseSQLPort + tenantName := "test-tenant" + var nodes intsets.Fast + for i := 0; i < numInstances; i++ { + node := (i % c.Spec().NodeCount) + 1 + sqlInstance := i / c.Spec().NodeCount + instStartOps := option.DefaultStartOpts() + instStartOps.RoachprodOpts.Target = install.StartServiceForVirtualCluster + instStartOps.RoachprodOpts.VirtualClusterName = tenantName + instStartOps.RoachprodOpts.SQLInstance = sqlInstance + // We set the ports to 0 so that ports are assigned dynamically. This is a + // temporary workaround until we use dynamic port assignment as the default. + // See: https://github.com/cockroachdb/cockroach/issues/111052 + // TODO(herko): remove this once dynamic port assignment is the default. + instStartOps.RoachprodOpts.SQLPort = 0 + instStartOps.RoachprodOpts.AdminUIPort = 0 + + t.L().Printf("Starting instance %d on node %d", i, node) + c.StartServiceForVirtualCluster(ctx, t.L(), c.Node(node), instStartOps, settings, storageNodes) + nodes.Add(i + 1) } storConn := c.Conn(ctx, t.L(), 1) - _, err := storConn.Exec(`SELECT crdb_internal.create_tenant($1::INT)`, tenantID) + // Open things up, so we can configure range sizes below. + _, err := storConn.Exec(`ALTER TENANT $1 SET CLUSTER SETTING sql.zone_configs.allow_for_secondary_tenant.enabled = true`, tenantName) require.NoError(t, err) - instances := make([]*tenantNode, 0, numInstances) - instance1 := createTenantNode(ctx, t, c, c.Node(1), tenantID, 2 /* node */, tenantHTTPPort(0), tenantSQLPort(0), - createTenantCertNodes(c.All()), tenantEnvOpt) - instances = append(instances, instance1) - defer instance1.stop(ctx, t, c) - instance1.start(ctx, t, c, "./cockroach") - - // Open things up so we can configure range sizes below. - _, err = storConn.Exec(`ALTER TENANT [$1] SET CLUSTER SETTING sql.zone_configs.allow_for_secondary_tenant.enabled = true`, tenantID) - require.NoError(t, err) - - // Create numInstances sql pods and spread them evenly across the machines. - var nodes intsets.Fast - nodes.Add(1) - for i := 1; i < numInstances; i++ { - node := ((i + 1) % c.Spec().NodeCount) + 1 - inst, err := newTenantInstance(ctx, instance1, t, c, node, tenantHTTPPort(i), tenantSQLPort(i)) - instances = append(instances, inst) - require.NoError(t, err) - defer inst.stop(ctx, t, c) - inst.start(ctx, t, c, "./cockroach") - nodes.Add(i + 1) - } - m := c.NewMonitor(ctx, c.Nodes(1, 2, 3)) - inst1Conn, err := gosql.Open("postgres", instance1.pgURL) + inst1Conn, err := c.ConnE(ctx, t.L(), 1, option.TenantName(tenantName)) require.NoError(t, err) _, err = inst1Conn.Exec("CREATE TABLE t(n INT, i INT,s STRING, PRIMARY KEY(n,i))") require.NoError(t, err) // DistSQL needs at least a range per node to distribute query everywhere - // and test takes too long and too much resources with default range sizes + // and test takes too long and too many resources with default range sizes // so make them much smaller. _, err = inst1Conn.Exec(`ALTER TABLE t CONFIGURE ZONE USING range_min_bytes = 1000,range_max_bytes = 100000`) require.NoError(t, err) @@ -135,11 +110,12 @@ func runMultiTenantDistSQL( insertCtx, cancel := context.WithCancel(ctx) defer cancel() - for i, inst := range instances { - url := inst.pgURL + for i := 0; i < numInstances; i++ { li := i m.Go(func(ctx context.Context) error { - dbi, err := gosql.Open("postgres", url) + node := (li % c.Spec().NodeCount) + 1 + sqlInstance := li / c.Spec().NodeCount + dbi, err := c.ConnE(ctx, t.L(), node, option.TenantName(tenantName), option.SQLInstance(sqlInstance)) require.NoError(t, err) iter := 0 for { @@ -149,7 +125,7 @@ func runMultiTenantDistSQL( t.L().Printf("worker %d done:%v", li, insertCtx.Err()) return nil default: - // procede to report error + // proceed to report error } require.NoError(t, err, "instance idx = %d, iter = %d", li, iter) iter++ @@ -189,7 +165,6 @@ func runMultiTenantDistSQL( } else { t.L().Printf("Only %d nodes present: %v", nodesInPlan.Len(), nodesInPlan) } - } m.Wait() @@ -233,7 +208,8 @@ func runMultiTenantDistSQL( if bundle { // Open bundle and verify its contents sqlConnCtx := clisqlclient.Context{} - conn := sqlConnCtx.MakeSQLConn(io.Discard, io.Discard, instance1.pgURL) + pgURL, err := c.ExternalPGUrl(ctx, t.L(), c.Node(1), tenantName, 0) + conn := sqlConnCtx.MakeSQLConn(io.Discard, io.Discard, pgURL[0]) bundles, err := clisqlclient.StmtDiagListBundles(ctx, conn) require.NoError(t, err) From 037f54a94e7f9db1dad0fc2fd6e05e9cc0e71cfa Mon Sep 17 00:00:00 2001 From: Herko Lategan Date: Wed, 6 Dec 2023 15:38:18 +0000 Subject: [PATCH 4/6] roachtest: remove unused functions from multitenant utils Epic: None Release Note: None --- .../roachtest/tests/multitenant_distsql.go | 1 + pkg/cmd/roachtest/tests/multitenant_utils.go | 48 ------------------- 2 files changed, 1 insertion(+), 48 deletions(-) diff --git a/pkg/cmd/roachtest/tests/multitenant_distsql.go b/pkg/cmd/roachtest/tests/multitenant_distsql.go index 80c8242b3c98..bd0e0320d08d 100644 --- a/pkg/cmd/roachtest/tests/multitenant_distsql.go +++ b/pkg/cmd/roachtest/tests/multitenant_distsql.go @@ -209,6 +209,7 @@ func runMultiTenantDistSQL( // Open bundle and verify its contents sqlConnCtx := clisqlclient.Context{} pgURL, err := c.ExternalPGUrl(ctx, t.L(), c.Node(1), tenantName, 0) + require.NoError(t, err) conn := sqlConnCtx.MakeSQLConn(io.Discard, io.Discard, pgURL[0]) bundles, err := clisqlclient.StmtDiagListBundles(ctx, conn) require.NoError(t, err) diff --git a/pkg/cmd/roachtest/tests/multitenant_utils.go b/pkg/cmd/roachtest/tests/multitenant_utils.go index 3ad051cc6d04..722bc1a1c64f 100644 --- a/pkg/cmd/roachtest/tests/multitenant_utils.go +++ b/pkg/cmd/roachtest/tests/multitenant_utils.go @@ -16,8 +16,6 @@ import ( "fmt" "math/rand" "net/url" - "os" - "path/filepath" "strconv" "strings" "time" @@ -63,14 +61,6 @@ type createTenantOptions struct { } type createTenantOpt func(*createTenantOptions) -func createTenantCertNodes(nodes option.NodeListOption) createTenantOpt { - return func(c *createTenantOptions) { c.certNodes = nodes } -} - -func createTenantEnvVar(envVar string) createTenantOpt { - return func(c *createTenantOptions) { c.envVars = append(c.envVars, envVar) } -} - func createTenantNodeInternal( ctx context.Context, t test.Test, @@ -284,44 +274,6 @@ func startTenantServer( return errCh } -func newTenantInstance( - ctx context.Context, tn *tenantNode, t test.Test, c cluster.Cluster, node, http, sql int, -) (*tenantNode, error) { - instID := tenantIds[tn.tenantID] + 1 - tenantIds[tn.tenantID] = instID - inst := tenantNode{ - tenantID: tn.tenantID, - instanceID: instID, - kvAddrs: tn.kvAddrs, - node: node, - httpPort: http, - sqlPort: sql, - envVars: tn.envVars, - } - tenantCertsDir, err := os.MkdirTemp("", "tenant-certs") - if err != nil { - return nil, err - } - key, crt := fmt.Sprintf("client-tenant.%d.key", tn.tenantID), fmt.Sprintf("client-tenant.%d.crt", tn.tenantID) - err = c.Get(ctx, t.L(), filepath.Join("certs", key), filepath.Join(tenantCertsDir, key), c.Node(tn.node)) - if err != nil { - return nil, err - } - err = c.Get(ctx, t.L(), filepath.Join("certs", crt), filepath.Join(tenantCertsDir, crt), c.Node(tn.node)) - if err != nil { - return nil, err - } - c.Put(ctx, filepath.Join(tenantCertsDir, key), filepath.Join("certs", key), c.Node(node)) - c.Put(ctx, filepath.Join(tenantCertsDir, crt), filepath.Join("certs", crt), c.Node(node)) - // sigh: locally theses are symlinked which breaks our crypto cert checks - if c.IsLocal() { - c.Run(ctx, c.Node(node), "rm", filepath.Join("certs", key)) - c.Run(ctx, c.Node(node), "cp", filepath.Join(tenantCertsDir, key), filepath.Join("certs", key)) - } - c.Run(ctx, c.Node(node), "chmod", "0600", filepath.Join("certs", key)) - return &inst, nil -} - // createTenantAdminRole creates a role that can be used to log into a secure cluster's db console. func createTenantAdminRole(t test.Test, tenantName string, tenantSQL *sqlutils.SQLRunner) { username := "secure" From a1f26ddb8fcc6968c5d4ab70e5c3a749b8c63c74 Mon Sep 17 00:00:00 2001 From: Herko Lategan Date: Tue, 12 Dec 2023 14:22:05 +0000 Subject: [PATCH 5/6] roachtest: document `StartServiceForVirtualClusterE` Epic: None Release Note: None --- pkg/cmd/roachtest/cluster.go | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/pkg/cmd/roachtest/cluster.go b/pkg/cmd/roachtest/cluster.go index e3ebb2a6a7b9..615cba1b6d6c 100644 --- a/pkg/cmd/roachtest/cluster.go +++ b/pkg/cmd/roachtest/cluster.go @@ -1987,6 +1987,17 @@ func (c *clusterImpl) StartE( return nil } +// StartServiceForVirtualClusterE can start either external or shared process +// virtual clusters. This can be specified in startOpts.RoachprodOpts. Set the +// `Target` to the required virtual cluster type. Refer to the virtual cluster +// section in the struct for more information on what fields are available for +// virtual clusters. +// +// With external process virtual clusters an external process will be started on +// each node specified in the externalNodes parameter. +// +// With shared process virtual clusters the required queries will be run on a +// storage node of the cluster specified in the opts parameter. func (c *clusterImpl) StartServiceForVirtualClusterE( ctx context.Context, l *logger.Logger, From 555437ed8a824eff7d0585b9f8821b697236e16c Mon Sep 17 00:00:00 2001 From: Herko Lategan Date: Tue, 12 Dec 2023 14:33:55 +0000 Subject: [PATCH 6/6] roachtest: add `DefaultStartVirtualClusterOpts` convenience function Add a convenience function to return `StartOpts` for starting an external process virtual cluster. Epic: None Release Note: None --- pkg/cmd/roachtest/option/options.go | 10 ++++++++++ pkg/cmd/roachtest/tests/multitenant_distsql.go | 12 +----------- 2 files changed, 11 insertions(+), 11 deletions(-) diff --git a/pkg/cmd/roachtest/option/options.go b/pkg/cmd/roachtest/option/options.go index b0beab4f00d9..3d24df4e9a79 100644 --- a/pkg/cmd/roachtest/option/options.go +++ b/pkg/cmd/roachtest/option/options.go @@ -56,6 +56,16 @@ func DefaultStartSingleNodeOpts() StartOpts { return startOpts } +// DefaultStartVirtualClusterOpts returns StartOpts for starting an external +// process virtual cluster with the given tenant name and SQL instance. +func DefaultStartVirtualClusterOpts(tenantName string, sqlInstance int) StartOpts { + startOpts := StartOpts{RoachprodOpts: roachprod.DefaultStartOpts()} + startOpts.RoachprodOpts.Target = install.StartServiceForVirtualCluster + startOpts.RoachprodOpts.VirtualClusterName = tenantName + startOpts.RoachprodOpts.SQLInstance = sqlInstance + return startOpts +} + // StopOpts is a type that combines the stop options needed by roachprod and roachtest. type StopOpts struct { // TODO(radu): we should use a higher-level abstraction instead of diff --git a/pkg/cmd/roachtest/tests/multitenant_distsql.go b/pkg/cmd/roachtest/tests/multitenant_distsql.go index bd0e0320d08d..0cf83d22cf77 100644 --- a/pkg/cmd/roachtest/tests/multitenant_distsql.go +++ b/pkg/cmd/roachtest/tests/multitenant_distsql.go @@ -73,17 +73,7 @@ func runMultiTenantDistSQL( for i := 0; i < numInstances; i++ { node := (i % c.Spec().NodeCount) + 1 sqlInstance := i / c.Spec().NodeCount - instStartOps := option.DefaultStartOpts() - instStartOps.RoachprodOpts.Target = install.StartServiceForVirtualCluster - instStartOps.RoachprodOpts.VirtualClusterName = tenantName - instStartOps.RoachprodOpts.SQLInstance = sqlInstance - // We set the ports to 0 so that ports are assigned dynamically. This is a - // temporary workaround until we use dynamic port assignment as the default. - // See: https://github.com/cockroachdb/cockroach/issues/111052 - // TODO(herko): remove this once dynamic port assignment is the default. - instStartOps.RoachprodOpts.SQLPort = 0 - instStartOps.RoachprodOpts.AdminUIPort = 0 - + instStartOps := option.DefaultStartVirtualClusterOpts(tenantName, sqlInstance) t.L().Printf("Starting instance %d on node %d", i, node) c.StartServiceForVirtualCluster(ctx, t.L(), c.Node(node), instStartOps, settings, storageNodes) nodes.Add(i + 1)