blob: e2309f6d86dc6cc16771c15991d3c47354d52102 [file] [log] [blame]
// Copyright 2015 The Vanadium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package main
import (
"bytes"
"fmt"
"io/ioutil"
"os"
"strings"
"testing"
"v.io/v23"
"v.io/v23/context"
"v.io/v23/naming"
"v.io/v23/rpc"
"v.io/v23/security"
"v.io/v23/security/access"
"v.io/v23/services/application"
"v.io/x/lib/cmdline"
"v.io/x/ref/lib/v23cmd"
_ "v.io/x/ref/runtime/factories/generic"
"v.io/x/ref/services/repository"
"v.io/x/ref/test"
)
var (
envelope = application.Envelope{
Title: "fifa world cup",
Args: []string{"arg1", "arg2", "arg3"},
Binary: application.SignedFile{File: "/path/to/binary"},
Env: []string{"env1", "env2", "env3"},
Packages: map[string]application.SignedFile{
"pkg1": application.SignedFile{
File: "/path/to/package1",
},
},
Restarts: 0,
RestartTimeWindow: 0,
}
jsonEnv = `{
"Title": "fifa world cup",
"Args": [
"arg1",
"arg2",
"arg3"
],
"Binary": {
"File": "/path/to/binary",
"Signature": {
"Purpose": null,
"Hash": "",
"R": null,
"S": null
}
},
"Publisher": "",
"Env": [
"env1",
"env2",
"env3"
],
"Packages": {
"pkg1": {
"File": "/path/to/package1",
"Signature": {
"Purpose": null,
"Hash": "",
"R": null,
"S": null
}
}
},
"Restarts": 0,
"RestartTimeWindow": 0
}`
profiles = "a,b,c,d"
serverOut bytes.Buffer
)
//go:generate jiri test generate
type server struct {
suffix string
}
func (s *server) Match(ctx *context.T, _ rpc.ServerCall, profiles []string) (application.Envelope, error) {
ctx.VI(2).Infof("%v.Match(%v) was called", s.suffix, profiles)
return envelope, nil
}
func (s *server) Put(ctx *context.T, _ rpc.ServerCall, profile string, env application.Envelope, overwrite bool) error {
ctx.VI(2).Infof("%v.Put(%v, %v, %t) was called", s.suffix, profile, env, overwrite)
fmt.Fprintf(&serverOut, "Put(%s, ..., %t)\n", profile, overwrite)
return nil
}
func (s *server) Profiles(ctx *context.T, _ rpc.ServerCall) ([]string, error) {
ctx.VI(2).Infof("%v.Profiles() was called", s.suffix)
return strings.Split(profiles, ","), nil
}
func (s *server) Remove(ctx *context.T, _ rpc.ServerCall, profile string) error {
ctx.VI(2).Infof("%v.Remove(%v) was called", s.suffix, profile)
return nil
}
func (s *server) SetPermissions(ctx *context.T, _ rpc.ServerCall, perms access.Permissions, version string) error {
ctx.VI(2).Infof("%v.SetPermissions(%v, %v) was called", perms, version)
return nil
}
func (s *server) GetPermissions(ctx *context.T, _ rpc.ServerCall) (access.Permissions, string, error) {
ctx.VI(2).Infof("%v.GetPermissions() was called")
return nil, "", nil
}
func (s *server) TidyNow(ctx *context.T, _ rpc.ServerCall) error {
ctx.VI(2).Infof("%v.TidyNow() was called", s)
return nil
}
type dispatcher struct{}
func (d *dispatcher) Lookup(_ *context.T, suffix string) (interface{}, security.Authorizer, error) {
return repository.ApplicationServer(&server{suffix: suffix}), nil, nil
}
func TestApplicationClient(t *testing.T) {
ctx, shutdown := test.V23Init()
defer shutdown()
ctx, server, err := v23.WithNewDispatchingServer(ctx, "", &dispatcher{})
if err != nil {
t.Errorf("NewServer failed: %v", err)
return
}
endpoint := server.Status().Endpoints[0]
// Setup the command-line.
var stdout, stderr bytes.Buffer
resetOut := func() {
stdout.Reset()
stderr.Reset()
serverOut.Reset()
}
env := &cmdline.Env{Stdout: &stdout, Stderr: &stderr}
appName := naming.JoinAddressName(endpoint.String(), "myapp/1")
oneProfile := "myprofile"
severalProfiles := "myprofile1,myprofile2,myprofile1"
// Test the 'Match' command.
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"match", appName, oneProfile}); err != nil {
t.Fatalf("%v", err)
}
if expected, got := jsonEnv, strings.TrimSpace(stdout.String()); got != expected {
t.Errorf("Unexpected output from match. Got %q, expected %q", got, expected)
}
resetOut()
// Test the 'put' command.
f, err := ioutil.TempFile("", "test")
if err != nil {
t.Fatalf("%v", err)
}
fileName := f.Name()
defer os.Remove(fileName)
if _, err = f.Write([]byte(jsonEnv)); err != nil {
t.Fatalf("%v", err)
}
if err = f.Close(); err != nil {
t.Fatalf("%v", err)
}
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"put", appName, severalProfiles, fileName}); err != nil {
t.Fatalf("%v", err)
}
if expected, got := "Application envelope added for profile myprofile1.\nApplication envelope added for profile myprofile2.", strings.TrimSpace(stdout.String()); got != expected {
t.Errorf("Unexpected output from put. Got %q, expected %q", got, expected)
}
if expected1, expected2, got := "Put(myprofile1, ..., false)\nPut(myprofile2, ..., false)", "Put(myprofile2, ..., false)\nPut(myprofile1, ..., false)", strings.TrimSpace(serverOut.String()); got != expected1 && got != expected2 {
t.Errorf("Unexpected output from mock server. Got %q, expected %q or %q", got, expected1, expected2)
}
resetOut()
// Test the 'put' command with overwrite = true.
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"put", "--overwrite", appName, oneProfile, fileName}); err != nil {
t.Fatalf("%v", err)
}
if expected, got := "Application envelope added for profile myprofile.", strings.TrimSpace(stdout.String()); got != expected {
t.Errorf("Unexpected output from put. Got %q, expected %q", got, expected)
}
if expected, got := "Put(myprofile, ..., true)", strings.TrimSpace(serverOut.String()); got != expected {
t.Errorf("Unexpected output from mock server. Got %q, expected %q", got, expected)
}
resetOut()
// Test the 'put' command with no profiles.
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"put", appName, ",,,", fileName}); err == nil {
t.Errorf("Expected put with no profiles to fail")
} else if expected, got := "ERROR: put: no profiles specified", strings.TrimSpace(stderr.String()); !strings.HasPrefix(got, expected) {
t.Errorf("Unexpected stderr output from put. Got %q, expected %q", got, expected+" ...")
}
resetOut()
// Test the 'remove' command.
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"remove", appName, oneProfile}); err != nil {
t.Fatalf("%v", err)
}
if expected, got := "Application envelope removed successfully.", strings.TrimSpace(stdout.String()); got != expected {
t.Errorf("Unexpected output from remove. Got %q, expected %q", got, expected)
}
resetOut()
// Test the 'edit' command. (nothing changed)
env.Vars = map[string]string{"EDITOR": "true"}
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"edit", appName, oneProfile}); err != nil {
t.Fatalf("%v", err)
}
if expected, got := "Nothing changed", strings.TrimSpace(stdout.String()); got != expected {
t.Errorf("Unexpected output from edit. Got %q, expected %q", got, expected)
}
resetOut()
// Test the 'edit' command.
env.Vars = map[string]string{"EDITOR": "perl -pi -e 's/arg1/arg111/'"}
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"edit", appName, oneProfile}); err != nil {
t.Fatalf("%v", err)
}
if expected, got := "Application envelope added for profile myprofile.\nApplication envelope updated successfully.", strings.TrimSpace(stdout.String()); got != expected {
t.Errorf("Unexpected output from edit. Got %q, expected %q", got, expected)
}
resetOut()
// Test the 'edit' command with more than 1 profiles.
env.Vars = map[string]string{"EDITOR": "true"}
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"edit", appName, severalProfiles}); err == nil {
t.Errorf("Expected edit with two profiles to fail")
} else if expected, got := "ERROR: edit: incorrect number of profiles, expected 1, got 2", strings.TrimSpace(stderr.String()); !strings.HasPrefix(got, expected) {
t.Errorf("Unexpected stderr output from edit. Got %q, expected %q", got, expected+" ...")
}
resetOut()
// Test the 'profiles' command.
if err := v23cmd.ParseAndRunForTest(cmdRoot, ctx, env, []string{"profiles", appName}); err != nil {
t.Fatalf("%v", err)
}
if expected, got := profiles, strings.TrimSpace(stdout.String()); got != expected {
t.Errorf("Unexpected output from profiles. Got %q, expected %q", got, expected)
}
resetOut()
}