mirror of
https://github.com/TECHNOFAB11/jsonnet-bundler.git
synced 2025-12-12 08:00:05 +01:00
commit
7fc7c31856
21 changed files with 2754 additions and 501 deletions
|
|
@ -18,13 +18,13 @@ import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg"
|
kingpin "gopkg.in/alecthomas/kingpin.v2"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
||||||
"gopkg.in/alecthomas/kingpin.v2"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func initCommand(dir string) int {
|
func initCommand(dir string) int {
|
||||||
exists, err := pkg.FileExists(jsonnetfile.File)
|
exists, err := jsonnetfile.Exists(jsonnetfile.File)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
kingpin.Errorf("Failed to check for jsonnetfile.json: %v", err)
|
kingpin.Errorf("Failed to check for jsonnetfile.json: %v", err)
|
||||||
return 1
|
return 1
|
||||||
|
|
|
||||||
|
|
@ -15,104 +15,79 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
kingpin "gopkg.in/alecthomas/kingpin.v2"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg"
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg"
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
"gopkg.in/alecthomas/kingpin.v2"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func installCommand(dir, jsonnetHome string, uris ...string) int {
|
func installCommand(dir, jsonnetHome string, uris []string) int {
|
||||||
if dir == "" {
|
if dir == "" {
|
||||||
dir = "."
|
dir = "."
|
||||||
}
|
}
|
||||||
|
|
||||||
filename, isLock, err := jsonnetfile.Choose(dir)
|
jsonnetFile, err := jsonnetfile.Load(filepath.Join(dir, jsonnetfile.File))
|
||||||
if err != nil {
|
kingpin.FatalIfError(err, "failed to load jsonnetfile")
|
||||||
kingpin.Fatalf("failed to choose jsonnetfile: %v", err)
|
|
||||||
return 1
|
lockFile, err := jsonnetfile.Load(filepath.Join(dir, jsonnetfile.LockFile))
|
||||||
|
if !os.IsNotExist(err) {
|
||||||
|
kingpin.FatalIfError(err, "failed to load lockfile")
|
||||||
}
|
}
|
||||||
|
|
||||||
jsonnetFile, err := jsonnetfile.Load(filename)
|
kingpin.FatalIfError(
|
||||||
if err != nil {
|
os.MkdirAll(filepath.Join(dir, jsonnetHome, ".tmp"), os.ModePerm),
|
||||||
kingpin.Fatalf("failed to load jsonnetfile: %v", err)
|
"creating vendor folder")
|
||||||
return 1
|
|
||||||
|
for _, u := range uris {
|
||||||
|
d := parseDependency(dir, u)
|
||||||
|
if d == nil {
|
||||||
|
kingpin.Fatalf("Unable to parse package URI `%s`", u)
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(uris) > 0 {
|
if !depEqual(jsonnetFile.Dependencies[d.Name], *d) {
|
||||||
for _, uri := range uris {
|
// the dep passed on the cli is different from the jsonnetFile
|
||||||
newDep := parseDependency(dir, uri)
|
jsonnetFile.Dependencies[d.Name] = *d
|
||||||
if newDep == nil {
|
|
||||||
kingpin.Errorf("ignoring unrecognized uri: %s", uri)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
oldDeps := jsonnetFile.Dependencies
|
// we want to install the passed version (ignore the lock)
|
||||||
newDeps := []spec.Dependency{}
|
delete(lockFile.Dependencies, d.Name)
|
||||||
oldDepReplaced := false
|
|
||||||
for _, d := range oldDeps {
|
|
||||||
if d.Name == newDep.Name {
|
|
||||||
newDeps = append(newDeps, *newDep)
|
|
||||||
oldDepReplaced = true
|
|
||||||
} else {
|
|
||||||
newDeps = append(newDeps, d)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !oldDepReplaced {
|
locked, err := pkg.Ensure(jsonnetFile, jsonnetHome, lockFile.Dependencies)
|
||||||
newDeps = append(newDeps, *newDep)
|
kingpin.FatalIfError(err, "failed to install packages")
|
||||||
}
|
|
||||||
|
|
||||||
jsonnetFile.Dependencies = newDeps
|
kingpin.FatalIfError(
|
||||||
}
|
writeJSONFile(filepath.Join(dir, jsonnetfile.File), jsonnetFile),
|
||||||
}
|
"updating jsonnetfile.json")
|
||||||
|
kingpin.FatalIfError(
|
||||||
srcPath := filepath.Join(jsonnetHome)
|
writeJSONFile(filepath.Join(dir, jsonnetfile.LockFile), spec.JsonnetFile{Dependencies: locked}),
|
||||||
err = os.MkdirAll(srcPath, os.ModePerm)
|
"updating jsonnetfile.lock.json")
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to create jsonnet home path: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
|
|
||||||
lock, err := pkg.Install(context.TODO(), isLock, filename, jsonnetFile, jsonnetHome)
|
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to install: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
|
|
||||||
// If installing from lock file there is no need to write any files back.
|
|
||||||
if !isLock {
|
|
||||||
b, err := json.MarshalIndent(jsonnetFile, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to encode jsonnet file: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
b = append(b, []byte("\n")...)
|
|
||||||
|
|
||||||
err = ioutil.WriteFile(filepath.Join(dir, jsonnetfile.File), b, 0644)
|
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to write jsonnet file: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
|
|
||||||
b, err = json.MarshalIndent(lock, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to encode jsonnet file: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
b = append(b, []byte("\n")...)
|
|
||||||
|
|
||||||
err = ioutil.WriteFile(filepath.Join(dir, jsonnetfile.LockFile), b, 0644)
|
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to write lock file: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func depEqual(d1, d2 spec.Dependency) bool {
|
||||||
|
name := d1.Name == d2.Name
|
||||||
|
version := d1.Version == d2.Version
|
||||||
|
source := reflect.DeepEqual(d1.Source, d2.Source)
|
||||||
|
|
||||||
|
return name && version && source
|
||||||
|
}
|
||||||
|
|
||||||
|
func writeJSONFile(name string, d interface{}) error {
|
||||||
|
b, err := json.MarshalIndent(d, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "encoding json")
|
||||||
|
}
|
||||||
|
b = append(b, []byte("\n")...)
|
||||||
|
|
||||||
|
return ioutil.WriteFile(name, b, 0644)
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -21,8 +21,9 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestInstallCommand(t *testing.T) {
|
func TestInstallCommand(t *testing.T) {
|
||||||
|
|
@ -43,7 +44,7 @@ func TestInstallCommand(t *testing.T) {
|
||||||
URIs: []string{"github.com/jsonnet-bundler/jsonnet-bundler@v0.1.0"},
|
URIs: []string{"github.com/jsonnet-bundler/jsonnet-bundler@v0.1.0"},
|
||||||
ExpectedCode: 0,
|
ExpectedCode: 0,
|
||||||
ExpectedJsonnetFile: []byte(`{"dependencies": [{"name": "jsonnet-bundler", "source": {"git": {"remote": "https://github.com/jsonnet-bundler/jsonnet-bundler", "subdir": ""}}, "version": "v0.1.0"}]}`),
|
ExpectedJsonnetFile: []byte(`{"dependencies": [{"name": "jsonnet-bundler", "source": {"git": {"remote": "https://github.com/jsonnet-bundler/jsonnet-bundler", "subdir": ""}}, "version": "v0.1.0"}]}`),
|
||||||
ExpectedJsonnetLockFile: []byte(`{"dependencies": [{"name": "jsonnet-bundler", "source": {"git": {"remote": "https://github.com/jsonnet-bundler/jsonnet-bundler", "subdir": ""}}, "version": "080f157c7fb85ad0281ea78f6c641eaa570a582f"}]}`),
|
ExpectedJsonnetLockFile: []byte(`{"dependencies": [{"name": "jsonnet-bundler", "source": {"git": {"remote": "https://github.com/jsonnet-bundler/jsonnet-bundler", "subdir": ""}}, "version": "080f157c7fb85ad0281ea78f6c641eaa570a582f", "sum": "W1uI550rQ66axRpPXA2EZDquyPg/5PHZlvUz1NEzefg="}]}`),
|
||||||
}, {
|
}, {
|
||||||
Name: "Relative",
|
Name: "Relative",
|
||||||
URIs: []string{"jsonnet/foobar"},
|
URIs: []string{"jsonnet/foobar"},
|
||||||
|
|
@ -73,7 +74,7 @@ func TestInstallCommand(t *testing.T) {
|
||||||
|
|
||||||
jsonnetFileContent(t, jsonnetfile.File, []byte(`{}`))
|
jsonnetFileContent(t, jsonnetfile.File, []byte(`{}`))
|
||||||
|
|
||||||
installCommand("", "vendor", tc.URIs...)
|
installCommand("", "vendor", tc.URIs)
|
||||||
|
|
||||||
jsonnetFileContent(t, jsonnetfile.File, tc.ExpectedJsonnetFile)
|
jsonnetFileContent(t, jsonnetfile.File, tc.ExpectedJsonnetFile)
|
||||||
jsonnetFileContent(t, jsonnetfile.LockFile, tc.ExpectedJsonnetLockFile)
|
jsonnetFileContent(t, jsonnetfile.LockFile, tc.ExpectedJsonnetLockFile)
|
||||||
|
|
|
||||||
|
|
@ -20,11 +20,12 @@ import (
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
|
||||||
|
"github.com/fatih/color"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
kingpin "gopkg.in/alecthomas/kingpin.v2"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
"github.com/pkg/errors"
|
|
||||||
"gopkg.in/alecthomas/kingpin.v2"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
|
@ -34,10 +35,10 @@ const (
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
gitSSHRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git")
|
gitSSHRegex = regexp.MustCompile(`git\+ssh://git@([^:]+):([^/]+)/([^/]+).git`)
|
||||||
gitSSHWithVersionRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git@(.*)")
|
gitSSHWithVersionRegex = regexp.MustCompile(`git\+ssh://git@([^:]+):([^/]+)/([^/]+).git@(.*)`)
|
||||||
gitSSHWithPathRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git/(.*)")
|
gitSSHWithPathRegex = regexp.MustCompile(`git\+ssh://git@([^:]+):([^/]+)/([^/]+).git/(.*)`)
|
||||||
gitSSHWithPathAndVersionRegex = regexp.MustCompile("git\\+ssh://git@([^:]+):([^/]+)/([^/]+).git/(.*)@(.*)")
|
gitSSHWithPathAndVersionRegex = regexp.MustCompile(`git\+ssh://git@([^:]+):([^/]+)/([^/]+).git/(.*)@(.*)`)
|
||||||
|
|
||||||
githubSlugRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)")
|
githubSlugRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)")
|
||||||
githubSlugWithVersionRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)@(.*)")
|
githubSlugWithVersionRegex = regexp.MustCompile("github.com/([-_a-zA-Z0-9]+)/([-_a-zA-Z0-9]+)@(.*)")
|
||||||
|
|
@ -54,6 +55,8 @@ func Main() int {
|
||||||
JsonnetHome string
|
JsonnetHome string
|
||||||
}{}
|
}{}
|
||||||
|
|
||||||
|
color.Output = color.Error
|
||||||
|
|
||||||
a := kingpin.New(filepath.Base(os.Args[0]), "A jsonnet package manager")
|
a := kingpin.New(filepath.Base(os.Args[0]), "A jsonnet package manager")
|
||||||
a.HelpFlag.Short('h')
|
a.HelpFlag.Short('h')
|
||||||
|
|
||||||
|
|
@ -83,63 +86,60 @@ func Main() int {
|
||||||
case initCmd.FullCommand():
|
case initCmd.FullCommand():
|
||||||
return initCommand(workdir)
|
return initCommand(workdir)
|
||||||
case installCmd.FullCommand():
|
case installCmd.FullCommand():
|
||||||
return installCommand(workdir, cfg.JsonnetHome, *installCmdURIs...)
|
return installCommand(workdir, cfg.JsonnetHome, *installCmdURIs)
|
||||||
case updateCmd.FullCommand():
|
case updateCmd.FullCommand():
|
||||||
return updateCommand(cfg.JsonnetHome)
|
return updateCommand(workdir, cfg.JsonnetHome)
|
||||||
default:
|
default:
|
||||||
installCommand(workdir, cfg.JsonnetHome)
|
installCommand(workdir, cfg.JsonnetHome, []string{})
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseDependency(dir, uri string) *spec.Dependency {
|
func parseDependency(dir, uri string) *spec.Dependency {
|
||||||
if d := parseGitSSHDependency(uri); d != nil {
|
if uri == "" {
|
||||||
return d
|
|
||||||
}
|
|
||||||
|
|
||||||
if d := parseGithubDependency(uri); d != nil {
|
|
||||||
return d
|
|
||||||
}
|
|
||||||
|
|
||||||
if d := parseLocalDependency(dir, uri); d != nil {
|
|
||||||
return d
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if githubSlugRegex.MatchString(uri) {
|
||||||
|
return parseGithubDependency(uri)
|
||||||
|
}
|
||||||
|
|
||||||
|
if gitSSHRegex.MatchString(uri) {
|
||||||
|
return parseGitSSHDependency(uri)
|
||||||
|
}
|
||||||
|
|
||||||
|
return parseLocalDependency(dir, uri)
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseGitSSHDependency(p string) *spec.Dependency {
|
func parseGitSSHDependency(p string) *spec.Dependency {
|
||||||
if !gitSSHRegex.MatchString(p) {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
subdir := ""
|
subdir := ""
|
||||||
host := ""
|
host := ""
|
||||||
org := ""
|
org := ""
|
||||||
repo := ""
|
repo := ""
|
||||||
version := "master"
|
version := "master"
|
||||||
|
|
||||||
if gitSSHWithPathAndVersionRegex.MatchString(p) {
|
switch {
|
||||||
|
case gitSSHWithPathAndVersionRegex.MatchString(p):
|
||||||
matches := gitSSHWithPathAndVersionRegex.FindStringSubmatch(p)
|
matches := gitSSHWithPathAndVersionRegex.FindStringSubmatch(p)
|
||||||
host = matches[1]
|
host = matches[1]
|
||||||
org = matches[2]
|
org = matches[2]
|
||||||
repo = matches[3]
|
repo = matches[3]
|
||||||
subdir = matches[4]
|
subdir = matches[4]
|
||||||
version = matches[5]
|
version = matches[5]
|
||||||
} else if gitSSHWithPathRegex.MatchString(p) {
|
case gitSSHWithPathRegex.MatchString(p):
|
||||||
matches := gitSSHWithPathRegex.FindStringSubmatch(p)
|
matches := gitSSHWithPathRegex.FindStringSubmatch(p)
|
||||||
host = matches[1]
|
host = matches[1]
|
||||||
org = matches[2]
|
org = matches[2]
|
||||||
repo = matches[3]
|
repo = matches[3]
|
||||||
subdir = matches[4]
|
subdir = matches[4]
|
||||||
} else if gitSSHWithVersionRegex.MatchString(p) {
|
case gitSSHWithVersionRegex.MatchString(p):
|
||||||
matches := gitSSHWithVersionRegex.FindStringSubmatch(p)
|
matches := gitSSHWithVersionRegex.FindStringSubmatch(p)
|
||||||
host = matches[1]
|
host = matches[1]
|
||||||
org = matches[2]
|
org = matches[2]
|
||||||
repo = matches[3]
|
repo = matches[3]
|
||||||
version = matches[4]
|
version = matches[4]
|
||||||
} else {
|
default:
|
||||||
matches := gitSSHRegex.FindStringSubmatch(p)
|
matches := gitSSHRegex.FindStringSubmatch(p)
|
||||||
host = matches[1]
|
host = matches[1]
|
||||||
org = matches[2]
|
org = matches[2]
|
||||||
|
|
@ -212,16 +212,6 @@ func parseGithubDependency(p string) *spec.Dependency {
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseLocalDependency(dir, p string) *spec.Dependency {
|
func parseLocalDependency(dir, p string) *spec.Dependency {
|
||||||
if p == "" {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if strings.HasPrefix(p, "github.com") {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if strings.HasPrefix(p, "git+ssh") {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
clean := filepath.Clean(p)
|
clean := filepath.Clean(p)
|
||||||
abs := filepath.Join(dir, clean)
|
abs := filepath.Join(dir, clean)
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -18,8 +18,9 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestParseDependency(t *testing.T) {
|
func TestParseDependency(t *testing.T) {
|
||||||
|
|
|
||||||
|
|
@ -15,50 +15,39 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"encoding/json"
|
|
||||||
"io/ioutil"
|
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
kingpin "gopkg.in/alecthomas/kingpin.v2"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg"
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg"
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
||||||
"gopkg.in/alecthomas/kingpin.v2"
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
)
|
)
|
||||||
|
|
||||||
func updateCommand(jsonnetHome string, urls ...*url.URL) int {
|
func updateCommand(dir, jsonnetHome string, urls ...*url.URL) int {
|
||||||
m, err := pkg.LoadJsonnetfile(jsonnetfile.File)
|
if dir == "" {
|
||||||
if err != nil {
|
dir = "."
|
||||||
kingpin.Fatalf("failed to load jsonnetfile: %v", err)
|
|
||||||
return 1
|
|
||||||
}
|
}
|
||||||
|
|
||||||
err = os.MkdirAll(jsonnetHome, os.ModePerm)
|
jsonnetFile, err := jsonnetfile.Load(filepath.Join(dir, jsonnetfile.File))
|
||||||
if err != nil {
|
kingpin.FatalIfError(err, "failed to load jsonnetfile")
|
||||||
kingpin.Fatalf("failed to create jsonnet home path: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
|
|
||||||
// When updating, the lockfile is explicitly ignored.
|
kingpin.FatalIfError(
|
||||||
isLock := false
|
os.MkdirAll(filepath.Join(dir, jsonnetHome, ".tmp"), os.ModePerm),
|
||||||
lock, err := pkg.Install(context.TODO(), isLock, jsonnetfile.File, m, jsonnetHome)
|
"creating vendor folder")
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to install: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
|
|
||||||
b, err := json.MarshalIndent(lock, "", " ")
|
// When updating, locks are ignored.
|
||||||
if err != nil {
|
locks := map[string]spec.Dependency{}
|
||||||
kingpin.Fatalf("failed to encode jsonnet file: %v", err)
|
locked, err := pkg.Ensure(jsonnetFile, jsonnetHome, locks)
|
||||||
return 3
|
kingpin.FatalIfError(err, "failed to install packages")
|
||||||
}
|
|
||||||
b = append(b, []byte("\n")...)
|
|
||||||
|
|
||||||
err = ioutil.WriteFile(jsonnetfile.LockFile, b, 0644)
|
|
||||||
if err != nil {
|
|
||||||
kingpin.Fatalf("failed to write lock file: %v", err)
|
|
||||||
return 3
|
|
||||||
}
|
|
||||||
|
|
||||||
|
kingpin.FatalIfError(
|
||||||
|
writeJSONFile(filepath.Join(dir, jsonnetfile.File), jsonnetFile),
|
||||||
|
"updating jsonnetfile.json")
|
||||||
|
kingpin.FatalIfError(
|
||||||
|
writeJSONFile(filepath.Join(dir, jsonnetfile.LockFile), spec.JsonnetFile{Dependencies: locked}),
|
||||||
|
"updating jsonnetfile.lock.json")
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -31,8 +31,9 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/fatih/color"
|
"github.com/fatih/color"
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
|
||||||
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
)
|
)
|
||||||
|
|
||||||
type GitPackage struct {
|
type GitPackage struct {
|
||||||
|
|
|
||||||
|
|
@ -18,56 +18,40 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
"github.com/pkg/errors"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const File = "jsonnetfile.json"
|
const (
|
||||||
const LockFile = "jsonnetfile.lock.json"
|
File = "jsonnetfile.json"
|
||||||
|
LockFile = "jsonnetfile.lock.json"
|
||||||
|
)
|
||||||
|
|
||||||
var ErrNoFile = errors.New("no jsonnetfile")
|
var ErrNoFile = errors.New("no jsonnetfile")
|
||||||
|
|
||||||
func Choose(dir string) (string, bool, error) {
|
// Load reads a jsonnetfile.(lock).json from disk
|
||||||
jsonnetfileLock := path.Join(dir, LockFile)
|
|
||||||
jsonnetfile := path.Join(dir, File)
|
|
||||||
|
|
||||||
lockExists, err := fileExists(jsonnetfileLock)
|
|
||||||
if err != nil {
|
|
||||||
return "", false, err
|
|
||||||
}
|
|
||||||
if lockExists {
|
|
||||||
return jsonnetfileLock, true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
fileExists, err := fileExists(jsonnetfile)
|
|
||||||
if err != nil {
|
|
||||||
return "", false, err
|
|
||||||
}
|
|
||||||
if fileExists {
|
|
||||||
return jsonnetfile, false, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return "", false, ErrNoFile
|
|
||||||
}
|
|
||||||
|
|
||||||
func Load(filepath string) (spec.JsonnetFile, error) {
|
func Load(filepath string) (spec.JsonnetFile, error) {
|
||||||
m := spec.JsonnetFile{}
|
m := spec.New()
|
||||||
|
|
||||||
bytes, err := ioutil.ReadFile(filepath)
|
bytes, err := ioutil.ReadFile(filepath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return m, errors.Wrap(err, "failed to read file")
|
return m, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := json.Unmarshal(bytes, &m); err != nil {
|
if err := json.Unmarshal(bytes, &m); err != nil {
|
||||||
return m, errors.Wrap(err, "failed to unmarshal file")
|
return m, errors.Wrap(err, "failed to unmarshal file")
|
||||||
}
|
}
|
||||||
|
if m.Dependencies == nil {
|
||||||
|
m.Dependencies = make(map[string]spec.Dependency)
|
||||||
|
}
|
||||||
|
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func fileExists(path string) (bool, error) {
|
// Exists returns whether the file at the given path exists
|
||||||
|
func Exists(path string) (bool, error) {
|
||||||
_, err := os.Stat(path)
|
_, err := os.Stat(path)
|
||||||
if os.IsNotExist(err) {
|
if os.IsNotExist(err) {
|
||||||
return false, nil
|
return false, nil
|
||||||
|
|
|
||||||
|
|
@ -18,75 +18,18 @@ import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const notExist = "/this/does/not/exist"
|
const notExist = "/this/does/not/exist"
|
||||||
|
|
||||||
func TestChoose(t *testing.T) {
|
|
||||||
testcases := []struct {
|
|
||||||
Name string
|
|
||||||
Jsonnetfile []byte
|
|
||||||
JsonnetfileLock []byte
|
|
||||||
ExpectedFilename string
|
|
||||||
ExpectedLock bool
|
|
||||||
ExpectedError error
|
|
||||||
}{{
|
|
||||||
Name: "NoFiles",
|
|
||||||
ExpectedFilename: "",
|
|
||||||
ExpectedLock: false,
|
|
||||||
ExpectedError: jsonnetfile.ErrNoFile,
|
|
||||||
}, {
|
|
||||||
Name: "Jsonnetfile",
|
|
||||||
Jsonnetfile: []byte(`{}`),
|
|
||||||
ExpectedFilename: jsonnetfile.File,
|
|
||||||
ExpectedLock: false,
|
|
||||||
ExpectedError: nil,
|
|
||||||
}, {
|
|
||||||
Name: "JsonnetfileLock",
|
|
||||||
Jsonnetfile: []byte(`{}`),
|
|
||||||
JsonnetfileLock: []byte(`{}`),
|
|
||||||
ExpectedFilename: jsonnetfile.LockFile,
|
|
||||||
ExpectedLock: true,
|
|
||||||
ExpectedError: nil,
|
|
||||||
}}
|
|
||||||
|
|
||||||
for _, tc := range testcases {
|
|
||||||
t.Run(tc.Name, func(t *testing.T) {
|
|
||||||
dir, err := ioutil.TempDir("", "jsonnetfile-choose")
|
|
||||||
assert.Nil(t, err)
|
|
||||||
defer os.Remove(dir)
|
|
||||||
|
|
||||||
if tc.Jsonnetfile != nil {
|
|
||||||
err := ioutil.WriteFile(filepath.Join(dir, jsonnetfile.File), tc.Jsonnetfile, os.ModePerm)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
}
|
|
||||||
if tc.JsonnetfileLock != nil {
|
|
||||||
err := ioutil.WriteFile(filepath.Join(dir, jsonnetfile.LockFile), tc.JsonnetfileLock, os.ModePerm)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
filename, isLock, err := jsonnetfile.Choose(dir)
|
|
||||||
|
|
||||||
assert.Equal(t, tc.ExpectedFilename, strings.TrimPrefix(filename, dir+"/"))
|
|
||||||
assert.Equal(t, tc.ExpectedLock, isLock)
|
|
||||||
|
|
||||||
if tc.ExpectedError != nil {
|
|
||||||
assert.EqualError(t, err, tc.ExpectedError.Error())
|
|
||||||
} else {
|
|
||||||
assert.NoError(t, err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestLoad(t *testing.T) {
|
func TestLoad(t *testing.T) {
|
||||||
empty := spec.JsonnetFile{}
|
empty := spec.New()
|
||||||
|
|
||||||
jsonnetfileContent := `{
|
jsonnetfileContent := `{
|
||||||
"dependencies": [
|
"dependencies": [
|
||||||
|
|
@ -104,7 +47,8 @@ func TestLoad(t *testing.T) {
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
jsonnetFileExpected := spec.JsonnetFile{
|
jsonnetFileExpected := spec.JsonnetFile{
|
||||||
Dependencies: []spec.Dependency{{
|
Dependencies: map[string]spec.Dependency{
|
||||||
|
"foobar": {
|
||||||
Name: "foobar",
|
Name: "foobar",
|
||||||
Source: spec.Source{
|
Source: spec.Source{
|
||||||
GitSource: &spec.GitSource{
|
GitSource: &spec.GitSource{
|
||||||
|
|
@ -159,3 +103,26 @@ func TestLoad(t *testing.T) {
|
||||||
assert.Equal(t, jsonnetFileExpected, jf)
|
assert.Equal(t, jsonnetFileExpected, jf)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestFileExists(t *testing.T) {
|
||||||
|
{
|
||||||
|
exists, err := jsonnetfile.Exists(notExist)
|
||||||
|
assert.False(t, exists)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
tempFile, err := ioutil.TempFile("", "jb-exists")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
err := os.Remove(tempFile.Name())
|
||||||
|
assert.Nil(t, err)
|
||||||
|
}()
|
||||||
|
|
||||||
|
exists, err := jsonnetfile.Exists(tempFile.Name())
|
||||||
|
assert.True(t, exists)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
265
pkg/packages.go
265
pkg/packages.go
|
|
@ -16,166 +16,209 @@ package pkg
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"crypto/sha256"
|
||||||
|
"encoding/base64"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
"github.com/fatih/color"
|
"github.com/fatih/color"
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
||||||
"github.com/pkg/errors"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
VersionMismatch = errors.New("multiple colliding versions specified")
|
VersionMismatch = errors.New("multiple colliding versions specified")
|
||||||
)
|
)
|
||||||
|
|
||||||
func Install(ctx context.Context, isLock bool, dependencySourceIdentifier string, m spec.JsonnetFile, dir string) (*spec.JsonnetFile, error) {
|
// Ensure receives all direct packages as, the directory to vendor in and all known locks.
|
||||||
lockfile := &spec.JsonnetFile{}
|
// It then makes sure all direct and nested dependencies are present in vendor at the correct version:
|
||||||
for _, dep := range m.Dependencies {
|
//
|
||||||
|
// If the package is locked and the files in vendor match the sha256 checksum,
|
||||||
tmp := filepath.Join(dir, ".tmp")
|
// nothing needs to be done. Otherwise, the package is retrieved from the
|
||||||
err := os.MkdirAll(tmp, os.ModePerm)
|
// upstream source and added into vendor. If previously locked, the sums are
|
||||||
|
// checked as well.
|
||||||
|
// In case a (nested) package is already present in the lock,
|
||||||
|
// the one from the lock takes precedence. This allows the user to set the
|
||||||
|
// desired version in case by `jb install`ing it.
|
||||||
|
//
|
||||||
|
// Finally, all unknown files and directories are removed from vendor/
|
||||||
|
func Ensure(direct spec.JsonnetFile, vendorDir string, locks map[string]spec.Dependency) (map[string]spec.Dependency, error) {
|
||||||
|
// ensure all required files are in vendor
|
||||||
|
deps, err := ensure(direct.Dependencies, vendorDir, locks)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "failed to create general tmp dir")
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
var p Interface
|
// cleanup unknown dirs from vendor/
|
||||||
if dep.Source.GitSource != nil {
|
f, err := os.Open(vendorDir)
|
||||||
p = NewGitPackage(dep.Source.GitSource)
|
|
||||||
}
|
|
||||||
if dep.Source.LocalSource != nil {
|
|
||||||
p = NewLocalPackage(dep.Source.LocalSource)
|
|
||||||
}
|
|
||||||
|
|
||||||
lockVersion, err := p.Install(ctx, dep.Name, dir, dep.Version)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "failed to install package")
|
return nil, err
|
||||||
}
|
}
|
||||||
|
names, err := f.Readdirnames(0)
|
||||||
color.Green(">>> Installed %s version %s\n", dep.Name, dep.Version)
|
|
||||||
|
|
||||||
destPath := path.Join(dir, dep.Name)
|
|
||||||
|
|
||||||
lockfile.Dependencies, err = insertDependency(lockfile.Dependencies, spec.Dependency{
|
|
||||||
Name: dep.Name,
|
|
||||||
Source: dep.Source,
|
|
||||||
Version: lockVersion,
|
|
||||||
DepSource: dependencySourceIdentifier,
|
|
||||||
})
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "failed to insert dependency to lock dependencies")
|
return nil, err
|
||||||
|
}
|
||||||
|
for _, name := range names {
|
||||||
|
if _, ok := deps[name]; !ok {
|
||||||
|
dir := filepath.Join(vendorDir, name)
|
||||||
|
if err := os.RemoveAll(dir); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if name != ".tmp" {
|
||||||
|
color.Magenta("CLEAN %s", dir)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// If dependencies are being installed from a lock file, the transitive
|
// return the final lockfile contents
|
||||||
// dependencies are not questioned, the locked dependencies are just
|
return deps, nil
|
||||||
// installed.
|
}
|
||||||
if isLock {
|
|
||||||
|
func ensure(direct map[string]spec.Dependency, vendorDir string, locks map[string]spec.Dependency) (map[string]spec.Dependency, error) {
|
||||||
|
deps := make(map[string]spec.Dependency)
|
||||||
|
|
||||||
|
for _, d := range direct {
|
||||||
|
l, present := locks[d.Name]
|
||||||
|
|
||||||
|
// already locked and the integrity is intact
|
||||||
|
if present {
|
||||||
|
d.Version = locks[d.Name].Version
|
||||||
|
|
||||||
|
if check(l, vendorDir) {
|
||||||
|
deps[d.Name] = l
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
expectedSum := locks[d.Name].Sum
|
||||||
|
|
||||||
filepath, isLock, err := ChooseJsonnetFile(destPath)
|
// either not present or not intact: download again
|
||||||
|
dir := filepath.Join(vendorDir, d.Name)
|
||||||
|
os.RemoveAll(dir)
|
||||||
|
|
||||||
|
locked, err := download(d, vendorDir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, errors.Wrap(err, "downloading")
|
||||||
}
|
}
|
||||||
depsDeps, err := LoadJsonnetfile(filepath)
|
if expectedSum != "" && locked.Sum != expectedSum {
|
||||||
// It is ok for dependencies not to have a JsonnetFile, it just means
|
return nil, fmt.Errorf("checksum mismatch for %s. Expected %s but got %s", d.Name, expectedSum, locked.Sum)
|
||||||
// they do not have transitive dependencies of their own.
|
}
|
||||||
if err != nil && !os.IsNotExist(err) {
|
deps[d.Name] = *locked
|
||||||
return nil, err
|
// we settled on a new version, add it to the locks for recursion
|
||||||
|
locks[d.Name] = *locked
|
||||||
}
|
}
|
||||||
|
|
||||||
depsInstalledByDependency, err := Install(ctx, isLock, filepath, depsDeps, dir)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, d := range depsInstalledByDependency.Dependencies {
|
|
||||||
lockfile.Dependencies, err = insertDependency(lockfile.Dependencies, d)
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrap(err, "failed to insert dependency to lock dependencies")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return lockfile, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func insertDependency(deps []spec.Dependency, newDep spec.Dependency) ([]spec.Dependency, error) {
|
|
||||||
if len(deps) == 0 {
|
|
||||||
return []spec.Dependency{newDep}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
res := []spec.Dependency{}
|
|
||||||
newDepPreviouslyPresent := false
|
|
||||||
for _, d := range deps {
|
for _, d := range deps {
|
||||||
if d.Name == newDep.Name {
|
f, err := jsonnetfile.Load(filepath.Join(vendorDir, d.Name, jsonnetfile.File))
|
||||||
if d.Version != newDep.Version {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("multiple colliding versions specified for %s: %s (from %s) and %s (from %s)", d.Name, d.Version, d.DepSource, newDep.Version, newDep.DepSource)
|
|
||||||
}
|
|
||||||
res = append(res, d)
|
|
||||||
newDepPreviouslyPresent = true
|
|
||||||
} else {
|
|
||||||
res = append(res, d)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !newDepPreviouslyPresent {
|
|
||||||
res = append(res, newDep)
|
|
||||||
}
|
|
||||||
|
|
||||||
return res, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func FileExists(path string) (bool, error) {
|
|
||||||
_, err := os.Stat(path)
|
|
||||||
if os.IsNotExist(err) {
|
if os.IsNotExist(err) {
|
||||||
return false, nil
|
continue
|
||||||
}
|
}
|
||||||
if err != nil {
|
return nil, err
|
||||||
return false, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true, nil
|
nested, err := ensure(f.Dependencies, vendorDir, locks)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, d := range nested {
|
||||||
|
if _, ok := deps[d.Name]; !ok {
|
||||||
|
deps[d.Name] = d
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return deps, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func ChooseJsonnetFile(dir string) (string, bool, error) {
|
// download retrieves a package from a remote upstream. The checksum of the
|
||||||
lockfilePath := path.Join(dir, jsonnetfile.LockFile)
|
// files is generated afterwards.
|
||||||
jsonnetfilePath := path.Join(dir, jsonnetfile.File)
|
func download(d spec.Dependency, vendorDir string) (*spec.Dependency, error) {
|
||||||
filename := lockfilePath
|
var p Interface
|
||||||
isLock := true
|
switch {
|
||||||
|
case d.Source.GitSource != nil:
|
||||||
|
p = NewGitPackage(d.Source.GitSource)
|
||||||
|
case d.Source.LocalSource != nil:
|
||||||
|
p = NewLocalPackage(d.Source.LocalSource)
|
||||||
|
}
|
||||||
|
|
||||||
lockExists, err := FileExists(filepath.Join(dir, jsonnetfile.LockFile))
|
if p == nil {
|
||||||
|
return nil, errors.New("either git or local source is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
version, err := p.Install(context.TODO(), d.Name, vendorDir, d.Version)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", false, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if !lockExists {
|
var sum string
|
||||||
filename = jsonnetfilePath
|
if d.Source.LocalSource == nil {
|
||||||
isLock = false
|
sum = hashDir(filepath.Join(vendorDir, d.Name))
|
||||||
}
|
}
|
||||||
|
|
||||||
return filename, isLock, err
|
return &spec.Dependency{
|
||||||
|
Name: d.Name,
|
||||||
|
Source: d.Source,
|
||||||
|
Version: version,
|
||||||
|
Sum: sum,
|
||||||
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func LoadJsonnetfile(filepath string) (spec.JsonnetFile, error) {
|
// check returns whether the files present at the vendor/ folder match the
|
||||||
m := spec.JsonnetFile{}
|
// sha256 sum of the package. local-directory dependencies are not checked as
|
||||||
|
// their purpose is to change during development where integrity checking would
|
||||||
if _, err := os.Stat(filepath); err != nil {
|
// be a hindrance.
|
||||||
return m, err
|
func check(d spec.Dependency, vendorDir string) bool {
|
||||||
|
// assume a local dependency is intact as long as it exists
|
||||||
|
if d.Source.LocalSource != nil {
|
||||||
|
x, err := jsonnetfile.Exists(filepath.Join(vendorDir, d.Name))
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return x
|
||||||
}
|
}
|
||||||
|
|
||||||
f, err := os.Open(filepath)
|
if d.Sum == "" {
|
||||||
|
// no sum available, need to download
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
dir := filepath.Join(vendorDir, d.Name)
|
||||||
|
sum := hashDir(dir)
|
||||||
|
return d.Sum == sum
|
||||||
|
}
|
||||||
|
|
||||||
|
// hashDir computes the checksum of a directory by concatenating all files and
|
||||||
|
// hashing this data using sha256. This can be memory heavy with lots of data,
|
||||||
|
// but jsonnet files should be fairly small
|
||||||
|
func hashDir(dir string) string {
|
||||||
|
hasher := sha256.New()
|
||||||
|
|
||||||
|
filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return m, err
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if info.IsDir() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := os.Open(path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
|
|
||||||
err = json.NewDecoder(f).Decode(&m)
|
if _, err := io.Copy(hasher, f); err != nil {
|
||||||
if err != nil {
|
return err
|
||||||
return m, err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return m, nil
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
return base64.StdEncoding.EncodeToString(hasher.Sum(nil))
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -13,128 +13,3 @@
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
package pkg
|
package pkg
|
||||||
|
|
||||||
import (
|
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/pkg/jsonnetfile"
|
|
||||||
"github.com/jsonnet-bundler/jsonnet-bundler/spec"
|
|
||||||
"github.com/stretchr/testify/assert"
|
|
||||||
)
|
|
||||||
|
|
||||||
const NotExist = "/this/does/not/exist"
|
|
||||||
|
|
||||||
func TestInsertDependency(t *testing.T) {
|
|
||||||
deps := []spec.Dependency{{Name: "test1", Version: "latest"}}
|
|
||||||
dep := spec.Dependency{Name: "test2", Version: "latest"}
|
|
||||||
|
|
||||||
res, err := insertDependency(deps, dep)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(res) != 2 {
|
|
||||||
t.Fatal("Incorrectly inserted")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFileExists(t *testing.T) {
|
|
||||||
{
|
|
||||||
exists, err := FileExists(NotExist)
|
|
||||||
assert.False(t, exists)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
}
|
|
||||||
{
|
|
||||||
tempFile, err := ioutil.TempFile("", "jb-exists")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
defer func() {
|
|
||||||
err := os.Remove(tempFile.Name())
|
|
||||||
assert.Nil(t, err)
|
|
||||||
}()
|
|
||||||
|
|
||||||
exists, err := FileExists(tempFile.Name())
|
|
||||||
assert.True(t, exists)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestLoadJsonnetfile(t *testing.T) {
|
|
||||||
empty := spec.JsonnetFile{}
|
|
||||||
|
|
||||||
jsonnetfileContent := `{
|
|
||||||
"dependencies": [
|
|
||||||
{
|
|
||||||
"name": "foobar",
|
|
||||||
"source": {
|
|
||||||
"git": {
|
|
||||||
"remote": "https://github.com/foobar/foobar",
|
|
||||||
"subdir": ""
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"version": "master"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
`
|
|
||||||
jsonnetFileExpected := spec.JsonnetFile{
|
|
||||||
Dependencies: []spec.Dependency{{
|
|
||||||
Name: "foobar",
|
|
||||||
Source: spec.Source{
|
|
||||||
GitSource: &spec.GitSource{
|
|
||||||
Remote: "https://github.com/foobar/foobar",
|
|
||||||
Subdir: "",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
Version: "master",
|
|
||||||
DepSource: "",
|
|
||||||
}},
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
jf, err := LoadJsonnetfile(NotExist)
|
|
||||||
assert.Equal(t, empty, jf)
|
|
||||||
assert.Error(t, err)
|
|
||||||
}
|
|
||||||
{
|
|
||||||
tempDir, err := ioutil.TempDir("", "jb-load-jsonnetfile")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
defer func() {
|
|
||||||
err := os.RemoveAll(tempDir)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
}()
|
|
||||||
|
|
||||||
tempFile := filepath.Join(tempDir, jsonnetfile.File)
|
|
||||||
err = ioutil.WriteFile(tempFile, []byte(`{}`), os.ModePerm)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
|
|
||||||
jf, err := LoadJsonnetfile(tempFile)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
assert.Equal(t, empty, jf)
|
|
||||||
}
|
|
||||||
{
|
|
||||||
tempDir, err := ioutil.TempDir("", "jb-load-jsonnetfile")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
defer func() {
|
|
||||||
err := os.RemoveAll(tempDir)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
}()
|
|
||||||
|
|
||||||
tempFile := filepath.Join(tempDir, jsonnetfile.File)
|
|
||||||
err = ioutil.WriteFile(tempFile, []byte(jsonnetfileContent), os.ModePerm)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
|
|
||||||
jf, err := LoadJsonnetfile(tempFile)
|
|
||||||
assert.Nil(t, err)
|
|
||||||
assert.Equal(t, jsonnetFileExpected, jf)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
49
spec/spec.go
49
spec/spec.go
|
|
@ -14,14 +14,63 @@
|
||||||
|
|
||||||
package spec
|
package spec
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"sort"
|
||||||
|
)
|
||||||
|
|
||||||
|
// JsonnetFile is the structure of a `.json` file describing a set of jsonnet
|
||||||
|
// dependencies. It is used for both, the jsonnetFile and the lockFile.
|
||||||
type JsonnetFile struct {
|
type JsonnetFile struct {
|
||||||
|
Dependencies map[string]Dependency
|
||||||
|
}
|
||||||
|
|
||||||
|
// New returns a new JsonnetFile with the dependencies map initialized
|
||||||
|
func New() JsonnetFile {
|
||||||
|
return JsonnetFile{
|
||||||
|
Dependencies: make(map[string]Dependency),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// jsonFile is the json representation of a JsonnetFile, which is different for
|
||||||
|
// compatibility reasons.
|
||||||
|
type jsonFile struct {
|
||||||
Dependencies []Dependency `json:"dependencies"`
|
Dependencies []Dependency `json:"dependencies"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UnmarshalJSON unmarshals a `jsonFile`'s json into a JsonnetFile
|
||||||
|
func (jf *JsonnetFile) UnmarshalJSON(data []byte) error {
|
||||||
|
var s jsonFile
|
||||||
|
if err := json.Unmarshal(data, &s); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
jf.Dependencies = make(map[string]Dependency)
|
||||||
|
for _, d := range s.Dependencies {
|
||||||
|
jf.Dependencies[d.Name] = d
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// MarshalJSON serializes a JsonnetFile into json of the format of a `jsonFile`
|
||||||
|
func (jf JsonnetFile) MarshalJSON() ([]byte, error) {
|
||||||
|
var s jsonFile
|
||||||
|
for _, d := range jf.Dependencies {
|
||||||
|
s.Dependencies = append(s.Dependencies, d)
|
||||||
|
}
|
||||||
|
|
||||||
|
sort.SliceStable(s.Dependencies, func(i int, j int) bool {
|
||||||
|
return s.Dependencies[i].Name < s.Dependencies[j].Name
|
||||||
|
})
|
||||||
|
|
||||||
|
return json.Marshal(s)
|
||||||
|
}
|
||||||
|
|
||||||
type Dependency struct {
|
type Dependency struct {
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Source Source `json:"source"`
|
Source Source `json:"source"`
|
||||||
Version string `json:"version"`
|
Version string `json:"version"`
|
||||||
|
Sum string `json:"sum,omitempty"`
|
||||||
DepSource string `json:"-"`
|
DepSource string `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
109
spec/spec_test.go
Normal file
109
spec/spec_test.go
Normal file
|
|
@ -0,0 +1,109 @@
|
||||||
|
// Copyright 2018 jsonnet-bundler authors
|
||||||
|
//
|
||||||
|
// 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 spec
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
const jsonJF = `{
|
||||||
|
"dependencies": [
|
||||||
|
{
|
||||||
|
"name": "grafana-builder",
|
||||||
|
"source": {
|
||||||
|
"git": {
|
||||||
|
"remote": "https://github.com/grafana/jsonnet-libs",
|
||||||
|
"subdir": "grafana-builder"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"version": "54865853ebc1f901964e25a2e7a0e4d2cb6b9648",
|
||||||
|
"sum": "ELsYwK+kGdzX1mee2Yy+/b2mdO4Y503BOCDkFzwmGbE="
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "prometheus-mixin",
|
||||||
|
"source": {
|
||||||
|
"git": {
|
||||||
|
"remote": "https://github.com/prometheus/prometheus",
|
||||||
|
"subdir": "documentation/prometheus-mixin"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"version": "7c039a6b3b4b2a9d7c613ac8bd3fc16e8ca79684",
|
||||||
|
"sum": "bVGOsq3hLOw2irNPAS91a5dZJqQlBUNWy3pVwM4+kIY="
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}`
|
||||||
|
|
||||||
|
func testData() JsonnetFile {
|
||||||
|
return JsonnetFile{
|
||||||
|
Dependencies: map[string]Dependency{
|
||||||
|
"grafana-builder": {
|
||||||
|
Name: "grafana-builder",
|
||||||
|
Source: Source{
|
||||||
|
GitSource: &GitSource{
|
||||||
|
Remote: "https://github.com/grafana/jsonnet-libs",
|
||||||
|
Subdir: "grafana-builder",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Version: "54865853ebc1f901964e25a2e7a0e4d2cb6b9648",
|
||||||
|
Sum: "ELsYwK+kGdzX1mee2Yy+/b2mdO4Y503BOCDkFzwmGbE=",
|
||||||
|
},
|
||||||
|
"prometheus-mixin": {
|
||||||
|
Name: "prometheus-mixin",
|
||||||
|
Source: Source{
|
||||||
|
GitSource: &GitSource{
|
||||||
|
Remote: "https://github.com/prometheus/prometheus",
|
||||||
|
Subdir: "documentation/prometheus-mixin",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Version: "7c039a6b3b4b2a9d7c613ac8bd3fc16e8ca79684",
|
||||||
|
Sum: "bVGOsq3hLOw2irNPAS91a5dZJqQlBUNWy3pVwM4+kIY=",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestUnmarshal checks that unmarshalling works
|
||||||
|
func TestUnmarshal(t *testing.T) {
|
||||||
|
var dst JsonnetFile
|
||||||
|
err := json.Unmarshal([]byte(jsonJF), &dst)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, testData(), dst)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestMarshal checks that marshalling works
|
||||||
|
func TestMarshal(t *testing.T) {
|
||||||
|
data, err := json.Marshal(testData())
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.JSONEq(t, jsonJF, string(data))
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestRemarshal checks that unmarshalling a previously marshalled object yields
|
||||||
|
// the same object
|
||||||
|
func TestRemarshal(t *testing.T) {
|
||||||
|
jf := testData()
|
||||||
|
|
||||||
|
data, err := json.Marshal(jf)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
var dst JsonnetFile
|
||||||
|
err = json.Unmarshal(data, &dst)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
assert.Equal(t, jf, dst)
|
||||||
|
}
|
||||||
28
vendor/github.com/stretchr/testify/require/doc.go
generated
vendored
Normal file
28
vendor/github.com/stretchr/testify/require/doc.go
generated
vendored
Normal file
|
|
@ -0,0 +1,28 @@
|
||||||
|
// Package require implements the same assertions as the `assert` package but
|
||||||
|
// stops test execution when a test fails.
|
||||||
|
//
|
||||||
|
// Example Usage
|
||||||
|
//
|
||||||
|
// The following is a complete example using require in a standard test function:
|
||||||
|
// import (
|
||||||
|
// "testing"
|
||||||
|
// "github.com/stretchr/testify/require"
|
||||||
|
// )
|
||||||
|
//
|
||||||
|
// func TestSomething(t *testing.T) {
|
||||||
|
//
|
||||||
|
// var a string = "Hello"
|
||||||
|
// var b string = "Hello"
|
||||||
|
//
|
||||||
|
// require.Equal(t, a, b, "The two words should be the same.")
|
||||||
|
//
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// Assertions
|
||||||
|
//
|
||||||
|
// The `require` package have same global functions as in the `assert` package,
|
||||||
|
// but instead of returning a boolean result they call `t.FailNow()`.
|
||||||
|
//
|
||||||
|
// Every assertion function also takes an optional string message as the final argument,
|
||||||
|
// allowing custom error messages to be appended to the message the assertion method outputs.
|
||||||
|
package require
|
||||||
16
vendor/github.com/stretchr/testify/require/forward_requirements.go
generated
vendored
Normal file
16
vendor/github.com/stretchr/testify/require/forward_requirements.go
generated
vendored
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
package require
|
||||||
|
|
||||||
|
// Assertions provides assertion methods around the
|
||||||
|
// TestingT interface.
|
||||||
|
type Assertions struct {
|
||||||
|
t TestingT
|
||||||
|
}
|
||||||
|
|
||||||
|
// New makes a new Assertions object for the specified TestingT.
|
||||||
|
func New(t TestingT) *Assertions {
|
||||||
|
return &Assertions{
|
||||||
|
t: t,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//go:generate go run ../_codegen/main.go -output-package=require -template=require_forward.go.tmpl -include-format-funcs
|
||||||
1227
vendor/github.com/stretchr/testify/require/require.go
generated
vendored
Normal file
1227
vendor/github.com/stretchr/testify/require/require.go
generated
vendored
Normal file
File diff suppressed because it is too large
Load diff
6
vendor/github.com/stretchr/testify/require/require.go.tmpl
generated
vendored
Normal file
6
vendor/github.com/stretchr/testify/require/require.go.tmpl
generated
vendored
Normal file
|
|
@ -0,0 +1,6 @@
|
||||||
|
{{.Comment}}
|
||||||
|
func {{.DocInfo.Name}}(t TestingT, {{.Params}}) {
|
||||||
|
if assert.{{.DocInfo.Name}}(t, {{.ForwardedParams}}) { return }
|
||||||
|
if h, ok := t.(tHelper); ok { h.Helper() }
|
||||||
|
t.FailNow()
|
||||||
|
}
|
||||||
957
vendor/github.com/stretchr/testify/require/require_forward.go
generated
vendored
Normal file
957
vendor/github.com/stretchr/testify/require/require_forward.go
generated
vendored
Normal file
|
|
@ -0,0 +1,957 @@
|
||||||
|
/*
|
||||||
|
* CODE GENERATED AUTOMATICALLY WITH github.com/stretchr/testify/_codegen
|
||||||
|
* THIS FILE MUST NOT BE EDITED BY HAND
|
||||||
|
*/
|
||||||
|
|
||||||
|
package require
|
||||||
|
|
||||||
|
import (
|
||||||
|
assert "github.com/stretchr/testify/assert"
|
||||||
|
http "net/http"
|
||||||
|
url "net/url"
|
||||||
|
time "time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Condition uses a Comparison to assert a complex condition.
|
||||||
|
func (a *Assertions) Condition(comp assert.Comparison, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Condition(a.t, comp, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Conditionf uses a Comparison to assert a complex condition.
|
||||||
|
func (a *Assertions) Conditionf(comp assert.Comparison, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Conditionf(a.t, comp, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Contains asserts that the specified string, list(array, slice...) or map contains the
|
||||||
|
// specified substring or element.
|
||||||
|
//
|
||||||
|
// a.Contains("Hello World", "World")
|
||||||
|
// a.Contains(["Hello", "World"], "World")
|
||||||
|
// a.Contains({"Hello": "World"}, "Hello")
|
||||||
|
func (a *Assertions) Contains(s interface{}, contains interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Contains(a.t, s, contains, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Containsf asserts that the specified string, list(array, slice...) or map contains the
|
||||||
|
// specified substring or element.
|
||||||
|
//
|
||||||
|
// a.Containsf("Hello World", "World", "error message %s", "formatted")
|
||||||
|
// a.Containsf(["Hello", "World"], "World", "error message %s", "formatted")
|
||||||
|
// a.Containsf({"Hello": "World"}, "Hello", "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Containsf(s interface{}, contains interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Containsf(a.t, s, contains, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DirExists checks whether a directory exists in the given path. It also fails if the path is a file rather a directory or there is an error checking whether it exists.
|
||||||
|
func (a *Assertions) DirExists(path string, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
DirExists(a.t, path, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DirExistsf checks whether a directory exists in the given path. It also fails if the path is a file rather a directory or there is an error checking whether it exists.
|
||||||
|
func (a *Assertions) DirExistsf(path string, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
DirExistsf(a.t, path, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ElementsMatch asserts that the specified listA(array, slice...) is equal to specified
|
||||||
|
// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements,
|
||||||
|
// the number of appearances of each of them in both lists should match.
|
||||||
|
//
|
||||||
|
// a.ElementsMatch([1, 3, 2, 3], [1, 3, 3, 2])
|
||||||
|
func (a *Assertions) ElementsMatch(listA interface{}, listB interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
ElementsMatch(a.t, listA, listB, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ElementsMatchf asserts that the specified listA(array, slice...) is equal to specified
|
||||||
|
// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements,
|
||||||
|
// the number of appearances of each of them in both lists should match.
|
||||||
|
//
|
||||||
|
// a.ElementsMatchf([1, 3, 2, 3], [1, 3, 3, 2], "error message %s", "formatted")
|
||||||
|
func (a *Assertions) ElementsMatchf(listA interface{}, listB interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
ElementsMatchf(a.t, listA, listB, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Empty asserts that the specified object is empty. I.e. nil, "", false, 0 or either
|
||||||
|
// a slice or a channel with len == 0.
|
||||||
|
//
|
||||||
|
// a.Empty(obj)
|
||||||
|
func (a *Assertions) Empty(object interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Empty(a.t, object, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Emptyf asserts that the specified object is empty. I.e. nil, "", false, 0 or either
|
||||||
|
// a slice or a channel with len == 0.
|
||||||
|
//
|
||||||
|
// a.Emptyf(obj, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Emptyf(object interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Emptyf(a.t, object, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equal asserts that two objects are equal.
|
||||||
|
//
|
||||||
|
// a.Equal(123, 123)
|
||||||
|
//
|
||||||
|
// Pointer variable equality is determined based on the equality of the
|
||||||
|
// referenced values (as opposed to the memory addresses). Function equality
|
||||||
|
// cannot be determined and will always fail.
|
||||||
|
func (a *Assertions) Equal(expected interface{}, actual interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Equal(a.t, expected, actual, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EqualError asserts that a function returned an error (i.e. not `nil`)
|
||||||
|
// and that it is equal to the provided error.
|
||||||
|
//
|
||||||
|
// actualObj, err := SomeFunction()
|
||||||
|
// a.EqualError(err, expectedErrorString)
|
||||||
|
func (a *Assertions) EqualError(theError error, errString string, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
EqualError(a.t, theError, errString, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EqualErrorf asserts that a function returned an error (i.e. not `nil`)
|
||||||
|
// and that it is equal to the provided error.
|
||||||
|
//
|
||||||
|
// actualObj, err := SomeFunction()
|
||||||
|
// a.EqualErrorf(err, expectedErrorString, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) EqualErrorf(theError error, errString string, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
EqualErrorf(a.t, theError, errString, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EqualValues asserts that two objects are equal or convertable to the same types
|
||||||
|
// and equal.
|
||||||
|
//
|
||||||
|
// a.EqualValues(uint32(123), int32(123))
|
||||||
|
func (a *Assertions) EqualValues(expected interface{}, actual interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
EqualValues(a.t, expected, actual, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EqualValuesf asserts that two objects are equal or convertable to the same types
|
||||||
|
// and equal.
|
||||||
|
//
|
||||||
|
// a.EqualValuesf(uint32(123, "error message %s", "formatted"), int32(123))
|
||||||
|
func (a *Assertions) EqualValuesf(expected interface{}, actual interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
EqualValuesf(a.t, expected, actual, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equalf asserts that two objects are equal.
|
||||||
|
//
|
||||||
|
// a.Equalf(123, 123, "error message %s", "formatted")
|
||||||
|
//
|
||||||
|
// Pointer variable equality is determined based on the equality of the
|
||||||
|
// referenced values (as opposed to the memory addresses). Function equality
|
||||||
|
// cannot be determined and will always fail.
|
||||||
|
func (a *Assertions) Equalf(expected interface{}, actual interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Equalf(a.t, expected, actual, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error asserts that a function returned an error (i.e. not `nil`).
|
||||||
|
//
|
||||||
|
// actualObj, err := SomeFunction()
|
||||||
|
// if a.Error(err) {
|
||||||
|
// assert.Equal(t, expectedError, err)
|
||||||
|
// }
|
||||||
|
func (a *Assertions) Error(err error, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Error(a.t, err, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Errorf asserts that a function returned an error (i.e. not `nil`).
|
||||||
|
//
|
||||||
|
// actualObj, err := SomeFunction()
|
||||||
|
// if a.Errorf(err, "error message %s", "formatted") {
|
||||||
|
// assert.Equal(t, expectedErrorf, err)
|
||||||
|
// }
|
||||||
|
func (a *Assertions) Errorf(err error, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Errorf(a.t, err, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exactly asserts that two objects are equal in value and type.
|
||||||
|
//
|
||||||
|
// a.Exactly(int32(123), int64(123))
|
||||||
|
func (a *Assertions) Exactly(expected interface{}, actual interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Exactly(a.t, expected, actual, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exactlyf asserts that two objects are equal in value and type.
|
||||||
|
//
|
||||||
|
// a.Exactlyf(int32(123, "error message %s", "formatted"), int64(123))
|
||||||
|
func (a *Assertions) Exactlyf(expected interface{}, actual interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Exactlyf(a.t, expected, actual, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fail reports a failure through
|
||||||
|
func (a *Assertions) Fail(failureMessage string, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Fail(a.t, failureMessage, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// FailNow fails test
|
||||||
|
func (a *Assertions) FailNow(failureMessage string, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
FailNow(a.t, failureMessage, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// FailNowf fails test
|
||||||
|
func (a *Assertions) FailNowf(failureMessage string, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
FailNowf(a.t, failureMessage, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Failf reports a failure through
|
||||||
|
func (a *Assertions) Failf(failureMessage string, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Failf(a.t, failureMessage, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// False asserts that the specified value is false.
|
||||||
|
//
|
||||||
|
// a.False(myBool)
|
||||||
|
func (a *Assertions) False(value bool, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
False(a.t, value, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Falsef asserts that the specified value is false.
|
||||||
|
//
|
||||||
|
// a.Falsef(myBool, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Falsef(value bool, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Falsef(a.t, value, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// FileExists checks whether a file exists in the given path. It also fails if the path points to a directory or there is an error when trying to check the file.
|
||||||
|
func (a *Assertions) FileExists(path string, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
FileExists(a.t, path, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// FileExistsf checks whether a file exists in the given path. It also fails if the path points to a directory or there is an error when trying to check the file.
|
||||||
|
func (a *Assertions) FileExistsf(path string, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
FileExistsf(a.t, path, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPBodyContains asserts that a specified handler returns a
|
||||||
|
// body that contains a string.
|
||||||
|
//
|
||||||
|
// a.HTTPBodyContains(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky")
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPBodyContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPBodyContains(a.t, handler, method, url, values, str, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPBodyContainsf asserts that a specified handler returns a
|
||||||
|
// body that contains a string.
|
||||||
|
//
|
||||||
|
// a.HTTPBodyContainsf(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted")
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPBodyContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPBodyContainsf(a.t, handler, method, url, values, str, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPBodyNotContains asserts that a specified handler returns a
|
||||||
|
// body that does not contain a string.
|
||||||
|
//
|
||||||
|
// a.HTTPBodyNotContains(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky")
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPBodyNotContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPBodyNotContains(a.t, handler, method, url, values, str, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPBodyNotContainsf asserts that a specified handler returns a
|
||||||
|
// body that does not contain a string.
|
||||||
|
//
|
||||||
|
// a.HTTPBodyNotContainsf(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted")
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPBodyNotContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPBodyNotContainsf(a.t, handler, method, url, values, str, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPError asserts that a specified handler returns an error status code.
|
||||||
|
//
|
||||||
|
// a.HTTPError(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}}
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPError(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPError(a.t, handler, method, url, values, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPErrorf asserts that a specified handler returns an error status code.
|
||||||
|
//
|
||||||
|
// a.HTTPErrorf(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}}
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true, "error message %s", "formatted") or not (false).
|
||||||
|
func (a *Assertions) HTTPErrorf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPErrorf(a.t, handler, method, url, values, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPRedirect asserts that a specified handler returns a redirect status code.
|
||||||
|
//
|
||||||
|
// a.HTTPRedirect(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}}
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPRedirect(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPRedirect(a.t, handler, method, url, values, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPRedirectf asserts that a specified handler returns a redirect status code.
|
||||||
|
//
|
||||||
|
// a.HTTPRedirectf(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}}
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true, "error message %s", "formatted") or not (false).
|
||||||
|
func (a *Assertions) HTTPRedirectf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPRedirectf(a.t, handler, method, url, values, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPSuccess asserts that a specified handler returns a success status code.
|
||||||
|
//
|
||||||
|
// a.HTTPSuccess(myHandler, "POST", "http://www.google.com", nil)
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPSuccess(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPSuccess(a.t, handler, method, url, values, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTPSuccessf asserts that a specified handler returns a success status code.
|
||||||
|
//
|
||||||
|
// a.HTTPSuccessf(myHandler, "POST", "http://www.google.com", nil, "error message %s", "formatted")
|
||||||
|
//
|
||||||
|
// Returns whether the assertion was successful (true) or not (false).
|
||||||
|
func (a *Assertions) HTTPSuccessf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
HTTPSuccessf(a.t, handler, method, url, values, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Implements asserts that an object is implemented by the specified interface.
|
||||||
|
//
|
||||||
|
// a.Implements((*MyInterface)(nil), new(MyObject))
|
||||||
|
func (a *Assertions) Implements(interfaceObject interface{}, object interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Implements(a.t, interfaceObject, object, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Implementsf asserts that an object is implemented by the specified interface.
|
||||||
|
//
|
||||||
|
// a.Implementsf((*MyInterface, "error message %s", "formatted")(nil), new(MyObject))
|
||||||
|
func (a *Assertions) Implementsf(interfaceObject interface{}, object interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Implementsf(a.t, interfaceObject, object, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InDelta asserts that the two numerals are within delta of each other.
|
||||||
|
//
|
||||||
|
// a.InDelta(math.Pi, (22 / 7.0), 0.01)
|
||||||
|
func (a *Assertions) InDelta(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InDelta(a.t, expected, actual, delta, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InDeltaMapValues is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys.
|
||||||
|
func (a *Assertions) InDeltaMapValues(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InDeltaMapValues(a.t, expected, actual, delta, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InDeltaMapValuesf is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys.
|
||||||
|
func (a *Assertions) InDeltaMapValuesf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InDeltaMapValuesf(a.t, expected, actual, delta, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InDeltaSlice is the same as InDelta, except it compares two slices.
|
||||||
|
func (a *Assertions) InDeltaSlice(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InDeltaSlice(a.t, expected, actual, delta, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InDeltaSlicef is the same as InDelta, except it compares two slices.
|
||||||
|
func (a *Assertions) InDeltaSlicef(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InDeltaSlicef(a.t, expected, actual, delta, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InDeltaf asserts that the two numerals are within delta of each other.
|
||||||
|
//
|
||||||
|
// a.InDeltaf(math.Pi, (22 / 7.0, "error message %s", "formatted"), 0.01)
|
||||||
|
func (a *Assertions) InDeltaf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InDeltaf(a.t, expected, actual, delta, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InEpsilon asserts that expected and actual have a relative error less than epsilon
|
||||||
|
func (a *Assertions) InEpsilon(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InEpsilon(a.t, expected, actual, epsilon, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InEpsilonSlice is the same as InEpsilon, except it compares each value from two slices.
|
||||||
|
func (a *Assertions) InEpsilonSlice(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InEpsilonSlice(a.t, expected, actual, epsilon, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InEpsilonSlicef is the same as InEpsilon, except it compares each value from two slices.
|
||||||
|
func (a *Assertions) InEpsilonSlicef(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InEpsilonSlicef(a.t, expected, actual, epsilon, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InEpsilonf asserts that expected and actual have a relative error less than epsilon
|
||||||
|
func (a *Assertions) InEpsilonf(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
InEpsilonf(a.t, expected, actual, epsilon, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsType asserts that the specified objects are of the same type.
|
||||||
|
func (a *Assertions) IsType(expectedType interface{}, object interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
IsType(a.t, expectedType, object, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsTypef asserts that the specified objects are of the same type.
|
||||||
|
func (a *Assertions) IsTypef(expectedType interface{}, object interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
IsTypef(a.t, expectedType, object, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// JSONEq asserts that two JSON strings are equivalent.
|
||||||
|
//
|
||||||
|
// a.JSONEq(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`)
|
||||||
|
func (a *Assertions) JSONEq(expected string, actual string, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
JSONEq(a.t, expected, actual, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// JSONEqf asserts that two JSON strings are equivalent.
|
||||||
|
//
|
||||||
|
// a.JSONEqf(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) JSONEqf(expected string, actual string, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
JSONEqf(a.t, expected, actual, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Len asserts that the specified object has specific length.
|
||||||
|
// Len also fails if the object has a type that len() not accept.
|
||||||
|
//
|
||||||
|
// a.Len(mySlice, 3)
|
||||||
|
func (a *Assertions) Len(object interface{}, length int, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Len(a.t, object, length, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Lenf asserts that the specified object has specific length.
|
||||||
|
// Lenf also fails if the object has a type that len() not accept.
|
||||||
|
//
|
||||||
|
// a.Lenf(mySlice, 3, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Lenf(object interface{}, length int, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Lenf(a.t, object, length, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Nil asserts that the specified object is nil.
|
||||||
|
//
|
||||||
|
// a.Nil(err)
|
||||||
|
func (a *Assertions) Nil(object interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Nil(a.t, object, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Nilf asserts that the specified object is nil.
|
||||||
|
//
|
||||||
|
// a.Nilf(err, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Nilf(object interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Nilf(a.t, object, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NoError asserts that a function returned no error (i.e. `nil`).
|
||||||
|
//
|
||||||
|
// actualObj, err := SomeFunction()
|
||||||
|
// if a.NoError(err) {
|
||||||
|
// assert.Equal(t, expectedObj, actualObj)
|
||||||
|
// }
|
||||||
|
func (a *Assertions) NoError(err error, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NoError(a.t, err, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NoErrorf asserts that a function returned no error (i.e. `nil`).
|
||||||
|
//
|
||||||
|
// actualObj, err := SomeFunction()
|
||||||
|
// if a.NoErrorf(err, "error message %s", "formatted") {
|
||||||
|
// assert.Equal(t, expectedObj, actualObj)
|
||||||
|
// }
|
||||||
|
func (a *Assertions) NoErrorf(err error, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NoErrorf(a.t, err, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotContains asserts that the specified string, list(array, slice...) or map does NOT contain the
|
||||||
|
// specified substring or element.
|
||||||
|
//
|
||||||
|
// a.NotContains("Hello World", "Earth")
|
||||||
|
// a.NotContains(["Hello", "World"], "Earth")
|
||||||
|
// a.NotContains({"Hello": "World"}, "Earth")
|
||||||
|
func (a *Assertions) NotContains(s interface{}, contains interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotContains(a.t, s, contains, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotContainsf asserts that the specified string, list(array, slice...) or map does NOT contain the
|
||||||
|
// specified substring or element.
|
||||||
|
//
|
||||||
|
// a.NotContainsf("Hello World", "Earth", "error message %s", "formatted")
|
||||||
|
// a.NotContainsf(["Hello", "World"], "Earth", "error message %s", "formatted")
|
||||||
|
// a.NotContainsf({"Hello": "World"}, "Earth", "error message %s", "formatted")
|
||||||
|
func (a *Assertions) NotContainsf(s interface{}, contains interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotContainsf(a.t, s, contains, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotEmpty asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either
|
||||||
|
// a slice or a channel with len == 0.
|
||||||
|
//
|
||||||
|
// if a.NotEmpty(obj) {
|
||||||
|
// assert.Equal(t, "two", obj[1])
|
||||||
|
// }
|
||||||
|
func (a *Assertions) NotEmpty(object interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotEmpty(a.t, object, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotEmptyf asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either
|
||||||
|
// a slice or a channel with len == 0.
|
||||||
|
//
|
||||||
|
// if a.NotEmptyf(obj, "error message %s", "formatted") {
|
||||||
|
// assert.Equal(t, "two", obj[1])
|
||||||
|
// }
|
||||||
|
func (a *Assertions) NotEmptyf(object interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotEmptyf(a.t, object, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotEqual asserts that the specified values are NOT equal.
|
||||||
|
//
|
||||||
|
// a.NotEqual(obj1, obj2)
|
||||||
|
//
|
||||||
|
// Pointer variable equality is determined based on the equality of the
|
||||||
|
// referenced values (as opposed to the memory addresses).
|
||||||
|
func (a *Assertions) NotEqual(expected interface{}, actual interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotEqual(a.t, expected, actual, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotEqualf asserts that the specified values are NOT equal.
|
||||||
|
//
|
||||||
|
// a.NotEqualf(obj1, obj2, "error message %s", "formatted")
|
||||||
|
//
|
||||||
|
// Pointer variable equality is determined based on the equality of the
|
||||||
|
// referenced values (as opposed to the memory addresses).
|
||||||
|
func (a *Assertions) NotEqualf(expected interface{}, actual interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotEqualf(a.t, expected, actual, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotNil asserts that the specified object is not nil.
|
||||||
|
//
|
||||||
|
// a.NotNil(err)
|
||||||
|
func (a *Assertions) NotNil(object interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotNil(a.t, object, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotNilf asserts that the specified object is not nil.
|
||||||
|
//
|
||||||
|
// a.NotNilf(err, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) NotNilf(object interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotNilf(a.t, object, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotPanics asserts that the code inside the specified PanicTestFunc does NOT panic.
|
||||||
|
//
|
||||||
|
// a.NotPanics(func(){ RemainCalm() })
|
||||||
|
func (a *Assertions) NotPanics(f assert.PanicTestFunc, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotPanics(a.t, f, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotPanicsf asserts that the code inside the specified PanicTestFunc does NOT panic.
|
||||||
|
//
|
||||||
|
// a.NotPanicsf(func(){ RemainCalm() }, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) NotPanicsf(f assert.PanicTestFunc, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotPanicsf(a.t, f, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotRegexp asserts that a specified regexp does not match a string.
|
||||||
|
//
|
||||||
|
// a.NotRegexp(regexp.MustCompile("starts"), "it's starting")
|
||||||
|
// a.NotRegexp("^start", "it's not starting")
|
||||||
|
func (a *Assertions) NotRegexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotRegexp(a.t, rx, str, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotRegexpf asserts that a specified regexp does not match a string.
|
||||||
|
//
|
||||||
|
// a.NotRegexpf(regexp.MustCompile("starts", "error message %s", "formatted"), "it's starting")
|
||||||
|
// a.NotRegexpf("^start", "it's not starting", "error message %s", "formatted")
|
||||||
|
func (a *Assertions) NotRegexpf(rx interface{}, str interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotRegexpf(a.t, rx, str, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotSubset asserts that the specified list(array, slice...) contains not all
|
||||||
|
// elements given in the specified subset(array, slice...).
|
||||||
|
//
|
||||||
|
// a.NotSubset([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]")
|
||||||
|
func (a *Assertions) NotSubset(list interface{}, subset interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotSubset(a.t, list, subset, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotSubsetf asserts that the specified list(array, slice...) contains not all
|
||||||
|
// elements given in the specified subset(array, slice...).
|
||||||
|
//
|
||||||
|
// a.NotSubsetf([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]", "error message %s", "formatted")
|
||||||
|
func (a *Assertions) NotSubsetf(list interface{}, subset interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotSubsetf(a.t, list, subset, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotZero asserts that i is not the zero value for its type.
|
||||||
|
func (a *Assertions) NotZero(i interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotZero(a.t, i, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotZerof asserts that i is not the zero value for its type.
|
||||||
|
func (a *Assertions) NotZerof(i interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
NotZerof(a.t, i, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Panics asserts that the code inside the specified PanicTestFunc panics.
|
||||||
|
//
|
||||||
|
// a.Panics(func(){ GoCrazy() })
|
||||||
|
func (a *Assertions) Panics(f assert.PanicTestFunc, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Panics(a.t, f, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PanicsWithValue asserts that the code inside the specified PanicTestFunc panics, and that
|
||||||
|
// the recovered panic value equals the expected panic value.
|
||||||
|
//
|
||||||
|
// a.PanicsWithValue("crazy error", func(){ GoCrazy() })
|
||||||
|
func (a *Assertions) PanicsWithValue(expected interface{}, f assert.PanicTestFunc, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
PanicsWithValue(a.t, expected, f, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PanicsWithValuef asserts that the code inside the specified PanicTestFunc panics, and that
|
||||||
|
// the recovered panic value equals the expected panic value.
|
||||||
|
//
|
||||||
|
// a.PanicsWithValuef("crazy error", func(){ GoCrazy() }, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) PanicsWithValuef(expected interface{}, f assert.PanicTestFunc, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
PanicsWithValuef(a.t, expected, f, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Panicsf asserts that the code inside the specified PanicTestFunc panics.
|
||||||
|
//
|
||||||
|
// a.Panicsf(func(){ GoCrazy() }, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Panicsf(f assert.PanicTestFunc, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Panicsf(a.t, f, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Regexp asserts that a specified regexp matches a string.
|
||||||
|
//
|
||||||
|
// a.Regexp(regexp.MustCompile("start"), "it's starting")
|
||||||
|
// a.Regexp("start...$", "it's not starting")
|
||||||
|
func (a *Assertions) Regexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Regexp(a.t, rx, str, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Regexpf asserts that a specified regexp matches a string.
|
||||||
|
//
|
||||||
|
// a.Regexpf(regexp.MustCompile("start", "error message %s", "formatted"), "it's starting")
|
||||||
|
// a.Regexpf("start...$", "it's not starting", "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Regexpf(rx interface{}, str interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Regexpf(a.t, rx, str, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Subset asserts that the specified list(array, slice...) contains all
|
||||||
|
// elements given in the specified subset(array, slice...).
|
||||||
|
//
|
||||||
|
// a.Subset([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]")
|
||||||
|
func (a *Assertions) Subset(list interface{}, subset interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Subset(a.t, list, subset, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Subsetf asserts that the specified list(array, slice...) contains all
|
||||||
|
// elements given in the specified subset(array, slice...).
|
||||||
|
//
|
||||||
|
// a.Subsetf([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]", "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Subsetf(list interface{}, subset interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Subsetf(a.t, list, subset, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// True asserts that the specified value is true.
|
||||||
|
//
|
||||||
|
// a.True(myBool)
|
||||||
|
func (a *Assertions) True(value bool, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
True(a.t, value, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Truef asserts that the specified value is true.
|
||||||
|
//
|
||||||
|
// a.Truef(myBool, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) Truef(value bool, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Truef(a.t, value, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithinDuration asserts that the two times are within duration delta of each other.
|
||||||
|
//
|
||||||
|
// a.WithinDuration(time.Now(), time.Now(), 10*time.Second)
|
||||||
|
func (a *Assertions) WithinDuration(expected time.Time, actual time.Time, delta time.Duration, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
WithinDuration(a.t, expected, actual, delta, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithinDurationf asserts that the two times are within duration delta of each other.
|
||||||
|
//
|
||||||
|
// a.WithinDurationf(time.Now(), time.Now(), 10*time.Second, "error message %s", "formatted")
|
||||||
|
func (a *Assertions) WithinDurationf(expected time.Time, actual time.Time, delta time.Duration, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
WithinDurationf(a.t, expected, actual, delta, msg, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Zero asserts that i is the zero value for its type.
|
||||||
|
func (a *Assertions) Zero(i interface{}, msgAndArgs ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Zero(a.t, i, msgAndArgs...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Zerof asserts that i is the zero value for its type.
|
||||||
|
func (a *Assertions) Zerof(i interface{}, msg string, args ...interface{}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok {
|
||||||
|
h.Helper()
|
||||||
|
}
|
||||||
|
Zerof(a.t, i, msg, args...)
|
||||||
|
}
|
||||||
5
vendor/github.com/stretchr/testify/require/require_forward.go.tmpl
generated
vendored
Normal file
5
vendor/github.com/stretchr/testify/require/require_forward.go.tmpl
generated
vendored
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
{{.CommentWithoutT "a"}}
|
||||||
|
func (a *Assertions) {{.DocInfo.Name}}({{.Params}}) {
|
||||||
|
if h, ok := a.t.(tHelper); ok { h.Helper() }
|
||||||
|
{{.DocInfo.Name}}(a.t, {{.ForwardedParams}})
|
||||||
|
}
|
||||||
29
vendor/github.com/stretchr/testify/require/requirements.go
generated
vendored
Normal file
29
vendor/github.com/stretchr/testify/require/requirements.go
generated
vendored
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
package require
|
||||||
|
|
||||||
|
// TestingT is an interface wrapper around *testing.T
|
||||||
|
type TestingT interface {
|
||||||
|
Errorf(format string, args ...interface{})
|
||||||
|
FailNow()
|
||||||
|
}
|
||||||
|
|
||||||
|
type tHelper interface {
|
||||||
|
Helper()
|
||||||
|
}
|
||||||
|
|
||||||
|
// ComparisonAssertionFunc is a common function prototype when comparing two values. Can be useful
|
||||||
|
// for table driven tests.
|
||||||
|
type ComparisonAssertionFunc func(TestingT, interface{}, interface{}, ...interface{})
|
||||||
|
|
||||||
|
// ValueAssertionFunc is a common function prototype when validating a single value. Can be useful
|
||||||
|
// for table driven tests.
|
||||||
|
type ValueAssertionFunc func(TestingT, interface{}, ...interface{})
|
||||||
|
|
||||||
|
// BoolAssertionFunc is a common function prototype when validating a bool value. Can be useful
|
||||||
|
// for table driven tests.
|
||||||
|
type BoolAssertionFunc func(TestingT, bool, ...interface{})
|
||||||
|
|
||||||
|
// ErrorAssertionFunc is a common function prototype when validating an error value. Can be useful
|
||||||
|
// for table driven tests.
|
||||||
|
type ErrorAssertionFunc func(TestingT, error, ...interface{})
|
||||||
|
|
||||||
|
//go:generate go run ../_codegen/main.go -output-package=require -template=require.go.tmpl -include-format-funcs
|
||||||
1
vendor/modules.txt
vendored
1
vendor/modules.txt
vendored
|
|
@ -17,6 +17,7 @@ github.com/pkg/errors
|
||||||
github.com/pmezard/go-difflib/difflib
|
github.com/pmezard/go-difflib/difflib
|
||||||
# github.com/stretchr/testify v1.3.0
|
# github.com/stretchr/testify v1.3.0
|
||||||
github.com/stretchr/testify/assert
|
github.com/stretchr/testify/assert
|
||||||
|
github.com/stretchr/testify/require
|
||||||
# golang.org/x/sys v0.0.0-20190310054646-10058d7d4faa
|
# golang.org/x/sys v0.0.0-20190310054646-10058d7d4faa
|
||||||
golang.org/x/sys/unix
|
golang.org/x/sys/unix
|
||||||
# gopkg.in/alecthomas/kingpin.v2 v2.2.6
|
# gopkg.in/alecthomas/kingpin.v2 v2.2.6
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue