blob: 9d6c14e013da08f222856145ec92bf8fd2610cdf [file] [log] [blame]
// Copyright 2019 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package frontend
import (
"context"
"testing"
"github.com/google/go-cmp/cmp"
"golang.org/x/pkgsite/internal"
"golang.org/x/pkgsite/internal/postgres"
"golang.org/x/pkgsite/internal/stdlib"
"golang.org/x/pkgsite/internal/testing/sample"
"golang.org/x/pkgsite/internal/version"
)
var (
modulePath1 = "test.com/module"
modulePath2 = "test.com/module/v2"
commitTime = "0 hours ago"
)
func sampleModule(modulePath, version string, versionType version.Type, packages ...*internal.Package) *internal.Module {
m := sample.Module()
m.ModulePath = modulePath
m.Version = version
m.VersionType = versionType
if len(packages) > 0 {
m.Packages = packages
}
return m
}
func versionSummaries(path string, versions []string, linkify func(path, version string) string) []*VersionSummary {
vs := make([]*VersionSummary, len(versions))
for i, version := range versions {
var semver, displayVersion string
if stdlib.Contains(path) {
semver = version
displayVersion = version
} else {
semver = version
displayVersion = formatVersion(semver)
}
vs[i] = &VersionSummary{
TooltipVersion: semver,
DisplayVersion: displayVersion,
Link: linkify(path, version),
CommitTime: commitTime,
}
}
return vs
}
func TestFetchModuleVersionDetails(t *testing.T) {
ctx, cancel := context.WithTimeout(context.Background(), testTimeout)
defer cancel()
info1 := sample.ModuleInfo()
info1.ModulePath = modulePath1
info1.Version = "v1.2.1"
info2 := sample.ModuleInfo()
info2.ModulePath = modulePath2
info2.Version = "v2.2.1-alpha.1"
makeList := func(path, major string, versions []string) *VersionList {
return &VersionList{
VersionListKey: VersionListKey{ModulePath: path, Major: major},
Versions: versionSummaries(path, versions, func(path, version string) string {
return constructModuleURL(path, version)
}),
}
}
for _, tc := range []struct {
name string
info *internal.ModuleInfo
modules []*internal.Module
wantDetails *VersionsDetails
}{
{
name: "want v1 first",
info: info1,
modules: []*internal.Module{
sampleModule(modulePath1, "v0.0.0-20140414041502-3c2ca4d52544", version.TypePseudo),
sampleModule(modulePath1, "v1.2.3", version.TypeRelease),
sampleModule(modulePath2, "v2.0.0", version.TypeRelease),
sampleModule(modulePath1, "v1.3.0", version.TypeRelease),
sampleModule(modulePath1, "v1.2.1", version.TypeRelease),
sampleModule(modulePath2, "v2.2.1-alpha.1", version.TypePrerelease),
},
wantDetails: &VersionsDetails{
ThisModule: []*VersionList{
makeList("test.com/module", "v1", []string{"v1.3.0", "v1.2.3", "v1.2.1"}),
},
OtherModules: []*VersionList{
makeList("test.com/module/v2", "v2", []string{"v2.2.1-alpha.1", "v2.0.0"}),
},
},
},
{
name: "want v2 first",
info: info2,
modules: []*internal.Module{
sampleModule(modulePath1, "v0.0.0-20140414041502-3c2ca4d52544", version.TypePseudo),
sampleModule(modulePath1, "v1.2.1", version.TypeRelease),
sampleModule(modulePath1, "v1.2.3", version.TypeRelease),
sampleModule(modulePath1, "v2.1.0+incompatible", version.TypeRelease),
sampleModule(modulePath2, "v2.0.0", version.TypeRelease),
sampleModule(modulePath2, "v2.2.1-alpha.1", version.TypePrerelease),
},
wantDetails: &VersionsDetails{
ThisModule: []*VersionList{
makeList("test.com/module/v2", "v2", []string{"v2.2.1-alpha.1", "v2.0.0"}),
},
OtherModules: []*VersionList{
makeList("test.com/module", "v1", []string{"v2.1.0+incompatible", "v1.2.3", "v1.2.1"}),
},
},
},
{
name: "want only pseudo",
info: info2,
modules: []*internal.Module{
sampleModule(modulePath1, "v0.0.0-20140414041501-3c2ca4d52544", version.TypePseudo),
sampleModule(modulePath1, "v0.0.0-20140414041502-4c2ca4d52544", version.TypePseudo),
},
wantDetails: &VersionsDetails{
OtherModules: []*VersionList{
makeList("test.com/module", "v0", []string{
"v0.0.0-20140414041502-4c2ca4d52544",
"v0.0.0-20140414041501-3c2ca4d52544"},
),
},
},
},
} {
t.Run(tc.name, func(t *testing.T) {
defer postgres.ResetTestDB(testDB, t)
for _, v := range tc.modules {
if err := testDB.InsertModule(ctx, v); err != nil {
t.Fatal(err)
}
}
got, err := fetchModuleVersionsDetails(ctx, testDB, tc.info)
if err != nil {
t.Fatalf("fetchModuleVersionsDetails(ctx, db, %v): %v", tc.info, err)
}
if diff := cmp.Diff(tc.wantDetails, got); diff != "" {
t.Errorf("fetchModuleVersionsDetails(ctx, db, %v) mismatch (-want +got):\n%s", tc.info, diff)
}
})
}
}
func TestFetchPackageVersionsDetails(t *testing.T) {
ctx, cancel := context.WithTimeout(context.Background(), testTimeout)
defer cancel()
var (
v2Path = "test.com/module/v2/foo"
v1Path = "test.com/module/foo"
)
pkg1 := sample.VersionedPackage()
pkg1.Path = v1Path
pkg1.V1Path = v1Path
pkg1.ModulePath = modulePath1
pkg1.Version = "v1.2.1"
pkg2 := sample.VersionedPackage()
pkg2.Path = v2Path
pkg2.V1Path = v1Path
pkg2.ModulePath = modulePath2
pkg2.Version = "v2.2.1-alpha.1"
nethttpPkg := sample.VersionedPackage()
nethttpPkg.Path = "net/http"
nethttpPkg.V1Path = "net/http"
nethttpPkg.ModulePath = "std"
nethttpPkg.Version = "v1.12.5"
makeList := func(pkgPath, modulePath, major string, versions []string) *VersionList {
return &VersionList{
VersionListKey: VersionListKey{ModulePath: modulePath, Major: major},
Versions: versionSummaries(pkgPath, versions, func(path, version string) string {
return constructPackageURL(pkgPath, modulePath, version)
}),
}
}
for _, tc := range []struct {
name string
pkg *internal.VersionedPackage
modules []*internal.Module
wantDetails *VersionsDetails
}{
{
name: "want stdlib versions",
pkg: nethttpPkg,
modules: []*internal.Module{
sampleModule("std", "v1.12.5", version.TypeRelease, &nethttpPkg.Package),
sampleModule("std", "v1.11.6", version.TypeRelease, &nethttpPkg.Package),
},
wantDetails: &VersionsDetails{
ThisModule: []*VersionList{
makeList("net/http", "std", "go1", []string{"go1.12.5", "go1.11.6"}),
},
},
},
{
name: "want v1 first",
pkg: pkg1,
modules: []*internal.Module{
sampleModule(modulePath1, "v0.0.0-20140414041502-3c2ca4d52544", version.TypePseudo, &pkg2.Package),
sampleModule(modulePath1, "v1.2.3", version.TypeRelease, &pkg1.Package),
sampleModule(modulePath2, "v2.0.0", version.TypeRelease, &pkg2.Package),
sampleModule(modulePath1, "v1.3.0", version.TypeRelease, &pkg1.Package),
sampleModule(modulePath1, "v1.2.1", version.TypeRelease, &pkg1.Package),
sampleModule(modulePath2, "v2.2.1-alpha.1", version.TypePrerelease, &pkg2.Package),
sampleModule("test.com", "v1.2.1", version.TypeRelease, &pkg1.Package),
},
wantDetails: &VersionsDetails{
ThisModule: []*VersionList{
makeList(v1Path, modulePath1, "v1", []string{"v1.3.0", "v1.2.3", "v1.2.1"}),
},
OtherModules: []*VersionList{
makeList(v2Path, modulePath2, "v2", []string{"v2.2.1-alpha.1", "v2.0.0"}),
makeList(v1Path, "test.com", "v1", []string{"v1.2.1"}),
},
},
},
{
name: "want v2 first",
pkg: pkg2,
modules: []*internal.Module{
sampleModule(modulePath1, "v0.0.0-20140414041502-3c2ca4d52544", version.TypePseudo, &pkg1.Package),
sampleModule(modulePath1, "v1.2.1", version.TypeRelease, &pkg1.Package),
sampleModule(modulePath1, "v1.2.3", version.TypeRelease, &pkg1.Package),
sampleModule(modulePath1, "v2.1.0+incompatible", version.TypeRelease, &pkg1.Package),
sampleModule(modulePath2, "v2.0.0", version.TypeRelease, &pkg2.Package),
sampleModule(modulePath2, "v2.2.1-alpha.1", version.TypePrerelease, &pkg2.Package),
},
wantDetails: &VersionsDetails{
ThisModule: []*VersionList{
makeList(v2Path, modulePath2, "v2", []string{"v2.2.1-alpha.1", "v2.0.0"}),
},
OtherModules: []*VersionList{
makeList(v1Path, modulePath1, "v1", []string{"v2.1.0+incompatible", "v1.2.3", "v1.2.1"}),
},
},
},
{
name: "want only pseudo",
pkg: pkg2,
modules: []*internal.Module{
sampleModule(modulePath1, "v0.0.0-20140414041501-3c2ca4d52544", version.TypePseudo, &pkg2.Package),
sampleModule(modulePath1, "v0.0.0-20140414041502-4c2ca4d52544", version.TypePseudo, &pkg2.Package),
},
wantDetails: &VersionsDetails{
OtherModules: []*VersionList{
makeList(v1Path, modulePath1, "v0", []string{
"v0.0.0-20140414041502-4c2ca4d52544",
"v0.0.0-20140414041501-3c2ca4d52544",
}),
},
},
},
} {
t.Run(tc.name, func(t *testing.T) {
defer postgres.ResetTestDB(testDB, t)
for _, v := range tc.modules {
if err := testDB.InsertModule(ctx, v); err != nil {
t.Fatal(err)
}
}
got, err := fetchPackageVersionsDetails(ctx, testDB, tc.pkg)
if err != nil {
t.Fatalf("fetchPackageVersionsDetails(ctx, db, %v): %v", tc.pkg, err)
}
if diff := cmp.Diff(tc.wantDetails, got); diff != "" {
t.Errorf("fetchPackageVersionsDetails(ctx, db, %v) mismatch (-want +got):\n%s", tc.pkg, diff)
}
})
}
}
func TestPathInVersion(t *testing.T) {
tests := []struct {
v1Path, modulePath, want string
}{
{"foo.com/bar/baz", "foo.com/bar", "foo.com/bar/baz"},
{"foo.com/bar/baz", "foo.com/bar/v2", "foo.com/bar/v2/baz"},
{"foo.com/bar/baz", "foo.com/v3", "foo.com/v3/bar/baz"},
{"foo.com/bar/baz", "foo.com/bar/baz/v3", "foo.com/bar/baz/v3"},
}
for _, test := range tests {
mi := sample.ModuleInfo()
mi.ModulePath = test.modulePath
if got := pathInVersion(test.v1Path, mi); got != test.want {
t.Errorf("pathInVersion(%q, ModuleInfo{...ModulePath:%q}) = %s, want %v",
test.v1Path, mi.ModulePath, got, test.want)
}
}
}
func TestFormatVersion(t *testing.T) {
tests := []struct {
version, want string
}{
{"v1.2.3", "v1.2.3"},
{"v2.0.0", "v2.0.0"},
{"v1.2.3-alpha.1", "v1.2.3 (alpha.1)"},
{"v1.0.0-20190311183353-d8887717615a", "v1.0.0 (d888771)"},
{"v1.2.3-pre.0.20190311183353-d8887717615a", "v1.2.3 (d888771)"},
{"v1.2.4-0.20190311183353-d8887717615a", "v1.2.4 (d888771)"},
{"v1.0.0-20190311183353-d88877", "v1.0.0 (d88877)"},
}
for _, test := range tests {
t.Run(test.version, func(t *testing.T) {
if got := formatVersion(test.version); got != test.want {
t.Errorf("formatVersion(%q) = %q, want %q", test.version, got, test.want)
}
})
}
}