resource/package/apt/package_manager_test.go
// Copyright © 2016 Asteris, LLC
//
// 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 apt_test
import (
"fmt"
"os/exec"
"testing"
"github.com/asteris-llc/converge/resource/package/apt"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
)
// TestAptInstalledVersion validates that installation status is successfully
// validated
func TestAptInstalledVersion(t *testing.T) {
t.Parallel()
t.Run("when installed", func(t *testing.T) {
expected := "foo-0.1.2.3"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgInstalled)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
result, found := a.InstalledVersion("foo")
assert.True(t, found)
assert.Equal(t, expected, string(result))
})
t.Run("when not installed", func(t *testing.T) {
expected := ""
a := &apt.Manager{Sys: newRunner("", makeExitError("", 1))}
result, found := a.InstalledVersion("foo1")
assert.False(t, found)
assert.Equal(t, expected, string(result))
})
t.Run("when held", func(t *testing.T) {
expected := "foo-0.1.2.3"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgHold)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
result, found := a.InstalledVersion("foo")
assert.True(t, found)
assert.Equal(t, expected, string(result))
})
t.Run("when removed", func(t *testing.T) {
expected := ""
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgRemoved)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
result, found := a.InstalledVersion("foo")
assert.False(t, found)
assert.Equal(t, expected, string(result))
})
t.Run("when uninstalled", func(t *testing.T) {
expected := ""
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgUninstalled)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
result, found := a.InstalledVersion("foo")
assert.False(t, found)
assert.Equal(t, expected, string(result))
})
}
// TestAptInstallPackage validates that we successfully ask Apt to install a
// package
func TestAptInstallPackage(t *testing.T) {
t.Parallel()
t.Run("when installed", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgInstalled)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.InstallPackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 1)
})
t.Run("when not installed", func(t *testing.T) {
pkg := "foo1"
runner := newRunner("", makeExitError("", 1))
a := &apt.Manager{Sys: runner}
a.InstallPackage(pkg)
runner.AssertNumberOfCalls(t, "Run", 2)
})
t.Run("when removed", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgRemoved)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.InstallPackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 2)
})
t.Run("when uninstalled", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgUninstalled)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.InstallPackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 2)
})
t.Run("when held", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgHold)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.InstallPackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 1)
})
t.Run("when installation error", func(t *testing.T) {
pkg := "foo1"
runner := newRunner("", makeExitError("", 1))
a := &apt.Manager{Sys: runner}
_, err := a.InstallPackage(pkg)
assert.Error(t, err)
runner.AssertNumberOfCalls(t, "Run", 2)
})
}
// TestAptRemovePackage validates that we successfully ask Apt to uninstall a
// package
func TestAptRemovePackage(t *testing.T) {
t.Parallel()
t.Run("when installed", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgInstalled)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.RemovePackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 2)
})
t.Run("when held", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgHold)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.RemovePackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 2)
})
t.Run("when uninstalled", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgUninstalled)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.RemovePackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 1)
})
t.Run("when removed", func(t *testing.T) {
pkg := "foo1"
out := fmt.Sprintf("foo,%s,0.1.2.3", apt.PkgRemoved)
runner := newRunner(out, nil)
a := &apt.Manager{Sys: runner}
_, err := a.RemovePackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 1)
})
t.Run("when package never installed", func(t *testing.T) {
pkg := "foo1"
runner := newRunner("", makeExitError("", 1))
a := &apt.Manager{Sys: runner}
_, err := a.RemovePackage(pkg)
assert.NoError(t, err)
runner.AssertNumberOfCalls(t, "Run", 1)
})
}
// MockRunner mocks out SysCaller
type MockRunner struct {
mock.Mock
}
// Run mocks out Run
func (m *MockRunner) Run(cmd string) ([]byte, error) {
args := m.Called(1)
return args.Get(0).([]byte), args.Error(1)
}
// newRunner creates a new MockRunner that returns the output string and error
func newRunner(output string, err error) *MockRunner {
m := &MockRunner{}
m.On("Run", mock.Anything).Return([]byte(output), err)
return m
}
// makeExitError generates a new ExitError
func makeExitError(stderr string, exitCode uint32) error {
cmd := fmt.Sprintf("echo %q 1>&2; exit %d", stderr, exitCode)
_, err := exec.Command("/bin/bash", "-c", cmd).Output()
return err
}