diff --git a/x/mongo/driver/topology/polling_srv_records_test.go b/x/mongo/driver/topology/polling_srv_records_test.go index e0e00f2c4e..c1811aaeae 100644 --- a/x/mongo/driver/topology/polling_srv_records_test.go +++ b/x/mongo/driver/topology/polling_srv_records_test.go @@ -125,9 +125,21 @@ func compareHosts(t *testing.T, received []description.Server, expected []string } func TestPollingSRVRecordsSpec(t *testing.T) { + for _, uri := range []string{ + "mongodb+srv://test1.test.build.10gen.cc/?heartbeatFrequencyMS=100", + // Test with user:pass as a regression test for GODRIVER-2620 + "mongodb+srv://user:pass@test1.test.build.10gen.cc/?heartbeatFrequencyMS=100", + } { + t.Run(uri, func(t *testing.T) { + testPollingSRVRecordsSpec(t, uri) + }) + } +} + +func testPollingSRVRecordsSpec(t *testing.T, uri string) { + t.Helper() for _, tt := range srvPollingTests { t.Run(tt.name, func(t *testing.T) { - uri := "mongodb+srv://test1.test.build.10gen.cc/?heartbeatFrequencyMS=100" cfg, err := NewConfig(options.Client().ApplyURI(uri), nil) require.NoError(t, err, "error constructing topology configs: %v", err) diff --git a/x/mongo/driver/topology/topology.go b/x/mongo/driver/topology/topology.go index e63e6dd1b9..eb0ac425a1 100644 --- a/x/mongo/driver/topology/topology.go +++ b/x/mongo/driver/topology/topology.go @@ -14,6 +14,8 @@ import ( "context" "errors" "fmt" + "net" + "net/url" "strings" "sync" "sync/atomic" @@ -234,7 +236,21 @@ func (t *Topology) Connect() error { t.serversLock.Unlock() if t.pollingRequired { - go t.pollSRVRecords() + uri, err := url.Parse(t.cfg.URI) + if err != nil { + return err + } + // sanity check before passing the hostname to resolver + if parsedHosts := strings.Split(uri.Host, ","); len(parsedHosts) != 1 { + return fmt.Errorf("URI with SRV must include one and only one hostname") + } + _, _, err = net.SplitHostPort(uri.Host) + if err == nil { + // we were able to successfully extract a port from the host, + // but should not be able to when using SRV + return fmt.Errorf("URI with srv must not include a port number") + } + go t.pollSRVRecords(uri.Host) t.pollingwg.Add(1) } @@ -556,7 +572,7 @@ func (t *Topology) selectServerFromDescription(desc description.Topology, return suitable, nil } -func (t *Topology) pollSRVRecords() { +func (t *Topology) pollSRVRecords(hosts string) { defer t.pollingwg.Done() serverConfig := newServerConfig(t.cfg.ServerOpts...) @@ -573,13 +589,6 @@ func (t *Topology) pollSRVRecords() { } }() - // remove the scheme - uri := t.cfg.URI[14:] - hosts := uri - if idx := strings.IndexAny(uri, "/?@"); idx != -1 { - hosts = uri[:idx] - } - for { select { case <-pollTicker.C: