5d725e67b0
As one of the steps to simplifies the codebase, the AncestryWithContent struct is renamed to Ancestry, and Ancestry is removed. It will cause the PostAncestry request to be slower.
213 lines
4.9 KiB
Go
213 lines
4.9 KiB
Go
// Copyright 2017 clair 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 pgsql
|
|
|
|
import (
|
|
"sort"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/coreos/clair/database"
|
|
)
|
|
|
|
func TestUpsertAncestry(t *testing.T) {
|
|
store, tx := openSessionForTest(t, "UpsertAncestry", true)
|
|
defer closeTest(t, store, tx)
|
|
a1 := database.Ancestry{
|
|
Name: "a1",
|
|
Layers: []database.AncestryLayer{
|
|
{
|
|
Layer: database.Layer{
|
|
Hash: "layer-N",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
a2 := database.Ancestry{}
|
|
|
|
a3 := database.Ancestry{
|
|
Name: "a",
|
|
Layers: []database.AncestryLayer{
|
|
{
|
|
Layer: database.Layer{
|
|
Hash: "layer-0",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
a4 := database.Ancestry{
|
|
Name: "a",
|
|
Layers: []database.AncestryLayer{
|
|
{
|
|
Layer: database.Layer{
|
|
Hash: "layer-1",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
f1 := database.Feature{
|
|
Name: "wechat",
|
|
Version: "0.5",
|
|
VersionFormat: "dpkg",
|
|
}
|
|
|
|
// not in database
|
|
f2 := database.Feature{
|
|
Name: "wechat",
|
|
Version: "0.6",
|
|
VersionFormat: "dpkg",
|
|
}
|
|
|
|
n1 := database.Namespace{
|
|
Name: "debian:7",
|
|
VersionFormat: "dpkg",
|
|
}
|
|
|
|
p := database.Processors{
|
|
Listers: []string{"dpkg", "non-existing"},
|
|
Detectors: []string{"os-release", "non-existing"},
|
|
}
|
|
|
|
nsf1 := database.NamespacedFeature{
|
|
Namespace: n1,
|
|
Feature: f1,
|
|
}
|
|
|
|
// not in database
|
|
nsf2 := database.NamespacedFeature{
|
|
Namespace: n1,
|
|
Feature: f2,
|
|
}
|
|
|
|
a4.ProcessedBy = p
|
|
// invalid case
|
|
assert.NotNil(t, tx.UpsertAncestry(a1))
|
|
assert.NotNil(t, tx.UpsertAncestry(a2))
|
|
// valid case
|
|
assert.Nil(t, tx.UpsertAncestry(a3))
|
|
a4.Layers[0].DetectedFeatures = []database.NamespacedFeature{nsf1, nsf2}
|
|
// replace invalid case
|
|
assert.NotNil(t, tx.UpsertAncestry(a4))
|
|
a4.Layers[0].DetectedFeatures = []database.NamespacedFeature{nsf1}
|
|
// replace valid case
|
|
assert.Nil(t, tx.UpsertAncestry(a4))
|
|
// validate
|
|
ancestry, ok, err := tx.FindAncestry("a")
|
|
assert.Nil(t, err)
|
|
assert.True(t, ok)
|
|
assertAncestryEqual(t, a4, ancestry)
|
|
}
|
|
|
|
func assertProcessorsEqual(t *testing.T, expected database.Processors, actual database.Processors) bool {
|
|
sort.Strings(expected.Detectors)
|
|
sort.Strings(actual.Detectors)
|
|
sort.Strings(expected.Listers)
|
|
sort.Strings(actual.Listers)
|
|
return assert.Equal(t, expected.Detectors, actual.Detectors) && assert.Equal(t, expected.Listers, actual.Listers)
|
|
}
|
|
|
|
func assertAncestryEqual(t *testing.T, expected database.Ancestry, actual database.Ancestry) bool {
|
|
assert.Equal(t, expected.Name, actual.Name)
|
|
assertProcessorsEqual(t, expected.ProcessedBy, actual.ProcessedBy)
|
|
if assert.Equal(t, len(expected.Layers), len(actual.Layers)) {
|
|
for index, layer := range expected.Layers {
|
|
if !assertAncestryLayerEqual(t, layer, actual.Layers[index]) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func assertAncestryLayerEqual(t *testing.T, expected database.AncestryLayer, actual database.AncestryLayer) bool {
|
|
return assertLayerEqual(t, expected.Layer, actual.Layer) &&
|
|
assertNamespacedFeatureEqual(t, expected.DetectedFeatures, actual.DetectedFeatures)
|
|
}
|
|
|
|
func TestFindAncestry(t *testing.T) {
|
|
store, tx := openSessionForTest(t, "FindAncestry", true)
|
|
defer closeTest(t, store, tx)
|
|
|
|
// invalid
|
|
_, ok, err := tx.FindAncestry("ancestry-non")
|
|
if assert.Nil(t, err) {
|
|
assert.False(t, ok)
|
|
}
|
|
|
|
expected := database.Ancestry{
|
|
Name: "ancestry-2",
|
|
ProcessedBy: database.Processors{
|
|
Detectors: []string{"os-release"},
|
|
Listers: []string{"dpkg"},
|
|
},
|
|
Layers: []database.AncestryLayer{
|
|
{
|
|
Layer: database.Layer{
|
|
Hash: "layer-0",
|
|
},
|
|
DetectedFeatures: []database.NamespacedFeature{
|
|
{
|
|
Namespace: database.Namespace{
|
|
Name: "debian:7",
|
|
VersionFormat: "dpkg",
|
|
},
|
|
Feature: database.Feature{
|
|
Name: "wechat",
|
|
Version: "0.5",
|
|
VersionFormat: "dpkg",
|
|
},
|
|
},
|
|
{
|
|
Namespace: database.Namespace{
|
|
Name: "debian:8",
|
|
VersionFormat: "dpkg",
|
|
},
|
|
Feature: database.Feature{
|
|
Name: "openssl",
|
|
Version: "1.0",
|
|
VersionFormat: "dpkg",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
{
|
|
Layer: database.Layer{
|
|
Hash: "layer-1",
|
|
},
|
|
},
|
|
{
|
|
Layer: database.Layer{
|
|
Hash: "layer-2",
|
|
},
|
|
},
|
|
{
|
|
Layer: database.Layer{
|
|
Hash: "layer-3b",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
// valid
|
|
ancestry, ok, err := tx.FindAncestry("ancestry-2")
|
|
if assert.Nil(t, err) && assert.True(t, ok) {
|
|
assertAncestryEqual(t, expected, ancestry)
|
|
}
|
|
}
|