Retention: New Evaluator: Pulled within the last N Days

Signed-off-by: Nathan Lowe <public@nlowe.me>
This commit is contained in:
Nathan Lowe 2019-08-04 20:05:05 -04:00
parent ec4fa753d7
commit 9a7df265ce
No known key found for this signature in database
GPG Key ID: 1091439964459621
5 changed files with 215 additions and 4 deletions

View File

@ -93,7 +93,30 @@ func (r *RetentionAPI) GetMetadatas() {
"rule_template": "nothing", "rule_template": "nothing",
"display_text": "none", "display_text": "none",
"action": "retain", "action": "retain",
"params": []
},
{
"rule_template": "always",
"display_text": "always",
"action": "retain",
"params": [ "params": [
{
"type": "int",
"unit": "COUNT",
"required": true
}
]
},
{
"rule_template": "dayspl",
"display_text": "pulled within the last # days",
"action": "retain",
"params": [
{
"type": "int",
"unit": "DAYS",
"required": true
}
] ]
}, },
{ {

View File

@ -0,0 +1,69 @@
// Copyright Project Harbor 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 dayspl
import (
"time"
"github.com/goharbor/harbor/src/common/utils/log"
"github.com/goharbor/harbor/src/pkg/retention/policy/action"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule"
"github.com/goharbor/harbor/src/pkg/retention/res"
)
const (
// TemplateID of the rule
TemplateID = "nDaysSinceLastPull"
// ParameterN is the name of the metadata parameter for the N value
ParameterN = TemplateID
// DefaultN is the default number of days that an artifact must have
// been pulled within to retain the tag or artifact.
DefaultN = 30
)
type evaluator struct {
n int
}
func (e *evaluator) Process(artifacts []*res.Candidate) (result []*res.Candidate, err error) {
minPullTime := time.Now().UTC().Add(time.Duration(-1*24*e.n) * time.Hour).Unix()
for _, a := range artifacts {
if a.PulledTime >= minPullTime {
result = append(result, a)
}
}
return
}
func (e *evaluator) Action() string {
return action.Retain
}
// New constructs a new 'Days Since Last Pull' evaluator
func New(params rule.Parameters) rule.Evaluator {
if params != nil {
if p, ok := params[ParameterN]; ok {
if v, ok := p.(int); ok && v >= 0 {
return &evaluator{n: v}
}
}
}
log.Debugf("default parameter %d used for rule %s", DefaultN, TemplateID)
return &evaluator{n: DefaultN}
}

View File

@ -0,0 +1,104 @@
// Copyright Project Harbor 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 dayspl
import (
"strconv"
"testing"
"time"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule"
"github.com/goharbor/harbor/src/pkg/retention/res"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/stretchr/testify/suite"
)
type EvaluatorTestSuite struct {
suite.Suite
}
func (e *EvaluatorTestSuite) TestNew() {
tests := []struct {
Name string
args rule.Parameters
expectedN int
}{
{Name: "Valid", args: map[string]rule.Parameter{ParameterN: 5}, expectedN: 5},
{Name: "Default If Negative", args: map[string]rule.Parameter{ParameterN: -1}, expectedN: DefaultN},
{Name: "Default If Not Set", args: map[string]rule.Parameter{}, expectedN: DefaultN},
{Name: "Default If Wrong Type", args: map[string]rule.Parameter{ParameterN: "foo"}, expectedN: DefaultN},
}
for _, tt := range tests {
e.T().Run(tt.Name, func(t *testing.T) {
e := New(tt.args).(*evaluator)
require.Equal(t, tt.expectedN, e.n)
})
}
}
func (e *EvaluatorTestSuite) TestProcess() {
now := time.Now().UTC()
data := []*res.Candidate{
{PulledTime: daysAgo(now, 1)},
{PulledTime: daysAgo(now, 2)},
{PulledTime: daysAgo(now, 3)},
{PulledTime: daysAgo(now, 4)},
{PulledTime: daysAgo(now, 5)},
{PulledTime: daysAgo(now, 10)},
{PulledTime: daysAgo(now, 20)},
{PulledTime: daysAgo(now, 30)},
}
tests := []struct {
n int
expected int
minPullTime int64
}{
{n: 0, expected: 0, minPullTime: 0},
{n: 1, expected: 1, minPullTime: daysAgo(now, 1)},
{n: 2, expected: 2, minPullTime: daysAgo(now, 2)},
{n: 3, expected: 3, minPullTime: daysAgo(now, 3)},
{n: 4, expected: 4, minPullTime: daysAgo(now, 4)},
{n: 5, expected: 5, minPullTime: daysAgo(now, 5)},
{n: 15, expected: 6, minPullTime: daysAgo(now, 10)},
{n: 90, expected: 8, minPullTime: daysAgo(now, 30)},
}
for _, tt := range tests {
e.T().Run(strconv.Itoa(tt.n), func(t *testing.T) {
sut := New(map[string]rule.Parameter{ParameterN: tt.n})
result, err := sut.Process(data)
require.NoError(t, err)
require.Len(t, result, tt.expected)
for _, v := range result {
assert.False(t, v.PulledTime < tt.minPullTime)
}
})
}
}
func TestEvaluatorSuite(t *testing.T) {
suite.Run(t, &EvaluatorTestSuite{})
}
func daysAgo(from time.Time, n int) int64 {
return from.Add(time.Duration(-1*24*n) * time.Hour).Unix()
}

View File

@ -15,17 +15,18 @@
package index package index
import ( import (
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/nothing"
"sync" "sync"
"github.com/goharbor/harbor/src/pkg/retention/policy/action" "github.com/goharbor/harbor/src/pkg/retention/policy/action"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule" "github.com/goharbor/harbor/src/pkg/retention/policy/rule"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/always" "github.com/goharbor/harbor/src/pkg/retention/policy/rule/always"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/dayspl"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/daysps" "github.com/goharbor/harbor/src/pkg/retention/policy/rule/daysps"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/lastx" "github.com/goharbor/harbor/src/pkg/retention/policy/rule/lastx"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/latestk" "github.com/goharbor/harbor/src/pkg/retention/policy/rule/latestk"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/latestpl" "github.com/goharbor/harbor/src/pkg/retention/policy/rule/latestpl"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/latestps" "github.com/goharbor/harbor/src/pkg/retention/policy/rule/latestps"
"github.com/goharbor/harbor/src/pkg/retention/policy/rule/nothing"
"github.com/pkg/errors" "github.com/pkg/errors"
) )
@ -134,6 +135,20 @@ func init() {
Parameters: []*IndexedParam{}, Parameters: []*IndexedParam{},
}, always.New) }, always.New)
// Register dayspl
Register(&Metadata{
TemplateID: dayspl.TemplateID,
Action: action.Retain,
Parameters: []*IndexedParam{
{
Name: dayspl.ParameterN,
Type: "int",
Unit: "days",
Required: true,
},
},
}, dayspl.New)
// Register daysps // Register daysps
Register(&Metadata{ Register(&Metadata{
TemplateID: daysps.TemplateID, TemplateID: daysps.TemplateID,

View File

@ -84,7 +84,7 @@ func (suite *IndexTestSuite) TestGet() {
// TestIndex tests Index // TestIndex tests Index
func (suite *IndexTestSuite) TestIndex() { func (suite *IndexTestSuite) TestIndex() {
metas := Index() metas := Index()
require.Equal(suite.T(), 7, len(metas)) require.Equal(suite.T(), 9, len(metas))
assert.Condition(suite.T(), func() bool { assert.Condition(suite.T(), func() bool {
for _, m := range metas { for _, m := range metas {
if m.TemplateID == "fakeEvaluator" && if m.TemplateID == "fakeEvaluator" &&