mirror of
https://github.com/harness/drone.git
synced 2025-05-04 17:11:31 +08:00
218 lines
5.5 KiB
Go
218 lines
5.5 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.
|
|
|
|
// go:build harness
|
|
|
|
package exporter
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"crypto/tls"
|
|
"encoding/json"
|
|
"fmt"
|
|
"github.com/harness/gitness/internal/api/controller/repo"
|
|
"github.com/harness/gitness/types"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
pathCreateRepo = "/v1/accounts/%s/orgs/%s/projects/%s/repos"
|
|
pathDeleteRepo = "/v1/accounts/%s/orgs/%s/projects/%s/repos/%s"
|
|
headerApiKey = "X-Api-Key"
|
|
routingId = "routingId"
|
|
)
|
|
|
|
var (
|
|
ErrNotFound = fmt.Errorf("not found")
|
|
ErrBadRequest = fmt.Errorf("bad request")
|
|
ErrInternal = fmt.Errorf("internal error")
|
|
ErrDuplicate = fmt.Errorf("resource already exists")
|
|
)
|
|
|
|
type HarnessCodeClient struct {
|
|
client *Client
|
|
}
|
|
|
|
type Client struct {
|
|
baseURL string
|
|
httpClient http.Client
|
|
|
|
accountId string
|
|
orgId string
|
|
projectId string
|
|
|
|
token string
|
|
}
|
|
|
|
// NewClient creates a new harness Client for interacting with the platforms APIs.
|
|
func NewClient(baseURL string, accountID string, orgId string, projectId string, token string) (*Client, error) {
|
|
if baseURL == "" {
|
|
return nil, fmt.Errorf("baseUrl required")
|
|
}
|
|
if accountID == "" {
|
|
return nil, fmt.Errorf("accountID required")
|
|
}
|
|
if orgId == "" {
|
|
return nil, fmt.Errorf("orgId required")
|
|
}
|
|
if projectId == "" {
|
|
return nil, fmt.Errorf("projectId required")
|
|
}
|
|
if token == "" {
|
|
return nil, fmt.Errorf("token required")
|
|
}
|
|
|
|
return &Client{
|
|
baseURL: baseURL,
|
|
accountId: accountID,
|
|
orgId: orgId,
|
|
projectId: projectId,
|
|
token: token,
|
|
httpClient: http.Client{
|
|
Transport: &http.Transport{
|
|
TLSClientConfig: &tls.Config{
|
|
InsecureSkipVerify: false,
|
|
},
|
|
},
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
func NewHarnessCodeClient(baseUrl string, accountID string, orgId string, projectId string, token string) (*HarnessCodeClient, error) {
|
|
client, err := NewClient(baseUrl, accountID, orgId, projectId, token)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &HarnessCodeClient{
|
|
client: client,
|
|
}, nil
|
|
}
|
|
|
|
func (c *HarnessCodeClient) CreateRepo(ctx context.Context, input repo.CreateInput) (*types.Repository, error) {
|
|
path := fmt.Sprintf(pathCreateRepo, c.client.accountId, c.client.orgId, c.client.projectId)
|
|
bodyBytes, err := json.Marshal(input)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to serialize body: %w", err)
|
|
}
|
|
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, appendPath(c.client.baseURL, path), bytes.NewBuffer(bodyBytes))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("unable to create new http request : %w", err)
|
|
}
|
|
|
|
q := map[string]string{routingId: c.client.accountId}
|
|
addQueryParams(req, q)
|
|
req.Header.Add("Content-Type", "application/json")
|
|
req.ContentLength = int64(len(bodyBytes))
|
|
|
|
resp, err := c.client.Do(req)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("request execution failed: %w", err)
|
|
}
|
|
|
|
if resp != nil && resp.Body != nil {
|
|
defer func() { _ = resp.Body.Close() }()
|
|
}
|
|
|
|
repository := new(types.Repository)
|
|
err = mapStatusCodeToError(resp.StatusCode)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = unmarshalResponse(resp, repository)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return repository, err
|
|
}
|
|
|
|
func addQueryParams(req *http.Request, params map[string]string) {
|
|
if len(params) > 0 {
|
|
q := req.URL.Query()
|
|
for key, value := range params {
|
|
q.Add(key, value)
|
|
}
|
|
req.URL.RawQuery = q.Encode()
|
|
}
|
|
}
|
|
|
|
func (c *HarnessCodeClient) DeleteRepo(ctx context.Context, repoUid string) error {
|
|
path := fmt.Sprintf(pathDeleteRepo, c.client.accountId, c.client.orgId, c.client.projectId, repoUid)
|
|
req, err := http.NewRequestWithContext(ctx, http.MethodDelete, appendPath(c.client.baseURL, path), nil)
|
|
if err != nil {
|
|
return fmt.Errorf("unable to create new http request : %w", err)
|
|
}
|
|
|
|
q := map[string]string{routingId: c.client.accountId}
|
|
addQueryParams(req, q)
|
|
resp, err := c.client.Do(req)
|
|
if err != nil {
|
|
return fmt.Errorf("request execution failed: %w", err)
|
|
}
|
|
|
|
if resp != nil && resp.Body != nil {
|
|
defer func() { _ = resp.Body.Close() }()
|
|
}
|
|
return mapStatusCodeToError(resp.StatusCode)
|
|
}
|
|
|
|
func appendPath(uri string, path string) string {
|
|
if path == "" {
|
|
return uri
|
|
}
|
|
|
|
return strings.TrimRight(uri, "/") + "/" + strings.TrimLeft(path, "/")
|
|
}
|
|
|
|
func (c *Client) Do(r *http.Request) (*http.Response, error) {
|
|
addAuthHeader(r, c.token)
|
|
return c.httpClient.Do(r)
|
|
}
|
|
|
|
// addAuthHeader adds the Authorization header to the request.
|
|
func addAuthHeader(req *http.Request, token string) {
|
|
req.Header.Add(headerApiKey, token)
|
|
}
|
|
|
|
func unmarshalResponse(resp *http.Response, data interface{}) error {
|
|
if resp == nil {
|
|
return fmt.Errorf("http response is empty")
|
|
}
|
|
body, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return fmt.Errorf("error reading response body : %w", err)
|
|
}
|
|
err = json.Unmarshal(body, data)
|
|
if err != nil {
|
|
return fmt.Errorf("error deserializing response body : %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func mapStatusCodeToError(statusCode int) error {
|
|
switch {
|
|
case statusCode == 500:
|
|
return ErrInternal
|
|
case statusCode >= 500:
|
|
return fmt.Errorf("received server side error status code %d", statusCode)
|
|
case statusCode == 404:
|
|
return ErrNotFound
|
|
case statusCode == 400:
|
|
return ErrBadRequest
|
|
case statusCode == 409:
|
|
return ErrDuplicate
|
|
case statusCode >= 400:
|
|
return fmt.Errorf("received client side error status code %d", statusCode)
|
|
case statusCode >= 300:
|
|
return fmt.Errorf("received further action required status code %d", statusCode)
|
|
default:
|
|
// TODO: definitely more things to consider here ...
|
|
return nil
|
|
}
|
|
}
|