// Copyright (C) MongoDB, Inc. 2017-present. // // Licensed under the Apache License, Version 2.0 (the "License"); you may // not use this file except in compliance with the License. You may obtain // a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 package integration import ( "context" "testing" "time" "go.mongodb.org/mongo-driver/bson" "go.mongodb.org/mongo-driver/internal/testutil/assert" "go.mongodb.org/mongo-driver/mongo/integration/mtest" "go.mongodb.org/mongo-driver/mongo/options" ) func TestSDAMErrorHandling(t *testing.T) { mt := mtest.New(t, noClientOpts) clientOpts := options.Client(). ApplyURI(mt.ConnString()). SetRetryWrites(false). SetPoolMonitor(poolMonitor). SetWriteConcern(mtest.MajorityWc) baseMtOpts := func() *mtest.Options { mtOpts := mtest.NewOptions(). Topologies(mtest.ReplicaSet). // Don't run on sharded clusters to avoid complexity of sharded failpoints. MinServerVersion("4.0"). // 4.0+ is required to use failpoints on replica sets. ClientOptions(clientOpts) if mt.TopologyKind() == mtest.Sharded { // Pin to a single mongos because the tests use failpoints. mtOpts.ClientType(mtest.Pinned) } return mtOpts } // Set min server version of 4.4 because the during-handshake tests use failpoint features introduced in 4.4 like // blockConnection and appName. mt.RunOpts("before handshake completes", baseMtOpts().Auth(true).MinServerVersion("4.4"), func(mt *mtest.T) { mt.RunOpts("network errors", noClientOpts, func(mt *mtest.T) { mt.Run("pool cleared on network timeout", func(mt *mtest.T) { // Assert that the pool is cleared when a connection created by an application operation thread // encounters a network timeout during handshaking. Unlike the non-timeout test below, we only test // connections created in the foreground for timeouts because connections created by the pool // maintenance routine can't be timed out using a context. appName := "authNetworkTimeoutTest" // Set failpoint on saslContinue instead of saslStart because saslStart isn't done when using // speculative auth. mt.SetFailPoint(mtest.FailPoint{ ConfigureFailPoint: "failCommand", Mode: mtest.FailPointMode{ Times: 1, }, Data: mtest.FailPointData{ FailCommands: []string{"saslContinue"}, BlockConnection: true, BlockTimeMS: 150, AppName: appName, }, }) // Reset the client with the appName specified in the failpoint. clientOpts := options.Client(). SetAppName(appName). SetRetryWrites(false). SetPoolMonitor(poolMonitor) mt.ResetClient(clientOpts) clearPoolChan() // The saslContinue blocks for 150ms so run the InsertOne with a 100ms context to cause a network // timeout during auth and assert that the pool was cleared. timeoutCtx, cancel := context.WithTimeout(mtest.Background, 100*time.Millisecond) defer cancel() _, err := mt.Coll.InsertOne(timeoutCtx, bson.D{{"test", 1}}) assert.NotNil(mt, err, "expected InsertOne error, got nil") assert.True(mt, isPoolCleared(), "expected pool to be cleared but was not") }) mt.RunOpts("pool cleared on non-timeout network error", noClientOpts, func(mt *mtest.T) { mt.Run("background", func(mt *mtest.T) { // Assert that the pool is cleared when a connection created by the background pool maintenance // routine encounters a non-timeout network error during handshaking. appName := "authNetworkErrorTestBackground" mt.SetFailPoint(mtest.FailPoint{ ConfigureFailPoint: "failCommand", Mode: mtest.FailPointMode{ Times: 1, }, Data: mtest.FailPointData{ FailCommands: []string{"saslContinue"}, CloseConnection: true, AppName: appName, }, }) clientOpts := options.Client(). SetAppName(appName). SetMinPoolSize(5). SetPoolMonitor(poolMonitor) mt.ResetClient(clientOpts) clearPoolChan() time.Sleep(200 * time.Millisecond) assert.True(mt, isPoolCleared(), "expected pool to be cleared but was not") }) mt.Run("foreground", func(mt *mtest.T) { // Assert that the pool is cleared when a connection created by an application thread connection // checkout encounters a non-timeout network error during handshaking. appName := "authNetworkErrorTestForeground" mt.SetFailPoint(mtest.FailPoint{ ConfigureFailPoint: "failCommand", Mode: mtest.FailPointMode{ Times: 1, }, Data: mtest.FailPointData{ FailCommands: []string{"saslContinue"}, CloseConnection: true, AppName: appName, }, }) clientOpts := options.Client(). SetAppName(appName). SetPoolMonitor(poolMonitor) mt.ResetClient(clientOpts) clearPoolChan() _, err := mt.Coll.InsertOne(mtest.Background, bson.D{{"x", 1}}) assert.NotNil(mt, err, "expected InsertOne error, got nil") assert.True(mt, isPoolCleared(), "expected pool to be cleared but was not") }) }) }) }) mt.RunOpts("after handshake completes", baseMtOpts(), func(mt *mtest.T) { mt.RunOpts("network errors", noClientOpts, func(mt *mtest.T) { mt.Run("pool cleared on non-timeout network error", func(mt *mtest.T) { clearPoolChan() mt.SetFailPoint(mtest.FailPoint{ ConfigureFailPoint: "failCommand", Mode: mtest.FailPointMode{ Times: 1, }, Data: mtest.FailPointData{ FailCommands: []string{"insert"}, CloseConnection: true, }, }) _, err := mt.Coll.InsertOne(mtest.Background, bson.D{{"test", 1}}) assert.NotNil(mt, err, "expected InsertOne error, got nil") assert.True(mt, isPoolCleared(), "expected pool to be cleared but was not") }) mt.Run("pool not cleared on timeout network error", func(mt *mtest.T) { clearPoolChan() _, err := mt.Coll.InsertOne(mtest.Background, bson.D{{"x", 1}}) assert.Nil(mt, err, "InsertOne error: %v", err) filter := bson.M{ "$where": "function() { sleep(1000); return false; }", } timeoutCtx, cancel := context.WithTimeout(mtest.Background, 100*time.Millisecond) defer cancel() _, err = mt.Coll.Find(timeoutCtx, filter) assert.NotNil(mt, err, "expected Find error, got %v", err) assert.False(mt, isPoolCleared(), "expected pool to not be cleared but was") }) }) }) }