mirror of https://github.com/harness/drone.git
341 lines
9.7 KiB
Go
341 lines
9.7 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 types
|
|
|
|
import (
|
|
"encoding/csv"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/url"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/harness/gitness/types/enum"
|
|
|
|
"oras.land/oras-go/v2/registry"
|
|
)
|
|
|
|
const FeatureDefaultTag = "latest"
|
|
|
|
//nolint:tagliatelle
|
|
type DevcontainerConfig struct {
|
|
Image string `json:"image,omitempty"`
|
|
PostCreateCommand LifecycleCommand `json:"postCreateCommand,omitempty"`
|
|
PostStartCommand LifecycleCommand `json:"postStartCommand,omitempty"`
|
|
ForwardPorts []json.Number `json:"forwardPorts,omitempty"`
|
|
ContainerEnv map[string]string `json:"containerEnv,omitempty"`
|
|
Customizations DevContainerConfigCustomizations `json:"customizations,omitempty"`
|
|
RunArgs []string `json:"runArgs,omitempty"`
|
|
ContainerUser string `json:"containerUser,omitempty"`
|
|
RemoteUser string `json:"remoteUser,omitempty"`
|
|
Features *Features `json:"features,omitempty"`
|
|
OverrideFeatureInstallOrder []string `json:"overrideFeatureInstallOrder,omitempty"`
|
|
Privileged *bool `json:"privileged,omitempty"`
|
|
Init *bool `json:"init,omitempty"`
|
|
CapAdd []string `json:"capAdd,omitempty"`
|
|
SecurityOpt []string `json:"securityOpt,omitempty"`
|
|
Mounts []*Mount `json:"mounts,omitempty"`
|
|
}
|
|
|
|
// Constants for discriminator values.
|
|
const (
|
|
TypeString = "string"
|
|
TypeArray = "array"
|
|
TypeCommandMap = "commandMap"
|
|
)
|
|
|
|
//nolint:tagliatelle
|
|
type LifecycleCommand struct {
|
|
CommandString string `json:"commandString,omitempty"`
|
|
CommandArray []string `json:"commandArray,omitempty"`
|
|
CommandMap map[string]any `json:"commandMap,omitempty"`
|
|
Discriminator string `json:"-"` // Tracks the original type for proper re-marshaling
|
|
}
|
|
|
|
func (lc *LifecycleCommand) UnmarshalJSON(data []byte) error {
|
|
// Try to unmarshal as a single string
|
|
var commandStr string
|
|
if err := json.Unmarshal(data, &commandStr); err == nil {
|
|
lc.CommandString = commandStr
|
|
lc.Discriminator = TypeString
|
|
return nil
|
|
}
|
|
|
|
// Try to unmarshal as an array of strings
|
|
var commandArr []string
|
|
if err := json.Unmarshal(data, &commandArr); err == nil {
|
|
lc.CommandArray = commandArr
|
|
lc.Discriminator = TypeArray
|
|
return nil
|
|
}
|
|
|
|
// Try to unmarshal as a map with mixed types
|
|
var rawMap map[string]any
|
|
if err := json.Unmarshal(data, &rawMap); err == nil {
|
|
for key, value := range rawMap {
|
|
switch v := value.(type) {
|
|
case string:
|
|
// Valid string value
|
|
case []interface{}:
|
|
// Convert []interface{} to []string
|
|
var strArray []string
|
|
for _, item := range v {
|
|
if str, ok := item.(string); ok {
|
|
strArray = append(strArray, str)
|
|
} else {
|
|
return fmt.Errorf("invalid format: array contains non-string value")
|
|
}
|
|
}
|
|
rawMap[key] = strArray
|
|
default:
|
|
return fmt.Errorf("invalid format: map contains unsupported type")
|
|
}
|
|
}
|
|
lc.CommandMap = rawMap
|
|
lc.Discriminator = TypeCommandMap
|
|
return nil
|
|
}
|
|
|
|
return fmt.Errorf("invalid format: must be string, []string, or map[string]any")
|
|
}
|
|
|
|
func (lc *LifecycleCommand) MarshalJSON() ([]byte, error) {
|
|
// If Discriminator is empty, return an empty JSON object (i.e., {} or no content)
|
|
if lc.Discriminator == "" || lc == nil {
|
|
return []byte("{}"), nil
|
|
}
|
|
switch lc.Discriminator {
|
|
case TypeString:
|
|
return json.Marshal(lc.CommandString)
|
|
case TypeArray:
|
|
return json.Marshal(lc.CommandArray)
|
|
case TypeCommandMap:
|
|
return json.Marshal(lc.CommandMap)
|
|
default:
|
|
return nil, fmt.Errorf("unknown type for LifecycleCommand")
|
|
}
|
|
}
|
|
|
|
// ToCommandArray converts the LifecycleCommand into a slice of full commands.
|
|
func (lc *LifecycleCommand) ToCommandArray() []string {
|
|
// If Discriminator is empty, return nil
|
|
if lc.Discriminator == "" || lc == nil {
|
|
return nil
|
|
}
|
|
switch lc.Discriminator {
|
|
case TypeString:
|
|
return []string{lc.CommandString}
|
|
case TypeArray:
|
|
return []string{strings.Join(lc.CommandArray, " ")}
|
|
case TypeCommandMap:
|
|
var commands []string
|
|
for _, value := range lc.CommandMap {
|
|
switch v := value.(type) {
|
|
case string:
|
|
commands = append(commands, v)
|
|
case []string:
|
|
commands = append(commands, strings.Join(v, " "))
|
|
}
|
|
}
|
|
return commands
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
type Features map[string]*FeatureValue
|
|
|
|
type FeatureValue struct {
|
|
Source string `json:"source,omitempty"`
|
|
SourceType enum.FeatureSourceType `json:"source_type,omitempty"`
|
|
Options map[string]any `json:"options,omitempty"`
|
|
}
|
|
|
|
func (f *FeatureValue) UnmarshalJSON(data []byte) error {
|
|
var version string
|
|
if err := json.Unmarshal(data, &version); err == nil {
|
|
f.Options = make(map[string]any)
|
|
f.Options["version"] = version
|
|
return nil
|
|
}
|
|
|
|
var options map[string]any
|
|
if err := json.Unmarshal(data, &options); err == nil {
|
|
for key, value := range options {
|
|
switch value.(type) {
|
|
case string, bool:
|
|
continue
|
|
default:
|
|
return fmt.Errorf("invalid type for option '%s': must be string or boolean, got %T", key, value)
|
|
}
|
|
}
|
|
f.Options = options
|
|
return nil
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (f *FeatureValue) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(f.Options)
|
|
}
|
|
|
|
func (f *Features) UnmarshalJSON(data []byte) error {
|
|
if *f == nil {
|
|
*f = make(Features)
|
|
}
|
|
|
|
var raw map[string]json.RawMessage
|
|
if err := json.Unmarshal(data, &raw); err != nil {
|
|
return err
|
|
}
|
|
|
|
for key, value := range raw {
|
|
sanitizedSource, sourceType, validationErr := validateFeatureSource(key)
|
|
if validationErr != nil {
|
|
return validationErr
|
|
}
|
|
feature := &FeatureValue{Source: sanitizedSource, SourceType: sourceType}
|
|
if err := json.Unmarshal(value, feature); err != nil {
|
|
return fmt.Errorf("failed to unmarshal feature '%s': %w", key, err)
|
|
}
|
|
(*f)[sanitizedSource] = feature
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func validateFeatureSource(source string) (string, enum.FeatureSourceType, error) {
|
|
if _, err := registry.ParseReference(source); err == nil {
|
|
indexOfSeparator := strings.Index(source, ":")
|
|
if indexOfSeparator == -1 {
|
|
source += ":" + FeatureDefaultTag
|
|
}
|
|
return source, enum.FeatureSourceTypeOCI, nil
|
|
}
|
|
if err := validateTarballURL(source); err == nil {
|
|
return source, enum.FeatureSourceTypeTarball, nil
|
|
}
|
|
return source, enum.FeatureSourceTypeLocal, fmt.Errorf("unsupported feature source: %s", source)
|
|
}
|
|
|
|
func validateTarballURL(source string) error {
|
|
tarballURL, err := url.Parse(source)
|
|
if err != nil {
|
|
return fmt.Errorf("parsing feature URL: %w", err)
|
|
}
|
|
if tarballURL.Scheme != "http" && tarballURL.Scheme != "https" {
|
|
return fmt.Errorf("invalid feature URL: %s", tarballURL.String())
|
|
}
|
|
if !strings.HasSuffix(tarballURL.Path, ".tgz") {
|
|
return fmt.Errorf("invalid feature URL: %s", tarballURL.String())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type Mount struct {
|
|
Source string `json:"source,omitempty"`
|
|
Target string `json:"target,omitempty"`
|
|
Type string `json:"type,omitempty"`
|
|
}
|
|
|
|
func (m *Mount) UnmarshalJSON(data []byte) error {
|
|
type Alias Mount
|
|
aux := &struct {
|
|
*Alias
|
|
}{
|
|
Alias: (*Alias)(m),
|
|
}
|
|
if err := json.Unmarshal(data, &aux); err == nil {
|
|
return nil
|
|
}
|
|
|
|
var str string
|
|
if err := json.Unmarshal(data, &str); err == nil {
|
|
dst, err := stringToObject(str)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
*m = *dst
|
|
return nil
|
|
}
|
|
|
|
return fmt.Errorf("failed to unmarshal JSON: %s", string(data))
|
|
}
|
|
|
|
func ParseMountsFromRawSlice(values []any) ([]*Mount, error) {
|
|
var mounts []*Mount
|
|
for _, value := range values {
|
|
if mountValue, isObject := value.(*Mount); isObject {
|
|
mounts = append(mounts, mountValue)
|
|
} else if strVal, isString := value.(string); isString {
|
|
dst, err := stringToObject(strVal)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
mounts = append(mounts, dst)
|
|
} else {
|
|
return nil, fmt.Errorf("invalid mount value: %+v", value)
|
|
}
|
|
}
|
|
return mounts, nil
|
|
}
|
|
|
|
func ParseMountsFromStringSlice(values []string) ([]*Mount, error) {
|
|
var mounts []*Mount
|
|
for _, value := range values {
|
|
dst, err := stringToObject(value)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
mounts = append(mounts, dst)
|
|
}
|
|
return mounts, nil
|
|
}
|
|
|
|
func stringToObject(mountStr string) (*Mount, error) {
|
|
csvReader := csv.NewReader(strings.NewReader(mountStr))
|
|
fields, err := csvReader.Read()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
newMount := Mount{Type: "volume"}
|
|
for _, field := range fields {
|
|
key, val, ok := strings.Cut(field, "=")
|
|
|
|
key = strings.ToLower(key)
|
|
|
|
if !ok {
|
|
return nil, fmt.Errorf("invalid format for mount field: %s", field)
|
|
}
|
|
|
|
switch key {
|
|
case "type":
|
|
newMount.Type = strings.ToLower(val)
|
|
case "source", "src":
|
|
newMount.Source = val
|
|
if strings.HasPrefix(val, "."+string(filepath.Separator)) || val == "." {
|
|
if abs, err := filepath.Abs(val); err == nil {
|
|
newMount.Source = abs
|
|
}
|
|
}
|
|
case "target", "dst", "destination":
|
|
newMount.Target = val
|
|
}
|
|
}
|
|
return &newMount, nil
|
|
}
|