Fix a few bugs of replication (#7619)

1. handle the public/private property when creating the projects
2. extend the length of access_secret
3. update the task status by using orm functions

Signed-off-by: Wenkai Yin <yinw@vmware.com>
This commit is contained in:
Wenkai Yin 2019-05-07 10:47:14 +08:00 committed by Wang Yan
parent a58fb7086d
commit d27a6c0335
8 changed files with 119 additions and 61 deletions

View File

@ -59,7 +59,7 @@ ALTER TABLE registry ALTER COLUMN url TYPE varchar(256);
ALTER TABLE registry ADD COLUMN credential_type varchar(16); ALTER TABLE registry ADD COLUMN credential_type varchar(16);
ALTER TABLE registry RENAME COLUMN username TO access_key; ALTER TABLE registry RENAME COLUMN username TO access_key;
ALTER TABLE registry RENAME COLUMN password TO access_secret; ALTER TABLE registry RENAME COLUMN password TO access_secret;
ALTER TABLE registry ALTER COLUMN access_secret TYPE varchar(1024); ALTER TABLE registry ALTER COLUMN access_secret TYPE varchar(4096);
ALTER TABLE registry ADD COLUMN type varchar(32); ALTER TABLE registry ADD COLUMN type varchar(32);
ALTER TABLE registry DROP COLUMN target_type; ALTER TABLE registry DROP COLUMN target_type;
ALTER TABLE registry ADD COLUMN description text; ALTER TABLE registry ADD COLUMN description text;

View File

@ -95,6 +95,7 @@ func (r *ReplicationPolicyAPI) Create() {
return return
} }
policy.Creator = r.SecurityCtx.GetUsername()
id, err := replication.PolicyCtl.Create(policy) id, err := replication.PolicyCtl.Create(policy)
if err != nil { if err != nil {
r.SendInternalServerError(fmt.Errorf("failed to create the policy: %v", err)) r.SendInternalServerError(fmt.Errorf("failed to create the policy: %v", err))

View File

@ -18,6 +18,7 @@ import (
"errors" "errors"
"fmt" "fmt"
"net/http" "net/http"
"strconv"
"strings" "strings"
common_http "github.com/goharbor/harbor/src/common/http" common_http "github.com/goharbor/harbor/src/common/http"
@ -135,11 +136,18 @@ func (a *adapter) PrepareForPush(resources []*model.Resource) error {
if len(resource.Metadata.Repository.Name) == 0 { if len(resource.Metadata.Repository.Name) == 0 {
return errors.New("the name of the repository cannot be null") return errors.New("the name of the repository cannot be null")
} }
paths := strings.Split(resource.Metadata.Repository.Name, "/") paths := strings.Split(resource.Metadata.Repository.Name, "/")
projectName := paths[0] projectName := paths[0]
// TODO handle the public // handle the public properties
metadata := resource.Metadata.Repository.Metadata
pro, exist := projects[projectName]
if exist {
metadata = mergeMetadata(pro.Metadata, metadata)
}
projects[projectName] = &project{ projects[projectName] = &project{
Name: projectName, Name: projectName,
Metadata: metadata,
} }
} }
for _, project := range projects { for _, project := range projects {
@ -161,28 +169,38 @@ func (a *adapter) PrepareForPush(resources []*model.Resource) error {
log.Debugf("project %s created", project.Name) log.Debugf("project %s created", project.Name)
} }
return nil return nil
}
// TODO // currently, mergeMetadata only handles the public metadata
/* func mergeMetadata(metadata1, metadata2 map[string]interface{}) map[string]interface{} {
// handle the public of the project public := parsePublic(metadata1) && parsePublic(metadata2)
if meta, exist := namespace.Metadata["public"]; exist { return map[string]interface{}{
public := true "public": strconv.FormatBool(public),
// if one of them is "private", the set the public as false }
for _, value := range meta.(map[string]interface{}) { }
b, err := strconv.ParseBool(value.(string))
func parsePublic(metadata map[string]interface{}) bool {
if metadata == nil {
return false
}
pub, exist := metadata["public"]
if !exist {
return false
}
public, ok := pub.(bool)
if ok {
return public
}
pubstr, ok := pub.(string)
if ok {
public, err := strconv.ParseBool(pubstr)
if err != nil { if err != nil {
return err log.Errorf("failed to parse %s to bool: %v", pubstr, err)
return false
} }
if !b { return public
public = false
break
} }
} return false
project.Metadata = map[string]interface{}{
"public": public,
}
}
*/
} }
type project struct { type project struct {

View File

@ -16,6 +16,7 @@ package harbor
import ( import (
"net/http" "net/http"
"strconv"
"testing" "testing"
"github.com/goharbor/harbor/src/common/utils/test" "github.com/goharbor/harbor/src/common/utils/test"
@ -152,3 +153,60 @@ func TestPrepareForPush(t *testing.T) {
}) })
require.Nil(t, err) require.Nil(t, err)
} }
func TestParsePublic(t *testing.T) {
cases := []struct {
metadata map[string]interface{}
result bool
}{
{nil, false},
{map[string]interface{}{}, false},
{map[string]interface{}{"public": true}, true},
{map[string]interface{}{"public": "not_bool"}, false},
{map[string]interface{}{"public": "true"}, true},
{map[string]interface{}{"public": struct{}{}}, false},
}
for _, c := range cases {
assert.Equal(t, c.result, parsePublic(c.metadata))
}
}
func TestMergeMetadata(t *testing.T) {
cases := []struct {
m1 map[string]interface{}
m2 map[string]interface{}
public bool
}{
{
m1: map[string]interface{}{
"public": "true",
},
m2: map[string]interface{}{
"public": "true",
},
public: true,
},
{
m1: map[string]interface{}{
"public": "false",
},
m2: map[string]interface{}{
"public": "true",
},
public: false,
},
{
m1: map[string]interface{}{
"public": "false",
},
m2: map[string]interface{}{
"public": "false",
},
public: false,
},
}
for _, c := range cases {
m := mergeMetadata(c.m1, c.m2)
assert.Equal(t, strconv.FormatBool(c.public), m["public"].(string))
}
}

View File

@ -27,8 +27,6 @@ import (
"github.com/goharbor/harbor/src/replication/model" "github.com/goharbor/harbor/src/replication/model"
) )
// TODO review the logic in this file
type chart struct { type chart struct {
Name string `json:"name"` Name string `json:"name"`
Project string Project string
@ -111,7 +109,7 @@ func (a *adapter) FetchCharts(filters []*model.Filter) ([]*model.Resource, error
Metadata: &model.ResourceMetadata{ Metadata: &model.ResourceMetadata{
Repository: &model.Repository{ Repository: &model.Repository{
Name: fmt.Sprintf("%s/%s", project.Name, chart.Name), Name: fmt.Sprintf("%s/%s", project.Name, chart.Name),
// TODO handle the metadata Metadata: project.Metadata,
}, },
Vtags: []string{version.Version}, Vtags: []string{version.Version},
}, },

View File

@ -96,7 +96,7 @@ func (a *adapter) FetchImages(filters []*model.Filter) ([]*model.Resource, error
Metadata: &model.ResourceMetadata{ Metadata: &model.ResourceMetadata{
Repository: &model.Repository{ Repository: &model.Repository{
Name: repository.Name, Name: repository.Name,
// TODO handle the metadata Metadata: project.Metadata,
}, },
Vtags: vtags, Vtags: vtags,
}, },

View File

@ -125,7 +125,7 @@ func fillExecution(execution *models.Execution) error {
} }
if execution.Total != total { if execution.Total != total {
log.Errorf("execution task count inconsistent and fixed, executionID=%d, execution.total=%d, tasks.count=%d", log.Debugf("execution task count inconsistent and fixed, executionID=%d, execution.total=%d, tasks.count=%d",
execution.ID, execution.Total, total) execution.ID, execution.Total, total)
execution.Total = total execution.Total = total
} }
@ -324,38 +324,24 @@ func UpdateTask(task *models.Task, props ...string) (int64, error) {
// UpdateTaskStatus ... // UpdateTaskStatus ...
func UpdateTaskStatus(id int64, status string, statusCondition ...string) (int64, error) { func UpdateTaskStatus(id int64, status string, statusCondition ...string) (int64, error) {
o := dao.GetOrmer() qs := dao.GetOrmer().QueryTable(&models.Task{}).
Filter("id", id)
// update status
params := []interface{}{}
sql := `update replication_task set status = ?`
params = append(params, status)
if taskFinished(status) { // should update endTime
sql += ` ,end_time = ?`
params = append(params, time.Now())
}
sql += ` where id = ?`
params = append(params, id)
if len(statusCondition) > 0 { if len(statusCondition) > 0 {
sql += ` and status in (` qs = qs.Filter("status", statusCondition[0])
for _, stCondition := range statusCondition {
sql += ` ?,`
params = append(params, stCondition)
} }
sql = sql[0 : len(sql)-1] params := orm.Params{
sql += `)` "status": status,
} }
if taskFinished(status) {
log.Infof("Update task %d: -> %s", id, status) // should update endTime
res, err := o.Raw(sql, params).Exec() params["end_time"] = time.Now()
}
n, err := qs.Update(params)
if err != nil { if err != nil {
return 0, err return 0, err
} }
count, err := res.RowsAffected() log.Debugf("update task status %d: -> %s", id, status)
if err != nil { return n, err
return 0, err
}
return count, nil
} }
func taskFinished(status string) bool { func taskFinished(status string) bool {

View File

@ -40,18 +40,15 @@ type Policy struct {
ID int64 `json:"id"` ID int64 `json:"id"`
Name string `json:"name"` Name string `json:"name"`
Description string `json:"description"` Description string `json:"description"`
// TODO consider to remove this property?
Creator string `json:"creator"` Creator string `json:"creator"`
// source // source
SrcRegistry *Registry `json:"src_registry"` SrcRegistry *Registry `json:"src_registry"`
// destination // destination
// TODO rename to DstRegistry
DestRegistry *Registry `json:"dest_registry"` DestRegistry *Registry `json:"dest_registry"`
// Only support two dest namespace modes: // Only support two dest namespace modes:
// Put all the src resources to the one single dest namespace // Put all the src resources to the one single dest namespace
// or keep namespaces same with the source ones (under this case, // or keep namespaces same with the source ones (under this case,
// the DestNamespace should be set to empty) // the DestNamespace should be set to empty)
// TODO rename to DstNamespace
DestNamespace string `json:"dest_namespace"` DestNamespace string `json:"dest_namespace"`
// Filters // Filters
Filters []*Filter `json:"filters"` Filters []*Filter `json:"filters"`