func (s *ELBSuite) TestHandleMessageWithUnits(c *gocheck.C) { id1 := s.server.NewInstance() id2 := s.server.NewInstance() defer s.server.RemoveInstance(id1) defer s.server.RemoveInstance(id2) app := testing.NewFakeApp("symfonia", "python", 1) router, err := Router() c.Assert(err, gocheck.IsNil) router.AddBackend(app.GetName()) c.Assert(err, gocheck.IsNil) defer router.RemoveBackend(app.GetName()) output := strings.Replace(simpleCollectOutput, "i-00004444", id1, -1) output = strings.Replace(output, "i-00004445", id2, -1) tmpdir, err := commandmocker.Add("juju", output) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) msg := queue.Message{ Action: addUnitToLoadBalancer, Args: []string{"symfonia", "symfonia/0", "symfonia/1"}, } handle(&msg) resp, err := s.client.DescribeLoadBalancers(app.GetName()) c.Assert(err, gocheck.IsNil) c.Assert(resp.LoadBalancerDescriptions, gocheck.HasLen, 1) instances := resp.LoadBalancerDescriptions[0].Instances c.Assert(instances, gocheck.HasLen, 2) ids := []string{instances[0].InstanceId, instances[1].InstanceId} sort.Strings(ids) want := []string{id1, id2} sort.Strings(want) c.Assert(ids, gocheck.DeepEquals, want) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) }
func (s *S) TestReadMessageWithLaunchJSON(c *check.C) { cmd, err := commandmocker.Add("sudo", "service nginx reload") c.Assert(err, check.IsNil) defer commandmocker.Remove(cmd) payload := `{"TopicArn":"arn:test","Message":` + `"{\"AutoScalingGroupARN\":\"arn:asg-test\",\"Event\":\"autoscaling:EC2_INSTANCE_LAUNCH\",` + `\"EC2InstanceId\":\"%s\"}"}` b := strings.NewReader(fmt.Sprintf(payload, s.instance_ids[0])) recorder, request := newRequest("POST", "/", b, c) readMessage(recorder, request) body := readBody(recorder.Body, c) c.Assert(body, check.Equals, fmt.Sprintf(`Added instance "%s".`, s.instance_ids[0])) c.Assert(recorder.Code, check.Equals, 200) // Check upstreams file upstream := Config.Upstreams[0] content, err := ioutil.ReadFile(upstream.File) c.Assert(err, check.IsNil) serverLine := fmt.Sprintf("server %s.internal.invalid:80 max_fails=3 fail_timeout=60s; # %s\n", s.instance_ids[0], s.instance_ids[0]) c.Assert(string(content), check.Equals, fmt.Sprintf("upstream %s {\n %s}\n", upstream.Name, serverLine)) // Check run NGINX reload c.Assert(commandmocker.Ran(cmd), check.Equals, true) }
func (s *S) TestReadMessageWithTerminateJSON(c *check.C) { cmd, err := commandmocker.Add("sudo", "service nginx reload") c.Assert(err, check.IsNil) defer commandmocker.Remove(cmd) // Setup instance file u := Config.Upstreams[0] instance := &ec2.Instance{InstanceId: s.instance_ids[0], PrivateDNSName: "test.internal"} if err := addInstance(u, instance); err != nil { c.Error(err) } payload := `{"TopicArn":"arn:test","Message": "{\"AutoScalingGroupARN\":\"arn:asg-test\",\"Event\":\"autoscaling:EC2_INSTANCE_TERMINATE\",\"EC2InstanceId\":\"%s\"}"}` b := strings.NewReader(fmt.Sprintf(payload, s.instance_ids[0])) recorder, request := newRequest("POST", "/", b, c) readMessage(recorder, request) body := readBody(recorder.Body, c) c.Assert(body, check.Equals, fmt.Sprintf(`Removed instance "%s".`, s.instance_ids[0])) c.Assert(recorder.Code, check.Equals, 200) // Check upstreams file content, err := ioutil.ReadFile(u.File) c.Assert(err, check.IsNil) c.Assert(string(content), check.Equals, fmt.Sprintf("upstream %s {\n}\n", u.Name)) // Check run NGINX reload c.Assert(commandmocker.Ran(cmd), check.Equals, true) }
func (s *S) TestNewBareShouldCreateADir(c *gocheck.C) { dir, err := commandmocker.Add("git", "$*") c.Check(err, gocheck.IsNil) defer commandmocker.Remove(dir) err = newBare("myBare") c.Assert(err, gocheck.IsNil) c.Assert(commandmocker.Ran(dir), gocheck.Equals, true) }
func (s *S) TestCollectStatusDirtyOutput(c *gocheck.C) { server, url := badGatewayServer() defer server.Close() tmpdir, err := commandmocker.Add("juju", generateDirtyOutput(url)) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) expected := []provision.Unit{ { Name: "as_i_rise/0", AppName: "as_i_rise", Type: "django", Machine: 105, InstanceId: "i-00000439", Ip: url, Status: provision.StatusUnreachable, }, { Name: "the_infanta/1", AppName: "the_infanta", Type: "gunicorn", Machine: 107, InstanceId: "i-0000043e", Ip: url, Status: provision.StatusBuilding, }, } p := JujuProvisioner{} units, err := p.CollectStatus() c.Assert(err, gocheck.IsNil) cp := make([]provision.Unit, len(units)) copy(cp, units) if cp[0].Type == "gunicorn" { cp[0], cp[1] = cp[1], cp[0] } c.Assert(cp, gocheck.DeepEquals, expected) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) var wg sync.WaitGroup wg.Add(1) collection := p.unitsCollection() defer collection.Close() go func() { q := bson.M{"_id": bson.M{"$in": []string{"as_i_rise/0", "the_infanta/1"}}} for { if n, _ := collection.Find(q).Count(); n == 2 { break } runtime.Gosched() } collection.Remove(q) wg.Done() }() wg.Wait() }
func (s *S) TestNewShouldCreateNewGitBareRepository(c *gocheck.C) { tmpdir, err := commandmocker.Add("git", "$*") c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) _, err = New("myRepo", []string{"pumpkin"}, true) c.Assert(err, gocheck.IsNil) conn, err := db.Conn() c.Assert(err, gocheck.IsNil) defer conn.Close() defer conn.Repository().Remove(bson.M{"_id": "myRepo"}) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) }
func (s *S) TestExecuteWithoutArgs(c *gocheck.C) { tmpdir, err := commandmocker.Add("ls", "ok") c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) var e OsExecutor var b bytes.Buffer err = e.Execute("ls", nil, nil, &b, &b) c.Assert(err, gocheck.IsNil) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) c.Assert(commandmocker.Parameters(tmpdir), gocheck.IsNil) c.Assert(b.String(), gocheck.Equals, "ok") }
func (s *S) TestRemoveUnitUnknownByJuju(c *gocheck.C) { output := `013-01-11 20:02:07,883 INFO Connecting to environment... 2013-01-11 20:02:10,147 INFO Connected to environment. 2013-01-11 20:02:10,160 ERROR Service unit 'two/2' was not found` tmpdir, err := commandmocker.Error("juju", output, 1) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) app := testing.NewFakeApp("two", "rush", 3) p := JujuProvisioner{} err = p.RemoveUnit(app, "two/2") c.Assert(err, gocheck.IsNil) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) }
func (s *S) TestCollectStatusFailure(c *gocheck.C) { tmpdir, err := commandmocker.Error("juju", "juju failed", 1) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) p := JujuProvisioner{} _, err = p.CollectStatus() c.Assert(err, gocheck.NotNil) pErr, ok := err.(*provision.Error) c.Assert(ok, gocheck.Equals, true) c.Assert(pErr.Reason, gocheck.Equals, "juju failed") c.Assert(pErr.Err.Error(), gocheck.Equals, "exit status 1") c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) }
func (s *S) TestExecute(c *gocheck.C) { tmpdir, err := commandmocker.Add("ls", "ok") c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) var e OsExecutor var b bytes.Buffer err = e.Execute("ls", []string{"-lsa"}, nil, &b, &b) c.Assert(err, gocheck.IsNil) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) expected := []string{"-lsa"} c.Assert(commandmocker.Parameters(tmpdir), gocheck.DeepEquals, expected) c.Assert(b.String(), gocheck.Equals, "ok") }
func (s *S) TestNewBareShouldNotPassTemplateOptionWhenItsNotSetInConfig(c *gocheck.C) { config.Unset("git:bare:template") bareLocation, err := config.GetString("git:bare:location") c.Assert(err, gocheck.IsNil) barePath := path.Join(bareLocation, "foo.git") dir, err := commandmocker.Add("git", "$*") c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(dir) err = newBare("foo") c.Assert(err, gocheck.IsNil) c.Assert(commandmocker.Ran(dir), gocheck.Equals, true) expected := fmt.Sprintf("init %s --bare", barePath) c.Assert(commandmocker.Output(dir), gocheck.Equals, expected) }
func (s *S) TestNewBareShouldPassTemplateOptionWhenItExistsOnConfig(c *check.C) { bareTemplate := "/var/templates" bareLocation, err := config.GetString("git:bare:location") config.Set("git:bare:template", bareTemplate) defer config.Unset("git:bare:template") barePath := path.Join(bareLocation, "foo.git") dir, err := commandmocker.Add("git", "$*") c.Assert(err, check.IsNil) defer commandmocker.Remove(dir) err = newBare("foo") c.Assert(err, check.IsNil) c.Assert(commandmocker.Ran(dir), check.Equals, true) expected := fmt.Sprintf("init %s --bare --template=%s", barePath, bareTemplate) c.Assert(commandmocker.Output(dir), check.Equals, expected) }
func (s *S) TestExecuteActionShouldNotCallSSH_ORIGINAL_COMMANDWhenRepositoryDoesNotExists(c *gocheck.C) { dir, err := commandmocker.Add("git-receive-pack", "$*") c.Check(err, gocheck.IsNil) defer commandmocker.Remove(dir) os.Args = []string{"gandalf", s.user.Name} os.Setenv("SSH_ORIGINAL_COMMAND", "git-receive-pack 'ghostapp.git'") defer func() { os.Args = []string{} os.Setenv("SSH_ORIGINAL_COMMAND", "") }() stdout := &bytes.Buffer{} errorMsg := "You don't have access to write in this repository." executeAction(hasWritePermission, errorMsg, stdout) c.Assert(commandmocker.Ran(dir), gocheck.Equals, false) }
func (s *S) TestInstanceMachineHealWhenEverythingIsOk(c *gocheck.C) { fexec := &etesting.FakeExecutor{} setExecut(fexec) defer setExecut(nil) jujuTmpdir, err := commandmocker.Add("juju", collectOutput) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(jujuTmpdir) h := instanceMachineHealer{} err = h.Heal() c.Assert(err, gocheck.IsNil) args := []string{ "status", // for juju status that gets the output } c.Assert(commandmocker.Ran(jujuTmpdir), gocheck.Equals, true) c.Assert(commandmocker.Parameters(jujuTmpdir), gocheck.DeepEquals, args) }
func (s *S) TestExecuteWithoutArgs(c *gocheck.C) { tmpdir, err := commandmocker.Add("ls", "ok") c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) var e OsExecutor var b bytes.Buffer opts := ExecuteOptions{ Cmd: "ls", Stdout: &b, Stderr: &b, } err = e.Execute(opts) c.Assert(err, gocheck.IsNil) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) c.Assert(commandmocker.Parameters(tmpdir), gocheck.IsNil) c.Assert(b.String(), gocheck.Equals, "ok") }
func (s *S) TestExecuteActionShouldExecuteGitReceivePackWhenUserHasWritePermission(c *gocheck.C) { dir, err := commandmocker.Add("git-receive-pack", "$*") c.Check(err, gocheck.IsNil) defer commandmocker.Remove(dir) os.Args = []string{"gandalf", s.user.Name} os.Setenv("SSH_ORIGINAL_COMMAND", "git-receive-pack 'myapp.git'") defer func() { os.Args = []string{} os.Setenv("SSH_ORIGINAL_COMMAND", "") }() stdout := &bytes.Buffer{} executeAction(hasWritePermission, "You don't have access to write in this repository.", stdout) c.Assert(commandmocker.Ran(dir), gocheck.Equals, true) p, err := config.GetString("git:bare:location") c.Assert(err, gocheck.IsNil) expected := path.Join(p, "myapp.git") c.Assert(stdout.String(), gocheck.Equals, expected) }
func (s *S) TestExecute(c *check.C) { tmpdir, err := commandmocker.Add("ls", "ok") c.Assert(err, check.IsNil) defer commandmocker.Remove(tmpdir) var e OsExecutor var b bytes.Buffer opts := ExecuteOptions{ Cmd: "ls", Args: []string{"-lsa"}, Stdout: &b, Stderr: &b, } err = e.Execute(opts) c.Assert(err, check.IsNil) c.Assert(commandmocker.Ran(tmpdir), check.Equals, true) expected := []string{"-lsa"} c.Assert(commandmocker.Parameters(tmpdir), check.DeepEquals, expected) c.Assert(b.String(), check.Equals, "ok") }
// This example demonstrates the mocking of the SSH command. func ExampleAdd() { msg := "HELP!" path, err := commandmocker.Error("ssh", msg, 1) if err != nil { panic(err) } defer commandmocker.Remove(path) cmd := exec.Command("ssh", "-l", "root", "127.0.0.1") out, err := cmd.CombinedOutput() fmt.Println(err) fmt.Printf("%s\n", out) fmt.Println(commandmocker.Parameters(path)) fmt.Println(commandmocker.Ran(path)) // Output: // exit status 1 // HELP! // [-l root 127.0.0.1] // true }
func (s *S) TestInstanceMachineHeal(c *gocheck.C) { fexec := &etesting.FakeExecutor{} setExecut(fexec) defer setExecut(nil) jujuTmpdir, err := commandmocker.Add("juju", collectOutputInstanceDown) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(jujuTmpdir) h := instanceMachineHealer{} err = h.Heal() c.Assert(err, gocheck.IsNil) args := []string{ "-o", "StrictHostKeyChecking no", "-q", "-l", "ubuntu", "10.10.10.163", "sudo", "stop", "juju-machine-agent", } c.Assert(fexec.ExecutedCmd("ssh", args), gocheck.Equals, true) args = []string{ "-o", "StrictHostKeyChecking no", "-q", "-l", "ubuntu", "10.10.10.163", "sudo", "start", "juju-machine-agent", } c.Assert(fexec.ExecutedCmd("ssh", args), gocheck.Equals, true) args = []string{ "status", // for juju status that gets the output } c.Assert(commandmocker.Ran(jujuTmpdir), gocheck.Equals, true) c.Assert(commandmocker.Parameters(jujuTmpdir), gocheck.DeepEquals, args) }
func (s *S) TestCollectStatus(c *gocheck.C) { server, url := badGatewayServer() defer server.Close() tmpdir, err := commandmocker.Add("juju", generateOutput(url)) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(tmpdir) p := JujuProvisioner{} collection := p.unitsCollection() defer collection.Close() err = collection.Insert(instance{UnitName: "as_i_rise/0", InstanceID: "i-00000439"}) c.Assert(err, gocheck.IsNil) defer collection.Remove(bson.M{"_id": bson.M{"$in": []string{"as_i_rise/0", "the_infanta/0"}}}) expected := []provision.Unit{ { Name: "as_i_rise/0", AppName: "as_i_rise", Type: "django", Machine: 105, InstanceId: "i-00000439", Ip: url, Status: provision.StatusUnreachable, }, { Name: "the_infanta/0", AppName: "the_infanta", Type: "gunicorn", Machine: 107, InstanceId: "i-0000043e", Ip: url, Status: provision.StatusBuilding, }, } units, err := p.CollectStatus() c.Assert(err, gocheck.IsNil) cp := make([]provision.Unit, len(units)) copy(cp, units) if cp[0].Type == "gunicorn" { cp[0], cp[1] = cp[1], cp[0] } c.Assert(cp, gocheck.DeepEquals, expected) c.Assert(commandmocker.Ran(tmpdir), gocheck.Equals, true) done := make(chan int8) go func() { for { ct, err := collection.Find(nil).Count() c.Assert(err, gocheck.IsNil) if ct == 2 { done <- 1 return } } }() select { case <-done: case <-time.After(5e9): c.Fatal("Did not save the unit after 5 seconds.") } var instances []instance err = collection.Find(nil).Sort("_id").All(&instances) c.Assert(err, gocheck.IsNil) c.Assert(instances, gocheck.HasLen, 2) c.Assert(instances[0].UnitName, gocheck.Equals, "as_i_rise/0") c.Assert(instances[0].InstanceID, gocheck.Equals, "i-00000439") c.Assert(instances[1].UnitName, gocheck.Equals, "the_infanta/0") c.Assert(instances[1].InstanceID, gocheck.Equals, "i-0000043e") var b machine err = collection.Find(nil).One(&b) c.Assert(err, gocheck.IsNil) }
func (s *S) TestInstanceAgenstConfigHealerHeal(c *gocheck.C) { server, err := ec2test.NewServer() c.Assert(err, gocheck.IsNil) defer server.Quit() id := server.NewInstances(1, "small", "ami-123", ec2.InstanceState{Code: 16, Name: "running"}, nil)[0] p := JujuProvisioner{} m := machine{ AgentState: "not-started", IPAddress: "localhost", InstanceID: id, InstanceState: "running", } p.saveBootstrapMachine(m) collection := p.bootstrapCollection() defer collection.Close() defer collection.Remove(m) a := app.App{ Name: "as_i_rise", Units: []app.Unit{{Name: "as_i_rise/0", State: "down", Ip: "server-1081.novalocal"}}, } err = s.conn.Apps().Insert(&a) c.Assert(err, gocheck.IsNil) defer s.conn.Apps().Remove(bson.M{"name": "as_i_rise"}) sshTmpdir, err := commandmocker.Error("ssh", "", 1) c.Assert(err, gocheck.IsNil) defer commandmocker.Remove(sshTmpdir) h := instanceAgentsConfigHealer{} auth := aws.Auth{AccessKey: "access", SecretKey: "secret"} region := aws.SAEast region.EC2Endpoint = server.URL() h.e = ec2.New(auth, region) err = h.Heal() c.Assert(err, gocheck.IsNil) sshOutput := []string{ "-o", "StrictHostKeyChecking no", "-q", "-l", "ubuntu", "server-1081.novalocal", "grep", "i-0.internal.invalid", "/etc/init/juju-machine-agent.conf", "-o", "StrictHostKeyChecking no", "-q", "-l", "ubuntu", "server-1081.novalocal", "sudo", "sed", "-i", "'s/env JUJU_ZOOKEEPER=.*/env JUJU_ZOOKEEPER=\"i-0.internal.invalid:2181\"/g'", "/etc/init/juju-machine-agent.conf", "-o", "StrictHostKeyChecking no", "-q", "-l", "ubuntu", "server-1081.novalocal", "grep", "i-0.internal.invalid", "/etc/init/juju-as_i_rise-0.conf", "-o", "StrictHostKeyChecking no", "-q", "-l", "ubuntu", "server-1081.novalocal", "sudo", "sed", "-i", "'s/env JUJU_ZOOKEEPER=.*/env JUJU_ZOOKEEPER=\"i-0.internal.invalid:2181\"/g'", "/etc/init/juju-as_i_rise-0.conf", } c.Assert(commandmocker.Ran(sshTmpdir), gocheck.Equals, true) c.Assert(commandmocker.Parameters(sshTmpdir), gocheck.DeepEquals, sshOutput) }