2017-04-13 12:54:58 +02:00
|
|
|
// Copyright (c) 2017 VMware, Inc. All Rights Reserved.
|
|
|
|
//
|
|
|
|
// 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.
|
2016-06-16 08:57:45 +02:00
|
|
|
|
2016-05-10 13:38:50 +02:00
|
|
|
package models
|
|
|
|
|
|
|
|
import (
|
2017-11-02 05:53:01 +01:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2016-05-10 13:38:50 +02:00
|
|
|
"time"
|
2016-06-06 11:31:47 +02:00
|
|
|
|
|
|
|
"github.com/astaxie/beego/validation"
|
2016-10-19 08:32:00 +02:00
|
|
|
"github.com/vmware/harbor/src/common/utils"
|
2017-11-02 05:53:01 +01:00
|
|
|
"github.com/vmware/harbor/src/replication"
|
2016-05-10 13:38:50 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2016-05-30 08:44:03 +02:00
|
|
|
//RepOpTransfer represents the operation of a job to transfer repository to a remote registry/harbor instance.
|
2016-05-10 13:38:50 +02:00
|
|
|
RepOpTransfer string = "transfer"
|
2016-05-30 08:44:03 +02:00
|
|
|
//RepOpDelete represents the operation of a job to remove repository from a remote registry/harbor instance.
|
|
|
|
RepOpDelete string = "delete"
|
|
|
|
//UISecretCookie is the cookie name to contain the UI secret
|
2017-02-23 11:02:19 +01:00
|
|
|
UISecretCookie string = "secret"
|
2017-05-23 07:33:20 +02:00
|
|
|
//RepTargetTable is the table name for replication targets
|
|
|
|
RepTargetTable = "replication_target"
|
|
|
|
//RepJobTable is the table name for replication jobs
|
|
|
|
RepJobTable = "replication_job"
|
|
|
|
//RepPolicyTable is table name for replication policies
|
|
|
|
RepPolicyTable = "replication_policy"
|
2016-05-10 13:38:50 +02:00
|
|
|
)
|
|
|
|
|
2016-05-30 08:44:03 +02:00
|
|
|
// RepPolicy is the model for a replication policy, which associate to a project and a target (destination)
|
2016-05-10 13:38:50 +02:00
|
|
|
type RepPolicy struct {
|
2017-11-02 05:53:01 +01:00
|
|
|
ID int64 `orm:"pk;auto;column(id)" json:"id"`
|
|
|
|
ProjectID int64 `orm:"column(project_id)" json:"project_id"`
|
|
|
|
ProjectName string `orm:"-" json:"project_name,omitempty"`
|
|
|
|
TargetID int64 `orm:"column(target_id)" json:"target_id"`
|
|
|
|
TargetName string `orm:"-" json:"target_name,omitempty"`
|
|
|
|
Name string `orm:"column(name)" json:"name"`
|
|
|
|
Enabled int `orm:"column(enabled)" json:"enabled"`
|
|
|
|
Description string `orm:"column(description)" json:"description"`
|
|
|
|
Trigger string `orm:"column(cron_str)" json:"trigger"`
|
|
|
|
Filters []*RepFilter `orm:"-" json:"filters"`
|
|
|
|
FiltersInDB string `orm:"column(filters)" json:"-"`
|
|
|
|
ReplicateDeletion bool `orm:"column(replicate_deletion)" json:"replicate_deletion"`
|
|
|
|
StartTime time.Time `orm:"column(start_time)" json:"start_time"`
|
|
|
|
CreationTime time.Time `orm:"column(creation_time);auto_now_add" json:"creation_time"`
|
|
|
|
UpdateTime time.Time `orm:"column(update_time);auto_now" json:"update_time"`
|
|
|
|
ErrorJobCount int `orm:"-" json:"error_job_count"`
|
|
|
|
Deleted int `orm:"column(deleted)" json:"deleted"`
|
2016-05-10 13:38:50 +02:00
|
|
|
}
|
|
|
|
|
2016-06-06 11:36:48 +02:00
|
|
|
// Valid ...
|
2016-06-06 11:31:47 +02:00
|
|
|
func (r *RepPolicy) Valid(v *validation.Validation) {
|
|
|
|
if len(r.Name) == 0 {
|
|
|
|
v.SetError("name", "can not be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(r.Name) > 256 {
|
|
|
|
v.SetError("name", "max length is 256")
|
|
|
|
}
|
|
|
|
|
|
|
|
if r.ProjectID <= 0 {
|
|
|
|
v.SetError("project_id", "invalid")
|
|
|
|
}
|
|
|
|
|
|
|
|
if r.TargetID <= 0 {
|
|
|
|
v.SetError("target_id", "invalid")
|
|
|
|
}
|
|
|
|
|
|
|
|
if r.Enabled != 0 && r.Enabled != 1 {
|
|
|
|
v.SetError("enabled", "must be 0 or 1")
|
|
|
|
}
|
|
|
|
|
2017-11-02 05:53:01 +01:00
|
|
|
if len(r.Trigger) > 256 {
|
|
|
|
v.SetError("trigger", "max length is 256")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, filter := range r.Filters {
|
|
|
|
filter.Valid(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := r.MarshalFilter(); err != nil {
|
|
|
|
v.SetError("filters", err.Error())
|
|
|
|
}
|
|
|
|
if len(r.Filters) > 1024 {
|
|
|
|
v.SetError("filters", "max length is 1024")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalFilter marshal RepFilter array to json string
|
|
|
|
func (r *RepPolicy) MarshalFilter() error {
|
|
|
|
if r.Filters != nil {
|
|
|
|
b, err := json.Marshal(r.Filters)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
r.FiltersInDB = string(b)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalFilter unmarshal json string to RepFilter array
|
|
|
|
func (r *RepPolicy) UnmarshalFilter() error {
|
|
|
|
if len(r.FiltersInDB) > 0 {
|
|
|
|
filter := []*RepFilter{}
|
|
|
|
if err := json.Unmarshal([]byte(r.FiltersInDB), &filter); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
r.Filters = filter
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// RepFilter holds information for the replication policy filter
|
|
|
|
type RepFilter struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Value string `json:"value"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valid ...
|
|
|
|
func (r *RepFilter) Valid(v *validation.Validation) {
|
|
|
|
if !(r.Type == replication.FilterItemKindProject ||
|
|
|
|
r.Type == replication.FilterItemKindRepository ||
|
|
|
|
r.Type == replication.FilterItemKindTag) {
|
|
|
|
v.SetError("filter.type", fmt.Sprintf("invalid filter type: %s", r.Type))
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(r.Value) == 0 {
|
|
|
|
v.SetError("filter.value", "can not be empty")
|
2016-06-06 11:31:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-30 08:44:03 +02:00
|
|
|
// RepJob is the model for a replication job, which is the execution unit on job service, currently it is used to transfer/remove
|
|
|
|
// a repository to/from a remote registry instance.
|
2016-05-10 13:38:50 +02:00
|
|
|
type RepJob struct {
|
2017-04-27 12:19:28 +02:00
|
|
|
ID int64 `orm:"pk;auto;column(id)" json:"id"`
|
2016-05-25 10:33:45 +02:00
|
|
|
Status string `orm:"column(status)" json:"status"`
|
|
|
|
Repository string `orm:"column(repository)" json:"repository"`
|
|
|
|
PolicyID int64 `orm:"column(policy_id)" json:"policy_id"`
|
|
|
|
Operation string `orm:"column(operation)" json:"operation"`
|
|
|
|
Tags string `orm:"column(tags)" json:"-"`
|
|
|
|
TagList []string `orm:"-" json:"tags"`
|
2016-05-17 12:49:02 +02:00
|
|
|
// Policy RepPolicy `orm:"-" json:"policy"`
|
2016-05-10 13:38:50 +02:00
|
|
|
CreationTime time.Time `orm:"column(creation_time);auto_now_add" json:"creation_time"`
|
|
|
|
UpdateTime time.Time `orm:"column(update_time);auto_now" json:"update_time"`
|
|
|
|
}
|
|
|
|
|
2016-05-30 08:44:03 +02:00
|
|
|
// RepTarget is the model for a replication targe, i.e. destination, which wraps the endpoint URL and username/password of a remote registry.
|
2016-05-10 13:38:50 +02:00
|
|
|
type RepTarget struct {
|
2017-04-27 12:19:28 +02:00
|
|
|
ID int64 `orm:"pk;auto;column(id)" json:"id"`
|
2016-06-03 10:16:31 +02:00
|
|
|
URL string `orm:"column(url)" json:"endpoint"`
|
2016-05-10 13:38:50 +02:00
|
|
|
Name string `orm:"column(name)" json:"name"`
|
|
|
|
Username string `orm:"column(username)" json:"username"`
|
|
|
|
Password string `orm:"column(password)" json:"password"`
|
2016-06-17 12:54:29 +02:00
|
|
|
Type int `orm:"column(target_type)" json:"type"`
|
2017-09-22 09:08:10 +02:00
|
|
|
Insecure bool `orm:"column(insecure)" json:"insecure"`
|
2016-05-10 13:38:50 +02:00
|
|
|
CreationTime time.Time `orm:"column(creation_time);auto_now_add" json:"creation_time"`
|
|
|
|
UpdateTime time.Time `orm:"column(update_time);auto_now" json:"update_time"`
|
|
|
|
}
|
|
|
|
|
2016-06-06 11:31:47 +02:00
|
|
|
// Valid ...
|
|
|
|
func (r *RepTarget) Valid(v *validation.Validation) {
|
|
|
|
if len(r.Name) == 0 {
|
|
|
|
v.SetError("name", "can not be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(r.Name) > 64 {
|
|
|
|
v.SetError("name", "max length is 64")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(r.URL) == 0 {
|
|
|
|
v.SetError("endpoint", "can not be empty")
|
|
|
|
}
|
|
|
|
|
2016-06-27 08:37:26 +02:00
|
|
|
r.URL = utils.FormatEndpoint(r.URL)
|
|
|
|
|
2016-06-06 11:31:47 +02:00
|
|
|
if len(r.URL) > 64 {
|
|
|
|
v.SetError("endpoint", "max length is 64")
|
|
|
|
}
|
|
|
|
|
|
|
|
// password is encoded using base64, the length of this field
|
|
|
|
// in DB is 64, so the max length in request is 48
|
|
|
|
if len(r.Password) > 48 {
|
|
|
|
v.SetError("password", "max length is 48")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-30 08:44:03 +02:00
|
|
|
//TableName is required by by beego orm to map RepTarget to table replication_target
|
2016-06-06 11:36:48 +02:00
|
|
|
func (r *RepTarget) TableName() string {
|
2017-05-23 07:33:20 +02:00
|
|
|
return RepTargetTable
|
2016-05-10 13:38:50 +02:00
|
|
|
}
|
|
|
|
|
2016-05-30 08:44:03 +02:00
|
|
|
//TableName is required by by beego orm to map RepJob to table replication_job
|
2016-06-06 11:36:48 +02:00
|
|
|
func (r *RepJob) TableName() string {
|
2017-05-23 07:33:20 +02:00
|
|
|
return RepJobTable
|
2016-05-10 13:38:50 +02:00
|
|
|
}
|
|
|
|
|
2016-05-30 08:44:03 +02:00
|
|
|
//TableName is required by by beego orm to map RepPolicy to table replication_policy
|
2016-06-06 11:36:48 +02:00
|
|
|
func (r *RepPolicy) TableName() string {
|
2017-05-23 07:33:20 +02:00
|
|
|
return RepPolicyTable
|
2016-05-10 13:38:50 +02:00
|
|
|
}
|