drone/internal/api/controller/repo/create.go

158 lines
3.9 KiB
Go

// Copyright 2022 Harness Inc. All rights reserved.
// Use of this source code is governed by the Polyform Free Trial License
// that can be found in the LICENSE.md file for this repository.
package repo
import (
"bytes"
"context"
"fmt"
"time"
"github.com/harness/gitness/gitrpc"
apiauth "github.com/harness/gitness/internal/api/auth"
"github.com/harness/gitness/internal/api/usererror"
"github.com/harness/gitness/internal/auth"
"github.com/harness/gitness/resources"
"github.com/harness/gitness/types"
"github.com/harness/gitness/types/enum"
zerolog "github.com/rs/zerolog/log"
)
type CreateInput struct {
ParentID int64 `json:"parentID"`
UID string `json:"uid"`
DefaultBranch string `json:"defaultBranch"`
Description string `json:"description"`
IsPublic bool `json:"isPublic"`
ForkID int64 `json:"forkId"`
Readme bool `json:"readme"`
License string `json:"license"`
GitIgnore string `json:"gitIgnore"`
}
// Create creates a new repository.
//
//nolint:funlen // needs refactor
func (c *Controller) Create(ctx context.Context, session *auth.Session, in *CreateInput) (*types.Repository, error) {
log := zerolog.Ctx(ctx)
// ensure we reference a space
if in.ParentID <= 0 {
return nil, usererror.BadRequest("A repository can't exist by itself.")
}
parentSpace, err := c.spaceStore.Find(ctx, in.ParentID)
if err != nil {
log.Err(err).Msgf("Failed to get space with id '%d'.", in.ParentID)
return nil, usererror.BadRequest("Parent not found'")
}
/*
* AUTHORIZATION
* Create is a special case - check permission without specific resource
*/
scope := &types.Scope{SpacePath: parentSpace.Path}
resource := &types.Resource{
Type: enum.ResourceTypeRepo,
Name: "",
}
err = apiauth.Check(ctx, c.authorizer, session, scope, resource, enum.PermissionRepoCreate)
if err != nil {
return nil, fmt.Errorf("auth check failed: %w", err)
}
// set default branch in case it wasn't passed
if in.DefaultBranch == "" {
in.DefaultBranch = c.defaultBranch
}
// create new repo object
repo := &types.Repository{
ParentID: in.ParentID,
UID: in.UID,
Description: in.Description,
IsPublic: in.IsPublic,
CreatedBy: session.Principal.ID,
Created: time.Now().UnixMilli(),
Updated: time.Now().UnixMilli(),
ForkID: in.ForkID,
DefaultBranch: in.DefaultBranch,
}
// validate repo
if err = c.repoCheck(repo); err != nil {
return nil, err
}
var content []byte
files := make([]gitrpc.File, 0, 3) // readme, gitignore, licence
if in.Readme {
content = createReadme(in.UID, in.Description)
files = append(files, gitrpc.File{
Path: "README.md",
Content: content,
})
}
if in.License != "" && in.License != "none" {
content, err = resources.ReadLicense(in.License)
if err != nil {
return nil, err
}
files = append(files, gitrpc.File{
Path: "LICENSE",
Content: content,
})
}
if in.GitIgnore != "" {
content, err = resources.ReadGitIgnore(in.GitIgnore)
if err != nil {
return nil, err
}
files = append(files, gitrpc.File{
Path: ".gitignore",
Content: content,
})
}
resp, err := c.gitRPCClient.CreateRepository(ctx, &gitrpc.CreateRepositoryParams{
DefaultBranch: repo.DefaultBranch,
Files: files,
})
if err != nil {
return nil, fmt.Errorf("error creating repository: %w", err)
}
repo.GitUID = resp.UID
// create in store
err = c.repoStore.Create(ctx, repo)
if err != nil {
log.Error().Err(err).
Msg("Repository creation failed.")
// TODO: cleanup git repo!
return nil, err
}
// populate repo url
repo.GitURL, err = GenerateRepoGitURL(c.gitBaseURL, repo.Path)
if err != nil {
return nil, err
}
return repo, nil
}
func createReadme(name, description string) []byte {
content := bytes.Buffer{}
content.WriteString("#" + name + "\n")
if description != "" {
content.WriteString(description)
}
return content.Bytes()
}