asteris-llc/converge

View on GitHub
resource/package/apt/package_manager_test.go

Summary

Maintainability
D
1 day
Test Coverage
// 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
}