2012-02-20 12:32:46 +00:00
|
|
|
// mgo - MongoDB driver for Go
|
|
|
|
//
|
|
|
|
// Copyright (c) 2010-2012 - Gustavo Niemeyer <gustavo@niemeyer.net>
|
|
|
|
//
|
|
|
|
// All rights reserved.
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are met:
|
|
|
|
//
|
|
|
|
// 1. Redistributions of source code must retain the above copyright notice, this
|
|
|
|
// list of conditions and the following disclaimer.
|
|
|
|
// 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
// this list of conditions and the following disclaimer in the documentation
|
|
|
|
// and/or other materials provided with the distribution.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
// ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
|
|
// ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
|
// ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
package mgo_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
2012-02-22 09:18:46 +00:00
|
|
|
. "camlistore.org/third_party/launchpad.net/gocheck"
|
2012-07-01 19:06:34 +00:00
|
|
|
"camlistore.org/third_party/labix.org/v2/mgo"
|
|
|
|
"camlistore.org/third_party/labix.org/v2/mgo/bson"
|
2012-02-20 12:32:46 +00:00
|
|
|
"os/exec"
|
|
|
|
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
var fast = flag.Bool("fast", false, "Skip slow tests")
|
|
|
|
|
|
|
|
type M bson.M
|
|
|
|
|
|
|
|
type cLogger C
|
|
|
|
|
|
|
|
func (c *cLogger) Output(calldepth int, s string) error {
|
|
|
|
ns := time.Now().UnixNano()
|
|
|
|
t := float64(ns%100e9) / 1e9
|
|
|
|
((*C)(c)).Logf("[LOG] %.05f %s", t, s)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAll(t *testing.T) {
|
|
|
|
TestingT(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
type S struct {
|
|
|
|
session *mgo.Session
|
|
|
|
stopped bool
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ = Suite(&S{})
|
|
|
|
|
|
|
|
func (s *S) SetUpSuite(c *C) {
|
|
|
|
mgo.SetDebug(true)
|
|
|
|
mgo.SetStats(true)
|
|
|
|
s.StartAll()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *S) SetUpTest(c *C) {
|
|
|
|
err := run("mongo --nodb testdb/dropall.js")
|
|
|
|
if err != nil {
|
|
|
|
panic(err.Error())
|
|
|
|
}
|
|
|
|
mgo.SetLogger((*cLogger)(c))
|
|
|
|
mgo.ResetStats()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *S) TearDownTest(c *C) {
|
|
|
|
if s.stopped {
|
|
|
|
s.StartAll()
|
|
|
|
}
|
|
|
|
for i := 0; ; i++ {
|
|
|
|
stats := mgo.GetStats()
|
|
|
|
if stats.SocketsInUse == 0 && stats.SocketsAlive == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if i == 20 {
|
|
|
|
c.Fatal("Test left sockets in a dirty state")
|
|
|
|
}
|
|
|
|
c.Logf("Waiting for sockets to die: %d in use, %d alive", stats.SocketsInUse, stats.SocketsAlive)
|
|
|
|
time.Sleep(5e8)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *S) Stop(host string) {
|
|
|
|
err := run("cd _testdb && supervisorctl stop " + supvName(host))
|
|
|
|
if err != nil {
|
|
|
|
panic(err.Error())
|
|
|
|
}
|
|
|
|
s.stopped = true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *S) StartAll() {
|
|
|
|
// Restart any stopped nodes.
|
|
|
|
run("cd _testdb && supervisorctl start all")
|
|
|
|
err := run("cd testdb && mongo --nodb wait.js")
|
|
|
|
if err != nil {
|
|
|
|
panic(err.Error())
|
|
|
|
}
|
|
|
|
s.stopped = false
|
|
|
|
}
|
|
|
|
|
|
|
|
func run(command string) error {
|
|
|
|
output, err := exec.Command("/bin/sh", "-c", command).CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("Failed to execute: %s: %s\n%s", command, err.Error(), string(output))
|
|
|
|
return errors.New(msg)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// supvName returns the supervisord name for the given host address.
|
|
|
|
func supvName(host string) string {
|
|
|
|
switch {
|
|
|
|
case strings.HasSuffix(host, ":40001"):
|
|
|
|
return "db1"
|
|
|
|
case strings.HasSuffix(host, ":40011"):
|
|
|
|
return "rs1a"
|
|
|
|
case strings.HasSuffix(host, ":40012"):
|
|
|
|
return "rs1b"
|
|
|
|
case strings.HasSuffix(host, ":40013"):
|
|
|
|
return "rs1c"
|
|
|
|
case strings.HasSuffix(host, ":40021"):
|
|
|
|
return "rs2a"
|
|
|
|
case strings.HasSuffix(host, ":40022"):
|
|
|
|
return "rs2b"
|
|
|
|
case strings.HasSuffix(host, ":40023"):
|
|
|
|
return "rs2c"
|
|
|
|
case strings.HasSuffix(host, ":40101"):
|
|
|
|
return "cfg1"
|
|
|
|
case strings.HasSuffix(host, ":40201"):
|
|
|
|
return "s1"
|
|
|
|
case strings.HasSuffix(host, ":40202"):
|
|
|
|
return "s2"
|
|
|
|
}
|
|
|
|
panic("Unknown host: " + host)
|
|
|
|
}
|