2018-09-06 23:15:06 +00:00
|
|
|
// Copyright 2018 clair authors
|
2016-05-17 21:22:18 +00:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2019-03-06 21:35:35 +00:00
|
|
|
package notification
|
2016-01-26 22:57:32 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2016-05-02 22:33:03 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2018-10-08 15:11:30 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2016-05-02 22:33:03 +00:00
|
|
|
|
2016-01-26 22:57:32 +00:00
|
|
|
"github.com/coreos/clair/database"
|
2019-03-06 21:35:35 +00:00
|
|
|
"github.com/coreos/clair/database/pgsql/page"
|
|
|
|
"github.com/coreos/clair/database/pgsql/testutil"
|
2018-10-08 15:11:30 +00:00
|
|
|
"github.com/coreos/clair/pkg/pagination"
|
2016-01-26 22:57:32 +00:00
|
|
|
)
|
|
|
|
|
2018-10-08 15:11:30 +00:00
|
|
|
type findVulnerabilityNotificationIn struct {
|
|
|
|
notificationName string
|
|
|
|
pageSize int
|
|
|
|
oldAffectedAncestryPage pagination.Token
|
|
|
|
newAffectedAncestryPage pagination.Token
|
|
|
|
}
|
2017-07-26 23:23:54 +00:00
|
|
|
|
2018-10-08 15:11:30 +00:00
|
|
|
type findVulnerabilityNotificationOut struct {
|
|
|
|
notification *database.VulnerabilityNotificationWithVulnerable
|
|
|
|
ok bool
|
|
|
|
err string
|
|
|
|
}
|
2016-02-04 16:34:01 +00:00
|
|
|
|
2019-03-06 21:35:35 +00:00
|
|
|
var testPaginationKey = pagination.Must(pagination.NewKey())
|
|
|
|
|
2018-10-08 15:11:30 +00:00
|
|
|
var findVulnerabilityNotificationTests = []struct {
|
|
|
|
title string
|
|
|
|
in findVulnerabilityNotificationIn
|
|
|
|
out findVulnerabilityNotificationOut
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
title: "find notification with invalid page",
|
|
|
|
in: findVulnerabilityNotificationIn{
|
|
|
|
notificationName: "test",
|
|
|
|
pageSize: 1,
|
|
|
|
oldAffectedAncestryPage: pagination.FirstPageToken,
|
|
|
|
newAffectedAncestryPage: pagination.Token("random non sense"),
|
|
|
|
},
|
|
|
|
out: findVulnerabilityNotificationOut{
|
|
|
|
err: pagination.ErrInvalidToken.Error(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "find non-existing notification",
|
|
|
|
in: findVulnerabilityNotificationIn{
|
|
|
|
notificationName: "non-existing",
|
|
|
|
pageSize: 1,
|
|
|
|
oldAffectedAncestryPage: pagination.FirstPageToken,
|
|
|
|
newAffectedAncestryPage: pagination.FirstPageToken,
|
|
|
|
},
|
|
|
|
out: findVulnerabilityNotificationOut{
|
|
|
|
ok: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "find existing notification first page",
|
|
|
|
in: findVulnerabilityNotificationIn{
|
|
|
|
notificationName: "test",
|
|
|
|
pageSize: 1,
|
|
|
|
oldAffectedAncestryPage: pagination.FirstPageToken,
|
|
|
|
newAffectedAncestryPage: pagination.FirstPageToken,
|
|
|
|
},
|
|
|
|
out: findVulnerabilityNotificationOut{
|
|
|
|
&database.VulnerabilityNotificationWithVulnerable{
|
2019-03-06 21:35:35 +00:00
|
|
|
NotificationHook: testutil.RealNotification[1].NotificationHook,
|
2018-10-08 15:11:30 +00:00
|
|
|
Old: &database.PagedVulnerableAncestries{
|
2019-03-06 21:35:35 +00:00
|
|
|
Vulnerability: testutil.RealVulnerability[2],
|
2018-10-08 15:11:30 +00:00
|
|
|
Limit: 1,
|
|
|
|
Affected: make(map[int]string),
|
2019-03-06 21:35:35 +00:00
|
|
|
Current: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{0}),
|
|
|
|
Next: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{0}),
|
2018-10-08 15:11:30 +00:00
|
|
|
End: true,
|
|
|
|
},
|
|
|
|
New: &database.PagedVulnerableAncestries{
|
2019-03-06 21:35:35 +00:00
|
|
|
Vulnerability: testutil.RealVulnerability[1],
|
2018-10-08 15:11:30 +00:00
|
|
|
Limit: 1,
|
|
|
|
Affected: map[int]string{3: "ancestry-3"},
|
2019-03-06 21:35:35 +00:00
|
|
|
Current: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{0}),
|
|
|
|
Next: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{4}),
|
2018-10-08 15:11:30 +00:00
|
|
|
End: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
true,
|
|
|
|
"",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
title: "find existing notification of second page of new affected ancestry",
|
|
|
|
in: findVulnerabilityNotificationIn{
|
|
|
|
notificationName: "test",
|
|
|
|
pageSize: 1,
|
|
|
|
oldAffectedAncestryPage: pagination.FirstPageToken,
|
2019-03-06 21:35:35 +00:00
|
|
|
newAffectedAncestryPage: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{4}),
|
2018-10-08 15:11:30 +00:00
|
|
|
},
|
|
|
|
out: findVulnerabilityNotificationOut{
|
|
|
|
&database.VulnerabilityNotificationWithVulnerable{
|
2019-03-06 21:35:35 +00:00
|
|
|
NotificationHook: testutil.RealNotification[1].NotificationHook,
|
2018-10-08 15:11:30 +00:00
|
|
|
Old: &database.PagedVulnerableAncestries{
|
2019-03-06 21:35:35 +00:00
|
|
|
Vulnerability: testutil.RealVulnerability[2],
|
2018-10-08 15:11:30 +00:00
|
|
|
Limit: 1,
|
|
|
|
Affected: make(map[int]string),
|
2019-03-06 21:35:35 +00:00
|
|
|
Current: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{0}),
|
|
|
|
Next: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{0}),
|
2018-10-08 15:11:30 +00:00
|
|
|
End: true,
|
|
|
|
},
|
|
|
|
New: &database.PagedVulnerableAncestries{
|
2019-03-06 21:35:35 +00:00
|
|
|
Vulnerability: testutil.RealVulnerability[1],
|
2018-10-08 15:11:30 +00:00
|
|
|
Limit: 1,
|
|
|
|
Affected: map[int]string{4: "ancestry-4"},
|
2019-03-06 21:35:35 +00:00
|
|
|
Current: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{4}),
|
|
|
|
Next: testutil.MustMarshalToken(testutil.TestPaginationKey, page.Page{0}),
|
2018-10-08 15:11:30 +00:00
|
|
|
End: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
true,
|
|
|
|
"",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2016-01-26 22:57:32 +00:00
|
|
|
|
2018-10-08 15:11:30 +00:00
|
|
|
func TestFindVulnerabilityNotification(t *testing.T) {
|
2019-03-06 21:35:35 +00:00
|
|
|
tx, cleanup := testutil.CreateTestTxWithFixtures(t, "pagination")
|
|
|
|
defer cleanup()
|
2016-01-26 22:57:32 +00:00
|
|
|
|
2018-10-08 15:11:30 +00:00
|
|
|
for _, test := range findVulnerabilityNotificationTests {
|
|
|
|
t.Run(test.title, func(t *testing.T) {
|
2019-03-06 21:35:35 +00:00
|
|
|
notification, ok, err := FindVulnerabilityNotification(tx, test.in.notificationName, test.in.pageSize, test.in.oldAffectedAncestryPage, test.in.newAffectedAncestryPage, testutil.TestPaginationKey)
|
2018-10-08 15:11:30 +00:00
|
|
|
if test.out.err != "" {
|
|
|
|
require.EqualError(t, err, test.out.err)
|
|
|
|
return
|
2017-07-26 23:23:54 +00:00
|
|
|
}
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2018-10-08 15:11:30 +00:00
|
|
|
require.Nil(t, err)
|
|
|
|
if !test.out.ok {
|
|
|
|
require.Equal(t, test.out.ok, ok)
|
|
|
|
return
|
2017-07-26 23:23:54 +00:00
|
|
|
}
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2018-10-08 15:11:30 +00:00
|
|
|
require.True(t, ok)
|
2019-03-06 21:35:35 +00:00
|
|
|
testutil.AssertVulnerabilityNotificationWithVulnerableEqual(t, testutil.TestPaginationKey, test.out.notification, ¬ification)
|
2018-10-08 15:11:30 +00:00
|
|
|
})
|
2017-07-26 23:23:54 +00:00
|
|
|
}
|
|
|
|
}
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2017-07-26 23:23:54 +00:00
|
|
|
func TestInsertVulnerabilityNotifications(t *testing.T) {
|
2019-03-06 21:35:35 +00:00
|
|
|
datastore, cleanup := testutil.CreateTestDBWithFixture(t, "InsertVulnerabilityNotifications")
|
|
|
|
defer cleanup()
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2017-07-26 23:23:54 +00:00
|
|
|
n1 := database.VulnerabilityNotification{}
|
|
|
|
n3 := database.VulnerabilityNotification{
|
|
|
|
NotificationHook: database.NotificationHook{
|
|
|
|
Name: "random name",
|
|
|
|
Created: time.Now(),
|
|
|
|
},
|
|
|
|
Old: nil,
|
|
|
|
New: &database.Vulnerability{},
|
|
|
|
}
|
|
|
|
n4 := database.VulnerabilityNotification{
|
|
|
|
NotificationHook: database.NotificationHook{
|
|
|
|
Name: "random name",
|
|
|
|
Created: time.Now(),
|
|
|
|
},
|
|
|
|
Old: nil,
|
|
|
|
New: &database.Vulnerability{
|
|
|
|
Name: "CVE-OPENSSL-1-DEB7",
|
|
|
|
Namespace: database.Namespace{
|
|
|
|
Name: "debian:7",
|
|
|
|
VersionFormat: "dpkg",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2019-03-06 21:35:35 +00:00
|
|
|
tx, err := datastore.Begin()
|
|
|
|
require.Nil(t, err)
|
|
|
|
|
2017-07-26 23:23:54 +00:00
|
|
|
// invalid case
|
2019-03-06 21:35:35 +00:00
|
|
|
err = InsertVulnerabilityNotifications(tx, []database.VulnerabilityNotification{n1})
|
|
|
|
require.NotNil(t, err)
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2017-07-26 23:23:54 +00:00
|
|
|
// invalid case: unknown vulnerability
|
2019-03-06 21:35:35 +00:00
|
|
|
err = InsertVulnerabilityNotifications(tx, []database.VulnerabilityNotification{n3})
|
|
|
|
require.NotNil(t, err)
|
2017-07-26 23:23:54 +00:00
|
|
|
|
|
|
|
// invalid case: duplicated input notification
|
2019-03-06 21:35:35 +00:00
|
|
|
err = InsertVulnerabilityNotifications(tx, []database.VulnerabilityNotification{n4, n4})
|
|
|
|
require.NotNil(t, err)
|
|
|
|
tx = testutil.RestartTransaction(datastore, tx, false)
|
2017-07-26 23:23:54 +00:00
|
|
|
|
|
|
|
// valid case
|
2019-03-06 21:35:35 +00:00
|
|
|
err = InsertVulnerabilityNotifications(tx, []database.VulnerabilityNotification{n4})
|
|
|
|
require.Nil(t, err)
|
2017-07-26 23:23:54 +00:00
|
|
|
// invalid case: notification is already in database
|
2019-03-06 21:35:35 +00:00
|
|
|
err = InsertVulnerabilityNotifications(tx, []database.VulnerabilityNotification{n4})
|
|
|
|
require.NotNil(t, err)
|
2017-07-26 23:23:54 +00:00
|
|
|
|
2019-03-06 21:35:35 +00:00
|
|
|
require.Nil(t, tx.Rollback())
|
2017-07-26 23:23:54 +00:00
|
|
|
}
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2017-07-26 23:23:54 +00:00
|
|
|
func TestFindNewNotification(t *testing.T) {
|
2019-03-06 21:35:35 +00:00
|
|
|
tx, cleanup := testutil.CreateTestTxWithFixtures(t, "TestFindNewNotification")
|
2019-02-19 21:42:00 +00:00
|
|
|
defer cleanup()
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2019-03-06 21:35:35 +00:00
|
|
|
noti, ok, err := FindNewNotification(tx, time.Now())
|
2017-07-26 23:23:54 +00:00
|
|
|
if assert.Nil(t, err) && assert.True(t, ok) {
|
|
|
|
assert.Equal(t, "test", noti.Name)
|
|
|
|
assert.Equal(t, time.Time{}, noti.Notified)
|
|
|
|
assert.Equal(t, time.Time{}, noti.Created)
|
|
|
|
assert.Equal(t, time.Time{}, noti.Deleted)
|
2016-02-04 16:34:01 +00:00
|
|
|
}
|
2016-01-26 22:57:32 +00:00
|
|
|
|
2017-07-26 23:23:54 +00:00
|
|
|
// can't find the notified
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.Nil(t, MarkNotificationAsRead(tx, "test"))
|
2017-07-26 23:23:54 +00:00
|
|
|
// if the notified time is before
|
2019-03-06 21:35:35 +00:00
|
|
|
noti, ok, err = FindNewNotification(tx, time.Now().Add(-time.Duration(10*time.Second)))
|
2017-07-26 23:23:54 +00:00
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.False(t, ok)
|
|
|
|
// can find the notified after a period of time
|
2019-03-06 21:35:35 +00:00
|
|
|
noti, ok, err = FindNewNotification(tx, time.Now().Add(time.Duration(10*time.Second)))
|
2017-07-26 23:23:54 +00:00
|
|
|
if assert.Nil(t, err) && assert.True(t, ok) {
|
|
|
|
assert.Equal(t, "test", noti.Name)
|
|
|
|
assert.NotEqual(t, time.Time{}, noti.Notified)
|
|
|
|
assert.Equal(t, time.Time{}, noti.Created)
|
|
|
|
assert.Equal(t, time.Time{}, noti.Deleted)
|
2016-02-04 20:12:41 +00:00
|
|
|
}
|
2016-01-26 22:57:32 +00:00
|
|
|
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.Nil(t, DeleteNotification(tx, "test"))
|
2017-07-26 23:23:54 +00:00
|
|
|
// can't find in any time
|
2019-03-06 21:35:35 +00:00
|
|
|
noti, ok, err = FindNewNotification(tx, time.Now().Add(-time.Duration(1000)))
|
2017-07-26 23:23:54 +00:00
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.False(t, ok)
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2019-03-06 21:35:35 +00:00
|
|
|
noti, ok, err = FindNewNotification(tx, time.Now().Add(time.Duration(1000)))
|
2017-07-26 23:23:54 +00:00
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.False(t, ok)
|
|
|
|
}
|
2016-01-26 22:57:32 +00:00
|
|
|
|
2018-09-11 18:24:09 +00:00
|
|
|
func TestMarkNotificationAsRead(t *testing.T) {
|
2019-03-06 21:35:35 +00:00
|
|
|
tx, cleanup := testutil.CreateTestTxWithFixtures(t, "MarkNotificationAsRead")
|
|
|
|
defer cleanup()
|
2016-02-04 20:12:41 +00:00
|
|
|
|
2017-07-26 23:23:54 +00:00
|
|
|
// invalid case: notification doesn't exist
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.NotNil(t, MarkNotificationAsRead(tx, "non-existing"))
|
2017-07-26 23:23:54 +00:00
|
|
|
// valid case
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.Nil(t, MarkNotificationAsRead(tx, "test"))
|
2017-07-26 23:23:54 +00:00
|
|
|
// valid case
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.Nil(t, MarkNotificationAsRead(tx, "test"))
|
2017-07-26 23:23:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestDeleteNotification(t *testing.T) {
|
2019-03-06 21:35:35 +00:00
|
|
|
tx, cleanup := testutil.CreateTestTxWithFixtures(t, "DeleteNotification")
|
|
|
|
defer cleanup()
|
2017-07-26 23:23:54 +00:00
|
|
|
|
|
|
|
// invalid case: notification doesn't exist
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.NotNil(t, DeleteNotification(tx, "non-existing"))
|
2017-07-26 23:23:54 +00:00
|
|
|
// valid case
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.Nil(t, DeleteNotification(tx, "test"))
|
2017-07-26 23:23:54 +00:00
|
|
|
// invalid case: notification is already deleted
|
2019-03-06 21:35:35 +00:00
|
|
|
assert.NotNil(t, DeleteNotification(tx, "test"))
|
2016-01-26 22:57:32 +00:00
|
|
|
}
|