mirror of
https://github.com/goharbor/harbor.git
synced 2025-01-12 02:41:50 +01:00
Retention: New Evaluator: Pulled within the last N Days
Signed-off-by: Nathan Lowe <public@nlowe.me>
This commit is contained in:
parent
ec4fa753d7
commit
9a7df265ce
@ -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
|
||||||
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
69
src/pkg/retention/policy/rule/dayspl/evaluator.go
Normal file
69
src/pkg/retention/policy/rule/dayspl/evaluator.go
Normal 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}
|
||||||
|
}
|
104
src/pkg/retention/policy/rule/dayspl/evaluator_test.go
Normal file
104
src/pkg/retention/policy/rule/dayspl/evaluator_test.go
Normal 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()
|
||||||
|
}
|
@ -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,
|
||||||
|
@ -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" &&
|
||||||
|
Loading…
Reference in New Issue
Block a user