diff --git a/src/controller/artifact/controller.go b/src/controller/artifact/controller.go index b183cf2f5..ab74352ae 100644 --- a/src/controller/artifact/controller.go +++ b/src/controller/artifact/controller.go @@ -492,10 +492,7 @@ tags: } func (c *controller) UpdatePullTime(ctx context.Context, artifactID int64, tagID int64, time time.Time) error { - if err := c.artMgr.Update(ctx, &artifact.Artifact{ - ID: artifactID, - PullTime: time, - }, "PullTime"); err != nil { + if err := c.artMgr.UpdatePullTime(ctx, artifactID, time); err != nil { return err } tg, err := c.tagCtl.Get(ctx, tagID, nil) diff --git a/src/controller/artifact/controller_test.go b/src/controller/artifact/controller_test.go index 3619e28ac..302c0c3a6 100644 --- a/src/controller/artifact/controller_test.go +++ b/src/controller/artifact/controller_test.go @@ -562,7 +562,7 @@ func (c *controllerTestSuite) TestUpdatePullTime() { }, }, nil) c.tagCtl.On("Update").Return(nil) - c.artMgr.On("Update", mock.Anything, mock.Anything, mock.Anything).Return(nil) + c.artMgr.On("UpdatePullTime", mock.Anything, mock.Anything, mock.Anything).Return(nil) err := c.ctl.UpdatePullTime(nil, 1, 1, time.Now()) c.Require().Nil(err) c.artMgr.AssertExpectations(c.T()) @@ -578,7 +578,7 @@ func (c *controllerTestSuite) TestUpdatePullTime() { ArtifactID: 2, }, }, nil) - c.artMgr.On("Update", mock.Anything, mock.Anything, mock.Anything).Return(nil) + c.artMgr.On("UpdatePullTime", mock.Anything, mock.Anything, mock.Anything).Return(nil) err = c.ctl.UpdatePullTime(nil, 1, 1, time.Now()) c.Require().NotNil(err) c.tagCtl.AssertExpectations(c.T()) diff --git a/src/controller/event/handler/internal/artifact.go b/src/controller/event/handler/internal/artifact.go index 9be7dfda3..cfc29354d 100644 --- a/src/controller/event/handler/internal/artifact.go +++ b/src/controller/event/handler/internal/artifact.go @@ -16,19 +16,63 @@ package internal import ( "context" - "github.com/goharbor/harbor/src/lib/config" + "fmt" + "os" + "strconv" + "strings" + "sync" "time" "github.com/goharbor/harbor/src/controller/artifact" "github.com/goharbor/harbor/src/controller/event" "github.com/goharbor/harbor/src/controller/repository" "github.com/goharbor/harbor/src/controller/tag" + "github.com/goharbor/harbor/src/lib/config" "github.com/goharbor/harbor/src/lib/log" + "github.com/goharbor/harbor/src/lib/orm" "github.com/goharbor/harbor/src/lib/q" ) +const ( + // defaultAsyncFlushDuration is the default flush interval. + defaultAsyncFlushDuration = 10 * time.Second +) + +var ( + asyncFlushDuration time.Duration +) + +func init() { + // get async flush duration from env, if not provide, + // use default value: 10*time.second + envDuration := os.Getenv("ARTIFACT_PULL_ASYNC_FLUSH_DURATION") + if len(envDuration) == 0 { + // use default value + asyncFlushDuration = defaultAsyncFlushDuration + } else { + duration, err := strconv.ParseInt(envDuration, 10, 64) + if err != nil { + log.Warningf("error to parse ARTIFACT_PULL_ASYNC_FLUSH_DURATION: %v, will use default value: %v", err, defaultAsyncFlushDuration) + asyncFlushDuration = defaultAsyncFlushDuration + } else { + asyncFlushDuration = time.Duration(duration) * time.Second + } + } +} + // Handler preprocess artifact event data type Handler struct { + once sync.Once + // pullCountStore caches the pull count group by repository + // map[repositoryID]counts + pullCountStore map[int64]uint64 + // pullCountLock mutex for pullCountStore + pullCountLock sync.Mutex + // pullTimeStore caches the latest pull time group by artifact + // map[artifactID:tagName]time + pullTimeStore map[string]time.Time + // pullTimeLock mutex for pullTimeStore + pullTimeLock sync.Mutex } // Name ... @@ -55,43 +99,136 @@ func (a *Handler) IsStateful() bool { } func (a *Handler) onPull(ctx context.Context, event *event.ArtifactEvent) error { + // if duration is equal to 0 or negative, keep original sync mode. + if asyncFlushDuration <= 0 { + var tagName string + if len(event.Tags) > 0 { + tagName = event.Tags[0] + } + + if !config.PullTimeUpdateDisable(ctx) { + a.syncFlushPullTime(ctx, event.Artifact.ID, tagName, time.Now()) + } + + if !config.PullCountUpdateDisable(ctx) { + a.syncFlushPullCount(ctx, event.Artifact.RepositoryID, 1) + } + + return nil + } + + // async mode, update in cache firstly and flush to db by workers periodically. + a.once.Do(func() { + if !config.PullTimeUpdateDisable(ctx) { + a.pullTimeStore = make(map[string]time.Time) + go a.asyncFlushPullTime(orm.Context()) + } + + if !config.PullCountUpdateDisable(ctx) { + a.pullCountStore = make(map[int64]uint64) + go a.asyncFlushPullCount(orm.Context()) + } + }) + if !config.PullTimeUpdateDisable(ctx) { - go func() { a.updatePullTime(ctx, event) }() + a.updatePullTimeInCache(ctx, event) } + if !config.PullCountUpdateDisable(ctx) { - go func() { a.addPullCount(ctx, event) }() + a.addPullCountInCache(ctx, event) } + return nil } -func (a *Handler) updatePullTime(ctx context.Context, event *event.ArtifactEvent) { - var tagID int64 +func (a *Handler) updatePullTimeInCache(ctx context.Context, event *event.ArtifactEvent) { + var tagName string if len(event.Tags) != 0 { - tags, err := tag.Ctl.List(ctx, &q.Query{ - Keywords: map[string]interface{}{ - "ArtifactID": event.Artifact.ID, - "Name": event.Tags[0], - }, - }, nil) + tagName = event.Tags[0] + } + + key := fmt.Sprintf("%d:%s", event.Artifact.ID, tagName) + + a.pullTimeLock.Lock() + defer a.pullTimeLock.Unlock() + + a.pullTimeStore[key] = time.Now() +} + +func (a *Handler) addPullCountInCache(ctx context.Context, event *event.ArtifactEvent) { + a.pullCountLock.Lock() + defer a.pullCountLock.Unlock() + + a.pullCountStore[event.Artifact.RepositoryID] = a.pullCountStore[event.Artifact.RepositoryID] + 1 +} + +func (a *Handler) syncFlushPullTime(ctx context.Context, artifactID int64, tagName string, time time.Time) { + var tagID int64 + + if tagName != "" { + tags, err := tag.Ctl.List(ctx, q.New( + map[string]interface{}{ + "ArtifactID": artifactID, + "Name": tagName, + }), nil) if err != nil { - log.Infof("failed to list tags when to update pull time, %v", err) + log.Warningf("failed to list tags when to update pull time, %v", err) } else { if len(tags) != 0 { tagID = tags[0].ID } } } - if err := artifact.Ctl.UpdatePullTime(ctx, event.Artifact.ID, tagID, time.Now()); err != nil { - log.Debugf("failed to update pull time form artifact %d, %v", event.Artifact.ID, err) + + if err := artifact.Ctl.UpdatePullTime(ctx, artifactID, tagID, time); err != nil { + log.Warningf("failed to update pull time for artifact %d, %v", artifactID, err) } - return } -func (a *Handler) addPullCount(ctx context.Context, event *event.ArtifactEvent) { - if err := repository.Ctl.AddPullCount(ctx, event.Artifact.RepositoryID); err != nil { - log.Debugf("failed to add pull count repository %d, %v", event.Artifact.RepositoryID, err) +func (a *Handler) syncFlushPullCount(ctx context.Context, repositoryID int64, count uint64) { + if err := repository.Ctl.AddPullCount(ctx, repositoryID, count); err != nil { + log.Warningf("failed to add pull count repository %d, %v", repositoryID, err) + } +} + +func (a *Handler) asyncFlushPullTime(ctx context.Context) { + for { + <-time.After(asyncFlushDuration) + a.pullTimeLock.Lock() + + for key, time := range a.pullTimeStore { + keys := strings.Split(key, ":") + artifactID, err := strconv.ParseInt(keys[0], 10, 64) + if err != nil { + log.Warningf("failed to parse artifact id %s, %v", key, err) + continue + } + + var tagName string + if len(keys) > 1 && keys[1] != "" { + tagName = keys[1] + } + + a.syncFlushPullTime(ctx, artifactID, tagName, time) + } + + a.pullTimeStore = make(map[string]time.Time) + a.pullTimeLock.Unlock() + } +} + +func (a *Handler) asyncFlushPullCount(ctx context.Context) { + for { + <-time.After(asyncFlushDuration) + a.pullCountLock.Lock() + + for repositoryID, count := range a.pullCountStore { + a.syncFlushPullCount(ctx, repositoryID, count) + } + + a.pullCountStore = make(map[int64]uint64) + a.pullCountLock.Unlock() } - return } func (a *Handler) onPush(ctx context.Context, event *event.ArtifactEvent) error { diff --git a/src/controller/event/handler/internal/artifact_test.go b/src/controller/event/handler/internal/artifact_test.go index 5bf0569ce..cdc499603 100644 --- a/src/controller/event/handler/internal/artifact_test.go +++ b/src/controller/event/handler/internal/artifact_test.go @@ -1 +1,144 @@ +// 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 internal + +import ( + "context" + "testing" + "time" + + beegoorm "github.com/astaxie/beego/orm" + common_dao "github.com/goharbor/harbor/src/common/dao" + "github.com/goharbor/harbor/src/controller/event" + "github.com/goharbor/harbor/src/lib/config" + "github.com/goharbor/harbor/src/lib/orm" + "github.com/goharbor/harbor/src/pkg/artifact" + _ "github.com/goharbor/harbor/src/pkg/config/db" + repo "github.com/goharbor/harbor/src/pkg/repository" + "github.com/goharbor/harbor/src/pkg/repository/model" + "github.com/goharbor/harbor/src/pkg/tag" + tagmodel "github.com/goharbor/harbor/src/pkg/tag/model/tag" + "github.com/stretchr/testify/suite" +) + +// ArtifactHandlerTestSuite is test suite for artifact handler. +type ArtifactHandlerTestSuite struct { + suite.Suite + + ctx context.Context + handler *Handler +} + +// TestArtifactHandler tests ArtifactHandler. +func TestArtifactHandler(t *testing.T) { + suite.Run(t, &ArtifactHandlerTestSuite{}) +} + +// SetupSuite prepares for running ArtifactHandlerTestSuite. +func (suite *ArtifactHandlerTestSuite) SetupSuite() { + common_dao.PrepareTestForPostgresSQL() + config.Init() + suite.handler = &Handler{} + suite.ctx = orm.NewContext(context.TODO(), beegoorm.NewOrm()) + + // mock artifact + _, err := artifact.Mgr.Create(suite.ctx, &artifact.Artifact{ID: 1, RepositoryID: 1}) + suite.Nil(err) + // mock repository + _, err = repo.Mgr.Create(suite.ctx, &model.RepoRecord{RepositoryID: 1}) + suite.Nil(err) + // mock tag + _, err = tag.Mgr.Create(suite.ctx, &tagmodel.Tag{ID: 1, RepositoryID: 1, ArtifactID: 1, Name: "latest"}) + suite.Nil(err) +} + +// TearDownSuite cleans environment. +func (suite *ArtifactHandlerTestSuite) TearDownSuite() { + // delete tag + err := tag.Mgr.Delete(suite.ctx, 1) + suite.Nil(err) + // delete artifact + err = artifact.Mgr.Delete(suite.ctx, 1) + suite.Nil(err) + // delete repository + err = repo.Mgr.Delete(suite.ctx, 1) + suite.Nil(err) + +} + +// TestName tests method Name. +func (suite *ArtifactHandlerTestSuite) TestName() { + suite.Equal("InternalArtifact", suite.handler.Name()) +} + +// TestIsStateful tests method IsStateful. +func (suite *ArtifactHandlerTestSuite) TestIsStateful() { + suite.False(suite.handler.IsStateful(), "artifact handler is not stateful") +} + +// TestDefaultAsyncDuration tests default value of async flush duration. +func (suite *ArtifactHandlerTestSuite) TestDefaultAsyncFlushDuration() { + suite.Equal(defaultAsyncFlushDuration, asyncFlushDuration, "default async flush duration") +} + +// TestOnPush tests handle push events. +func (suite *ArtifactHandlerTestSuite) TestOnPush() { + err := suite.handler.onPush(context.TODO(), &event.ArtifactEvent{Artifact: &artifact.Artifact{}}) + suite.Nil(err, "onPush should return nil") +} + +// TestOnPull tests handler pull events. +func (suite *ArtifactHandlerTestSuite) TestOnPull() { + // test sync mode + asyncFlushDuration = 0 + err := suite.handler.onPull(suite.ctx, &event.ArtifactEvent{Artifact: &artifact.Artifact{ID: 1, RepositoryID: 1}, Tags: []string{"latest"}}) + suite.Nil(err, "onPull should return nil") + // sync mode should update db immediately + // pull_time + art, err := artifact.Mgr.Get(suite.ctx, 1) + suite.Nil(err) + suite.False(art.PullTime.IsZero(), "sync update pull_time") + lastPullTime := art.PullTime + // pull_count + repository, err := repo.Mgr.Get(suite.ctx, 1) + suite.Nil(err) + suite.Equal(int64(1), repository.PullCount, "sync update pull_count") + + // test async mode + asyncFlushDuration = 200 * time.Millisecond + err = suite.handler.onPull(suite.ctx, &event.ArtifactEvent{Artifact: &artifact.Artifact{ID: 1, RepositoryID: 1}, Tags: []string{"latest"}}) + suite.Nil(err, "onPull should return nil") + // async mode should not update db immediately + // pull_time + art, err = artifact.Mgr.Get(suite.ctx, 1) + suite.Nil(err) + suite.Equal(lastPullTime, art.PullTime, "pull_time should not be updated immediately") + // pull_count + repository, err = repo.Mgr.Get(suite.ctx, 1) + suite.Nil(err) + suite.Equal(int64(1), repository.PullCount, "pull_count should not be updated immediately") + // wait for db update + suite.Eventually(func() bool { + art, err = artifact.Mgr.Get(suite.ctx, 1) + suite.Nil(err) + return art.PullTime.After(lastPullTime) + }, 3*asyncFlushDuration, asyncFlushDuration/2, "wait for pull_time async update") + + suite.Eventually(func() bool { + repository, err = repo.Mgr.Get(suite.ctx, 1) + suite.Nil(err) + return int64(2) == repository.PullCount + }, 3*asyncFlushDuration, asyncFlushDuration/2, "wait for pull_count async update") +} diff --git a/src/controller/repository/controller.go b/src/controller/repository/controller.go index 81cb46d89..24a97738e 100644 --- a/src/controller/repository/controller.go +++ b/src/controller/repository/controller.go @@ -52,8 +52,8 @@ type Controller interface { Delete(ctx context.Context, id int64) (err error) // Update the repository. Specify the properties or all properties will be updated Update(ctx context.Context, repository *model.RepoRecord, properties ...string) (err error) - // AddPullCount increase one pull count for the specified repository - AddPullCount(ctx context.Context, id int64) error + // AddPullCount increase pull count for the specified repository + AddPullCount(ctx context.Context, id int64, count uint64) error } // NewController creates an instance of the default repository controller @@ -182,6 +182,6 @@ func (c *controller) Update(ctx context.Context, repository *model.RepoRecord, p return c.repoMgr.Update(ctx, repository, properties...) } -func (c *controller) AddPullCount(ctx context.Context, id int64) error { - return c.repoMgr.AddPullCount(ctx, id) +func (c *controller) AddPullCount(ctx context.Context, id int64, count uint64) error { + return c.repoMgr.AddPullCount(ctx, id, count) } diff --git a/src/controller/repository/controller_test.go b/src/controller/repository/controller_test.go index f5a72deba..5cc58460d 100644 --- a/src/controller/repository/controller_test.go +++ b/src/controller/repository/controller_test.go @@ -143,8 +143,8 @@ func (c *controllerTestSuite) TestUpdate() { } func (c *controllerTestSuite) TestAddPullCount() { - c.repoMgr.On("AddPullCount", mock.Anything, mock.Anything).Return(nil) - err := c.ctl.AddPullCount(nil, 1) + c.repoMgr.On("AddPullCount", mock.Anything, mock.Anything, mock.Anything).Return(nil) + err := c.ctl.AddPullCount(nil, 1, 1) c.Require().Nil(err) } diff --git a/src/pkg/artifact/dao/dao.go b/src/pkg/artifact/dao/dao.go index cc9b034e8..938b38872 100644 --- a/src/pkg/artifact/dao/dao.go +++ b/src/pkg/artifact/dao/dao.go @@ -18,6 +18,7 @@ import ( "context" "fmt" "strings" + "time" beegoorm "github.com/astaxie/beego/orm" "github.com/goharbor/harbor/src/lib/errors" @@ -42,6 +43,8 @@ type DAO interface { Delete(ctx context.Context, id int64) (err error) // Update updates the artifact. Only the properties specified by "props" will be updated if it is set Update(ctx context.Context, artifact *Artifact, props ...string) (err error) + // UpdatePullTime updates artifact pull time by ID. + UpdatePullTime(ctx context.Context, id int64, pullTime time.Time) (err error) // CreateReference creates the artifact reference CreateReference(ctx context.Context, reference *ArtifactReference) (id int64, err error) // ListReferences lists the artifact references according to the query @@ -180,6 +183,7 @@ func (d *dao) Update(ctx context.Context, artifact *Artifact, props ...string) e if err != nil { return err } + n, err := ormer.Update(artifact, props...) if err != nil { return err @@ -189,6 +193,27 @@ func (d *dao) Update(ctx context.Context, artifact *Artifact, props ...string) e } return nil } + +func (d *dao) UpdatePullTime(ctx context.Context, id int64, pullTime time.Time) error { + ormer, err := orm.FromContext(ctx) + if err != nil { + return err + } + + // can only be retained to the second if not format + formatPullTime := pullTime.Format("2006-01-02 15:04:05.999999") + // update db only if pull_time is null or pull_time < (in-coming)pullTime + sql := "UPDATE artifact SET pull_time = ? WHERE id = ? AND (pull_time IS NULL OR pull_time < ?)" + args := []interface{}{formatPullTime, id, formatPullTime} + + _, err = ormer.Raw(sql, args...).Exec() + if err != nil { + return err + } + + return nil +} + func (d *dao) CreateReference(ctx context.Context, reference *ArtifactReference) (int64, error) { ormer, err := orm.FromContext(ctx) if err != nil { diff --git a/src/pkg/artifact/dao/dao_test.go b/src/pkg/artifact/dao/dao_test.go index a6da947ef..c6f70c6f0 100644 --- a/src/pkg/artifact/dao/dao_test.go +++ b/src/pkg/artifact/dao/dao_test.go @@ -16,6 +16,9 @@ package dao import ( "context" + "testing" + "time" + beegoorm "github.com/astaxie/beego/orm" common_dao "github.com/goharbor/harbor/src/common/dao" "github.com/goharbor/harbor/src/lib/errors" @@ -25,8 +28,6 @@ import ( "github.com/goharbor/harbor/src/pkg/tag/model/tag" v1 "github.com/opencontainers/image-spec/specs-go/v1" "github.com/stretchr/testify/suite" - "testing" - "time" ) type daoTestSuite struct { @@ -396,6 +397,32 @@ func (d *daoTestSuite) TestUpdate() { d.True(errors.IsErr(err, errors.NotFoundCode)) } +func (d *daoTestSuite) TestUpdatePullTime() { + artifact, err := d.dao.Get(d.ctx, d.parentArtID) + d.Require().Nil(err) + d.Require().NotNil(artifact) + oldPullTime := artifact.PullTime + // case for pull_time before db pull_time. + before := oldPullTime.AddDate(0, 0, -1) + err = d.dao.UpdatePullTime(d.ctx, d.parentArtID, before) + d.Require().Nil(err) + + artifact, err = d.dao.Get(d.ctx, d.parentArtID) + d.Require().Nil(err) + d.Require().NotNil(artifact) + d.Equal(oldPullTime.Unix(), artifact.PullTime.Unix()) + + // case for pull_time after db pull_time. + after := oldPullTime.AddDate(0, 0, 1) + err = d.dao.UpdatePullTime(d.ctx, d.parentArtID, after) + d.Require().Nil(err) + + artifact, err = d.dao.Get(d.ctx, d.parentArtID) + d.Require().Nil(err) + d.Require().NotNil(artifact) + d.Equal(after.Unix(), artifact.PullTime.Unix()) +} + func (d *daoTestSuite) TestCreateReference() { // happy pass is covered in SetupTest diff --git a/src/pkg/artifact/manager.go b/src/pkg/artifact/manager.go index e45bdf68a..c52df1a6a 100644 --- a/src/pkg/artifact/manager.go +++ b/src/pkg/artifact/manager.go @@ -16,6 +16,8 @@ package artifact import ( "context" + "time" + "github.com/goharbor/harbor/src/lib/q" "github.com/goharbor/harbor/src/pkg/artifact/dao" ) @@ -45,6 +47,8 @@ type Manager interface { Delete(ctx context.Context, id int64) (err error) // Update the artifact. Only the properties specified by "props" will be updated if it is set Update(ctx context.Context, artifact *Artifact, props ...string) (err error) + // UpdatePullTime updates artifact pull time by ID. + UpdatePullTime(ctx context.Context, id int64, pullTime time.Time) (err error) // ListReferences according to the query ListReferences(ctx context.Context, query *q.Query) (references []*Reference, err error) // DeleteReference specified by ID @@ -126,6 +130,10 @@ func (m *manager) Update(ctx context.Context, artifact *Artifact, props ...strin return m.dao.Update(ctx, artifact.To(), props...) } +func (m *manager) UpdatePullTime(ctx context.Context, id int64, pullTime time.Time) (err error) { + return m.dao.UpdatePullTime(ctx, id, pullTime) +} + func (m *manager) ListReferences(ctx context.Context, query *q.Query) ([]*Reference, error) { references, err := m.dao.ListReferences(ctx, query) if err != nil { diff --git a/src/pkg/artifact/manager_test.go b/src/pkg/artifact/manager_test.go index 63e5e5167..5867d7dad 100644 --- a/src/pkg/artifact/manager_test.go +++ b/src/pkg/artifact/manager_test.go @@ -58,6 +58,10 @@ func (f *fakeDao) Update(ctx context.Context, artifact *dao.Artifact, props ...s args := f.Called() return args.Error(0) } +func (f *fakeDao) UpdatePullTime(ctx context.Context, id int64, pullTime time.Time) error { + args := f.Called() + return args.Error(0) +} func (f *fakeDao) CreateReference(ctx context.Context, reference *dao.ArtifactReference) (int64, error) { args := f.Called() return int64(args.Int(0)), args.Error(1) @@ -233,6 +237,13 @@ func (m *managerTestSuite) TestUpdate() { m.dao.AssertExpectations(m.T()) } +func (m *managerTestSuite) TestUpdatePullTime() { + m.dao.On("UpdatePullTime", mock.Anything).Return(nil) + err := m.mgr.UpdatePullTime(nil, 1, time.Now()) + m.Require().Nil(err) + m.dao.AssertExpectations(m.T()) +} + func (m *managerTestSuite) TestListReferences() { m.dao.On("ListReferences").Return([]*dao.ArtifactReference{ { diff --git a/src/pkg/repository/dao/dao.go b/src/pkg/repository/dao/dao.go index 9631a8180..53a07a29e 100644 --- a/src/pkg/repository/dao/dao.go +++ b/src/pkg/repository/dao/dao.go @@ -39,8 +39,8 @@ type DAO interface { Delete(ctx context.Context, id int64) (err error) // Update updates the repository. Only the properties specified by "props" will be updated if it is set Update(ctx context.Context, repository *model.RepoRecord, props ...string) (err error) - // AddPullCount increase one pull count for the specified repository - AddPullCount(ctx context.Context, id int64) error + // AddPullCount increase pull count for the specified repository + AddPullCount(ctx context.Context, id int64, count uint64) error // NonEmptyRepos returns the repositories without any artifact or all the artifacts are untagged. NonEmptyRepos(ctx context.Context) ([]*model.RepoRecord, error) } @@ -132,14 +132,14 @@ func (d *dao) Update(ctx context.Context, repository *model.RepoRecord, props .. return nil } -func (d *dao) AddPullCount(ctx context.Context, id int64) error { +func (d *dao) AddPullCount(ctx context.Context, id int64, count uint64) error { ormer, err := orm.FromContext(ctx) if err != nil { return err } num, err := ormer.QueryTable(new(model.RepoRecord)).Filter("RepositoryID", id).Update( o.Params{ - "pull_count": o.ColValue(o.ColAdd, 1), + "pull_count": o.ColValue(o.ColAdd, count), "update_time": time.Now(), }) if err != nil { diff --git a/src/pkg/repository/dao/dao_test.go b/src/pkg/repository/dao/dao_test.go index 4ae043c6f..1d7280544 100644 --- a/src/pkg/repository/dao/dao_test.go +++ b/src/pkg/repository/dao/dao_test.go @@ -17,6 +17,9 @@ package dao import ( "context" "fmt" + "testing" + "time" + beegoorm "github.com/astaxie/beego/orm" common_dao "github.com/goharbor/harbor/src/common/dao" "github.com/goharbor/harbor/src/lib/errors" @@ -28,8 +31,6 @@ import ( "github.com/goharbor/harbor/src/pkg/tag/model/tag" v1 "github.com/opencontainers/image-spec/specs-go/v1" "github.com/stretchr/testify/suite" - "testing" - "time" ) var ( @@ -179,7 +180,7 @@ func (d *daoTestSuite) TestAddPullCount() { id, err := d.dao.Create(d.ctx, repository) d.Require().Nil(err) - err = d.dao.AddPullCount(d.ctx, id) + err = d.dao.AddPullCount(d.ctx, id, 1) d.Require().Nil(err) repository, err = d.dao.Get(d.ctx, id) diff --git a/src/pkg/repository/manager.go b/src/pkg/repository/manager.go index 221111473..a0932acea 100644 --- a/src/pkg/repository/manager.go +++ b/src/pkg/repository/manager.go @@ -42,8 +42,8 @@ type Manager interface { Delete(ctx context.Context, id int64) (err error) // Update updates the repository. Only the properties specified by "props" will be updated if it is set Update(ctx context.Context, repository *model.RepoRecord, props ...string) (err error) - // AddPullCount increase one pull count for the specified repository - AddPullCount(ctx context.Context, id int64) error + // AddPullCount increase pull count for the specified repository + AddPullCount(ctx context.Context, id int64, count uint64) error // NonEmptyRepos returns the repositories without any artifact or all the artifacts are untagged. NonEmptyRepos(ctx context.Context) ([]*model.RepoRecord, error) } @@ -102,8 +102,8 @@ func (m *manager) Update(ctx context.Context, repository *model.RepoRecord, prop return m.dao.Update(ctx, repository, props...) } -func (m *manager) AddPullCount(ctx context.Context, id int64) error { - return m.dao.AddPullCount(ctx, id) +func (m *manager) AddPullCount(ctx context.Context, id int64, count uint64) error { + return m.dao.AddPullCount(ctx, id, count) } func (m *manager) NonEmptyRepos(ctx context.Context) ([]*model.RepoRecord, error) { diff --git a/src/pkg/repository/manager_test.go b/src/pkg/repository/manager_test.go index b39b00b03..7f38314e5 100644 --- a/src/pkg/repository/manager_test.go +++ b/src/pkg/repository/manager_test.go @@ -108,8 +108,8 @@ func (m *managerTestSuite) TestUpdate() { } func (m *managerTestSuite) TestAddPullCount() { - m.dao.On("AddPullCount", mock.Anything, mock.Anything).Return(nil) - err := m.mgr.AddPullCount(context.Background(), 1) + m.dao.On("AddPullCount", mock.Anything, mock.Anything, mock.Anything).Return(nil) + err := m.mgr.AddPullCount(context.Background(), 1, 1) m.Require().Nil(err) m.dao.AssertExpectations(m.T()) } diff --git a/src/testing/controller/repository/controller.go b/src/testing/controller/repository/controller.go index 48c3ffd06..0ea05f55f 100644 --- a/src/testing/controller/repository/controller.go +++ b/src/testing/controller/repository/controller.go @@ -16,13 +16,13 @@ type Controller struct { mock.Mock } -// AddPullCount provides a mock function with given fields: ctx, id -func (_m *Controller) AddPullCount(ctx context.Context, id int64) error { - ret := _m.Called(ctx, id) +// AddPullCount provides a mock function with given fields: ctx, id, count +func (_m *Controller) AddPullCount(ctx context.Context, id int64, count uint64) error { + ret := _m.Called(ctx, id, count) var r0 error - if rf, ok := ret.Get(0).(func(context.Context, int64) error); ok { - r0 = rf(ctx, id) + if rf, ok := ret.Get(0).(func(context.Context, int64, uint64) error); ok { + r0 = rf(ctx, id, count) } else { r0 = ret.Error(0) } diff --git a/src/testing/pkg/artifact/manager.go b/src/testing/pkg/artifact/manager.go index bcc3c3f33..20d746c31 100644 --- a/src/testing/pkg/artifact/manager.go +++ b/src/testing/pkg/artifact/manager.go @@ -10,6 +10,8 @@ import ( mock "github.com/stretchr/testify/mock" q "github.com/goharbor/harbor/src/lib/q" + + time "time" ) // Manager is an autogenerated mock type for the Manager type @@ -199,3 +201,17 @@ func (_m *Manager) Update(ctx context.Context, _a1 *artifact.Artifact, props ... return r0 } + +// UpdatePullTime provides a mock function with given fields: ctx, id, pullTime +func (_m *Manager) UpdatePullTime(ctx context.Context, id int64, pullTime time.Time) error { + ret := _m.Called(ctx, id, pullTime) + + var r0 error + if rf, ok := ret.Get(0).(func(context.Context, int64, time.Time) error); ok { + r0 = rf(ctx, id, pullTime) + } else { + r0 = ret.Error(0) + } + + return r0 +} diff --git a/src/testing/pkg/repository/dao/dao.go b/src/testing/pkg/repository/dao/dao.go index cef19370c..8d71d966f 100644 --- a/src/testing/pkg/repository/dao/dao.go +++ b/src/testing/pkg/repository/dao/dao.go @@ -17,13 +17,13 @@ type DAO struct { mock.Mock } -// AddPullCount provides a mock function with given fields: ctx, id -func (_m *DAO) AddPullCount(ctx context.Context, id int64) error { - ret := _m.Called(ctx, id) +// AddPullCount provides a mock function with given fields: ctx, id, count +func (_m *DAO) AddPullCount(ctx context.Context, id int64, count uint64) error { + ret := _m.Called(ctx, id, count) var r0 error - if rf, ok := ret.Get(0).(func(context.Context, int64) error); ok { - r0 = rf(ctx, id) + if rf, ok := ret.Get(0).(func(context.Context, int64, uint64) error); ok { + r0 = rf(ctx, id, count) } else { r0 = ret.Error(0) } diff --git a/src/testing/pkg/repository/manager.go b/src/testing/pkg/repository/manager.go index 85086d050..d6e4abac0 100644 --- a/src/testing/pkg/repository/manager.go +++ b/src/testing/pkg/repository/manager.go @@ -16,13 +16,13 @@ type Manager struct { mock.Mock } -// AddPullCount provides a mock function with given fields: ctx, id -func (_m *Manager) AddPullCount(ctx context.Context, id int64) error { - ret := _m.Called(ctx, id) +// AddPullCount provides a mock function with given fields: ctx, id, count +func (_m *Manager) AddPullCount(ctx context.Context, id int64, count uint64) error { + ret := _m.Called(ctx, id, count) var r0 error - if rf, ok := ret.Get(0).(func(context.Context, int64) error); ok { - r0 = rf(ctx, id) + if rf, ok := ret.Get(0).(func(context.Context, int64, uint64) error); ok { + r0 = rf(ctx, id, count) } else { r0 = ret.Error(0) }