Skip to content

Commit

Permalink
server: deflake TestRapidRestarts
Browse files Browse the repository at this point in the history
In go1.10 and earlier, it was not safe to call `http.ServeMux.ServeHTTP`
concurrently with `http.ServeMux.Handle`. (This is fixed in go1.11). In
the interim, provide our own safeServeMux wrapper that provides proper
locking.

Fixes cockroachdb#29227

Release note: None
  • Loading branch information
petermattis committed Aug 29, 2018
1 parent 50ff8dc commit f932530
Show file tree
Hide file tree
Showing 2 changed files with 90 additions and 2 deletions.
66 changes: 66 additions & 0 deletions pkg/server/servemux_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
// Copyright 2018 The Cockroach Authors.
//
// 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
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
// implied. See the License for the specific language governing
// permissions and limitations under the License.

package server

import (
"fmt"
"net/http"
"net/http/httptest"
"net/url"
"sync"
"testing"
"time"

"github.com/cockroachdb/cockroach/pkg/util/leaktest"
"github.com/cockroachdb/cockroach/pkg/util/timeutil"
)

func TestServeMuxConcurrency(t *testing.T) {
defer leaktest.AfterTest(t)()

const duration = 20 * time.Millisecond
start := timeutil.Now()

// TODO(peter): This test reliably fails using http.ServeMux with a
// "concurrent map read and write error" on go1.10. The bug in http.ServeMux
// is fixed in go1.11.
var mux safeServeMux
var wg sync.WaitGroup
wg.Add(2)

go func() {
defer wg.Done()
f := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {})
for i := 1; timeutil.Since(start) < duration; i++ {
mux.Handle(fmt.Sprintf("/%d", i), f)
}
}()

go func() {
defer wg.Done()
for i := 1; timeutil.Since(start) < duration; i++ {
r := &http.Request{
Method: "GET",
URL: &url.URL{
Path: "/",
},
}
w := httptest.NewRecorder()
mux.ServeHTTP(w, r)
}
}()

wg.Wait()
}
26 changes: 24 additions & 2 deletions pkg/server/server.go
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ import (
"github.com/cockroachdb/cockroach/pkg/util/protoutil"
"github.com/cockroachdb/cockroach/pkg/util/retry"
"github.com/cockroachdb/cockroach/pkg/util/stop"
"github.com/cockroachdb/cockroach/pkg/util/syncutil"
"github.com/cockroachdb/cockroach/pkg/util/timeutil"
"github.com/cockroachdb/cockroach/pkg/util/uuid"
)
Expand Down Expand Up @@ -123,13 +124,35 @@ var (
)
)

// TODO(peter): Until go1.11, ServeMux.ServeHTTP was not safe to call
// concurrently with ServeMux.Handle. So we provide our own wrapper with proper
// locking. Slightly less efficient because it locks unnecessarily, but
// safe. See TestServeMuxConcurrency. Should remove once we've upgraded to
// go1.11.
type safeServeMux struct {
mu syncutil.RWMutex
mux http.ServeMux
}

func (mux *safeServeMux) Handle(pattern string, handler http.Handler) {
mux.mu.Lock()
mux.mux.Handle(pattern, handler)
mux.mu.Unlock()
}

func (mux *safeServeMux) ServeHTTP(w http.ResponseWriter, r *http.Request) {
mux.mu.RLock()
mux.mux.ServeHTTP(w, r)
mux.mu.RUnlock()
}

// Server is the cockroach server node.
type Server struct {
nodeIDContainer base.NodeIDContainer

cfg Config
st *cluster.Settings
mux *http.ServeMux
mux safeServeMux
clock *hlc.Clock
rpcContext *rpc.Context
grpc *grpc.Server
Expand Down Expand Up @@ -182,7 +205,6 @@ func NewServer(cfg Config, stopper *stop.Stopper) (*Server, error) {
clock := hlc.NewClock(hlc.UnixNano, time.Duration(cfg.MaxOffset))
s := &Server{
st: st,
mux: http.NewServeMux(),
clock: clock,
stopper: stopper,
cfg: cfg,
Expand Down

0 comments on commit f932530

Please sign in to comment.