mirror of https://github.com/harness/drone.git
301 lines
7.6 KiB
Go
301 lines
7.6 KiB
Go
// Copyright 2023 Harness, Inc.
|
|
//
|
|
// 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 database
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/harness/gitness/app/store"
|
|
gitness_store "github.com/harness/gitness/store"
|
|
"github.com/harness/gitness/store/database"
|
|
"github.com/harness/gitness/store/database/dbtx"
|
|
"github.com/harness/gitness/types"
|
|
"github.com/harness/gitness/types/enum"
|
|
|
|
"github.com/jmoiron/sqlx"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var _ store.TemplateStore = (*templateStore)(nil)
|
|
|
|
const (
|
|
templateQueryBase = `
|
|
SELECT` + templateColumns + `
|
|
FROM templates`
|
|
|
|
templateColumns = `
|
|
template_id,
|
|
template_description,
|
|
template_type,
|
|
template_space_id,
|
|
template_uid,
|
|
template_data,
|
|
template_created,
|
|
template_updated,
|
|
template_version
|
|
`
|
|
)
|
|
|
|
// NewTemplateStore returns a new TemplateStore.
|
|
func NewTemplateStore(db *sqlx.DB) store.TemplateStore {
|
|
return &templateStore{
|
|
db: db,
|
|
}
|
|
}
|
|
|
|
type templateStore struct {
|
|
db *sqlx.DB
|
|
}
|
|
|
|
// Find returns a template given a template ID.
|
|
func (s *templateStore) Find(ctx context.Context, id int64) (*types.Template, error) {
|
|
const findQueryStmt = templateQueryBase + `
|
|
WHERE template_id = $1`
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
dst := new(types.Template)
|
|
if err := db.GetContext(ctx, dst, findQueryStmt, id); err != nil {
|
|
return nil, database.ProcessSQLErrorf(ctx, err, "Failed to find template")
|
|
}
|
|
return dst, nil
|
|
}
|
|
|
|
// FindByIdentifierAndType returns a template in a space with a given identifier and a given type.
|
|
func (s *templateStore) FindByIdentifierAndType(
|
|
ctx context.Context,
|
|
spaceID int64,
|
|
identifier string,
|
|
resolverType enum.ResolverType) (*types.Template, error) {
|
|
const findQueryStmt = templateQueryBase + `
|
|
WHERE template_space_id = $1 AND template_uid = $2 AND template_type = $3`
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
dst := new(types.Template)
|
|
if err := db.GetContext(
|
|
ctx,
|
|
dst,
|
|
findQueryStmt,
|
|
spaceID,
|
|
identifier,
|
|
resolverType.String(),
|
|
); err != nil {
|
|
return nil, database.ProcessSQLErrorf(ctx, err, "Failed to find template")
|
|
}
|
|
return dst, nil
|
|
}
|
|
|
|
// Create creates a template.
|
|
func (s *templateStore) Create(ctx context.Context, template *types.Template) error {
|
|
const templateInsertStmt = `
|
|
INSERT INTO templates (
|
|
template_description,
|
|
template_space_id,
|
|
template_uid,
|
|
template_data,
|
|
template_type,
|
|
template_created,
|
|
template_updated,
|
|
template_version
|
|
) VALUES (
|
|
:template_description,
|
|
:template_space_id,
|
|
:template_uid,
|
|
:template_data,
|
|
:template_type,
|
|
:template_created,
|
|
:template_updated,
|
|
:template_version
|
|
) RETURNING template_id`
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
query, arg, err := db.BindNamed(templateInsertStmt, template)
|
|
if err != nil {
|
|
return database.ProcessSQLErrorf(ctx, err, "Failed to bind template object")
|
|
}
|
|
|
|
if err = db.QueryRowContext(ctx, query, arg...).Scan(&template.ID); err != nil {
|
|
return database.ProcessSQLErrorf(ctx, err, "template query failed")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *templateStore) Update(ctx context.Context, p *types.Template) error {
|
|
const templateUpdateStmt = `
|
|
UPDATE templates
|
|
SET
|
|
template_description = :template_description,
|
|
template_uid = :template_uid,
|
|
template_data = :template_data,
|
|
template_type = :template_type,
|
|
template_updated = :template_updated,
|
|
template_version = :template_version
|
|
WHERE template_id = :template_id AND template_version = :template_version - 1`
|
|
updatedAt := time.Now()
|
|
template := *p
|
|
|
|
template.Version++
|
|
template.Updated = updatedAt.UnixMilli()
|
|
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
query, arg, err := db.BindNamed(templateUpdateStmt, template)
|
|
if err != nil {
|
|
return database.ProcessSQLErrorf(ctx, err, "Failed to bind template object")
|
|
}
|
|
|
|
result, err := db.ExecContext(ctx, query, arg...)
|
|
if err != nil {
|
|
return database.ProcessSQLErrorf(ctx, err, "Failed to update template")
|
|
}
|
|
|
|
count, err := result.RowsAffected()
|
|
if err != nil {
|
|
return database.ProcessSQLErrorf(ctx, err, "Failed to get number of updated rows")
|
|
}
|
|
|
|
if count == 0 {
|
|
return gitness_store.ErrVersionConflict
|
|
}
|
|
|
|
p.Version = template.Version
|
|
p.Updated = template.Updated
|
|
return nil
|
|
}
|
|
|
|
// UpdateOptLock updates the pipeline using the optimistic locking mechanism.
|
|
func (s *templateStore) UpdateOptLock(ctx context.Context,
|
|
template *types.Template,
|
|
mutateFn func(template *types.Template) error,
|
|
) (*types.Template, error) {
|
|
for {
|
|
dup := *template
|
|
|
|
err := mutateFn(&dup)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = s.Update(ctx, &dup)
|
|
if err == nil {
|
|
return &dup, nil
|
|
}
|
|
if !errors.Is(err, gitness_store.ErrVersionConflict) {
|
|
return nil, err
|
|
}
|
|
|
|
template, err = s.Find(ctx, template.ID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
|
|
// List lists all the templates present in a space.
|
|
func (s *templateStore) List(
|
|
ctx context.Context,
|
|
parentID int64,
|
|
filter types.ListQueryFilter,
|
|
) ([]*types.Template, error) {
|
|
stmt := database.Builder.
|
|
Select(templateColumns).
|
|
From("templates").
|
|
Where("template_space_id = ?", fmt.Sprint(parentID))
|
|
|
|
if filter.Query != "" {
|
|
stmt = stmt.Where(PartialMatch("template_uid", filter.Query))
|
|
}
|
|
|
|
stmt = stmt.Limit(database.Limit(filter.Size))
|
|
stmt = stmt.Offset(database.Offset(filter.Page, filter.Size))
|
|
|
|
sql, args, err := stmt.ToSql()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "Failed to convert query to sql")
|
|
}
|
|
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
dst := []*types.Template{}
|
|
if err = db.SelectContext(ctx, &dst, sql, args...); err != nil {
|
|
return nil, database.ProcessSQLErrorf(ctx, err, "Failed executing custom list query")
|
|
}
|
|
|
|
return dst, nil
|
|
}
|
|
|
|
// Delete deletes a template given a template ID.
|
|
func (s *templateStore) Delete(ctx context.Context, id int64) error {
|
|
const templateDeleteStmt = `
|
|
DELETE FROM templates
|
|
WHERE template_id = $1`
|
|
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
if _, err := db.ExecContext(ctx, templateDeleteStmt, id); err != nil {
|
|
return database.ProcessSQLErrorf(ctx, err, "Could not delete template")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// DeleteByIdentifierAndType deletes a template with a given identifier in a space.
|
|
func (s *templateStore) DeleteByIdentifierAndType(
|
|
ctx context.Context,
|
|
spaceID int64,
|
|
identifier string,
|
|
resolverType enum.ResolverType,
|
|
) error {
|
|
const templateDeleteStmt = `
|
|
DELETE FROM templates
|
|
WHERE template_space_id = $1 AND template_uid = $2 AND template_type = $3`
|
|
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
if _, err := db.ExecContext(ctx, templateDeleteStmt, spaceID, identifier, resolverType.String()); err != nil {
|
|
return database.ProcessSQLErrorf(ctx, err, "Could not delete template")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Count of templates in a space.
|
|
func (s *templateStore) Count(ctx context.Context, parentID int64, filter types.ListQueryFilter) (int64, error) {
|
|
stmt := database.Builder.
|
|
Select("count(*)").
|
|
From("templates").
|
|
Where("template_space_id = ?", parentID)
|
|
|
|
if filter.Query != "" {
|
|
stmt = stmt.Where(PartialMatch("template_uid", filter.Query))
|
|
}
|
|
|
|
sql, args, err := stmt.ToSql()
|
|
if err != nil {
|
|
return 0, errors.Wrap(err, "Failed to convert query to sql")
|
|
}
|
|
|
|
db := dbtx.GetAccessor(ctx, s.db)
|
|
|
|
var count int64
|
|
err = db.QueryRowContext(ctx, sql, args...).Scan(&count)
|
|
if err != nil {
|
|
return 0, database.ProcessSQLErrorf(ctx, err, "Failed executing count query")
|
|
}
|
|
return count, nil
|
|
}
|