mirror of
https://github.com/coder/coder.git
synced 2026-06-06 22:48:19 +00:00
fb84e72319
Adds structured `secret_requirements` to dynamic parameter responses and enforces missing required secrets during workspace start. Stop, delete, and tag rendering paths skip secret requirement enforcement so unmet secrets do not prevent cleanup. The SDK, generated API docs/types, and backend render/resolver/wsbuilder tests are updated for the new contract.
204 lines
6.5 KiB
Go
204 lines
6.5 KiB
Go
package codersdk
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/google/uuid"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/coder/coder/v2/coderd/util/ptr"
|
|
"github.com/coder/coder/v2/coderd/util/slice"
|
|
"github.com/coder/coder/v2/codersdk/wsjson"
|
|
"github.com/coder/websocket"
|
|
)
|
|
|
|
type ParameterFormType string
|
|
|
|
const (
|
|
ParameterFormTypeDefault ParameterFormType = ""
|
|
ParameterFormTypeRadio ParameterFormType = "radio"
|
|
ParameterFormTypeSlider ParameterFormType = "slider"
|
|
ParameterFormTypeInput ParameterFormType = "input"
|
|
ParameterFormTypeDropdown ParameterFormType = "dropdown"
|
|
ParameterFormTypeCheckbox ParameterFormType = "checkbox"
|
|
ParameterFormTypeSwitch ParameterFormType = "switch"
|
|
ParameterFormTypeMultiSelect ParameterFormType = "multi-select"
|
|
ParameterFormTypeTagSelect ParameterFormType = "tag-select"
|
|
ParameterFormTypeTextArea ParameterFormType = "textarea"
|
|
ParameterFormTypeError ParameterFormType = "error"
|
|
)
|
|
|
|
type OptionType string
|
|
|
|
const (
|
|
OptionTypeString OptionType = "string"
|
|
OptionTypeNumber OptionType = "number"
|
|
OptionTypeBoolean OptionType = "bool"
|
|
OptionTypeListString OptionType = "list(string)"
|
|
)
|
|
|
|
type DiagnosticSeverityString string
|
|
|
|
const (
|
|
DiagnosticSeverityError DiagnosticSeverityString = "error"
|
|
DiagnosticSeverityWarning DiagnosticSeverityString = "warning"
|
|
)
|
|
|
|
// FriendlyDiagnostic == previewtypes.FriendlyDiagnostic
|
|
// Copied to avoid import deps
|
|
type FriendlyDiagnostic struct {
|
|
Severity DiagnosticSeverityString `json:"severity"`
|
|
Summary string `json:"summary"`
|
|
Detail string `json:"detail"`
|
|
|
|
Extra DiagnosticExtra `json:"extra"`
|
|
}
|
|
|
|
type DiagnosticExtra struct {
|
|
Code string `json:"code"`
|
|
}
|
|
|
|
// NullHCLString == `previewtypes.NullHCLString`.
|
|
type NullHCLString struct {
|
|
Value string `json:"value"`
|
|
Valid bool `json:"valid"`
|
|
}
|
|
|
|
type PreviewParameter struct {
|
|
PreviewParameterData
|
|
Value NullHCLString `json:"value"`
|
|
Diagnostics []FriendlyDiagnostic `json:"diagnostics"`
|
|
}
|
|
|
|
func (p PreviewParameter) TemplateVersionParameter() TemplateVersionParameter {
|
|
tp := TemplateVersionParameter{
|
|
Name: p.Name,
|
|
DisplayName: p.DisplayName,
|
|
Description: p.Description,
|
|
DescriptionPlaintext: p.Description,
|
|
Type: string(p.Type),
|
|
FormType: string(p.FormType),
|
|
Mutable: p.Mutable,
|
|
DefaultValue: p.DefaultValue.Value,
|
|
Icon: p.Icon,
|
|
Options: slice.List(p.Options, func(o PreviewParameterOption) TemplateVersionParameterOption {
|
|
return o.TemplateVersionParameterOption()
|
|
}),
|
|
Required: p.Required,
|
|
Ephemeral: p.Ephemeral,
|
|
}
|
|
|
|
if len(p.Validations) > 0 {
|
|
valid := p.Validations[0]
|
|
tp.ValidationError = valid.Error
|
|
if valid.Monotonic != nil {
|
|
tp.ValidationMonotonic = ValidationMonotonicOrder(*valid.Monotonic)
|
|
}
|
|
if valid.Regex != nil {
|
|
tp.ValidationRegex = *valid.Regex
|
|
}
|
|
if valid.Min != nil {
|
|
//nolint:gosec
|
|
tp.ValidationMin = ptr.Ref(int32(*valid.Min))
|
|
}
|
|
if valid.Max != nil {
|
|
//nolint:gosec
|
|
tp.ValidationMax = ptr.Ref(int32(*valid.Max))
|
|
}
|
|
}
|
|
return tp
|
|
}
|
|
|
|
func (o PreviewParameterOption) TemplateVersionParameterOption() TemplateVersionParameterOption {
|
|
return TemplateVersionParameterOption{
|
|
Name: o.Name,
|
|
Description: o.Description,
|
|
Value: o.Value.Value,
|
|
Icon: o.Icon,
|
|
}
|
|
}
|
|
|
|
type PreviewParameterData struct {
|
|
Name string `json:"name"`
|
|
DisplayName string `json:"display_name"`
|
|
Description string `json:"description"`
|
|
Type OptionType `json:"type"`
|
|
FormType ParameterFormType `json:"form_type"`
|
|
Styling PreviewParameterStyling `json:"styling"`
|
|
Mutable bool `json:"mutable"`
|
|
DefaultValue NullHCLString `json:"default_value"`
|
|
Icon string `json:"icon"`
|
|
Options []PreviewParameterOption `json:"options"`
|
|
Validations []PreviewParameterValidation `json:"validations"`
|
|
Required bool `json:"required"`
|
|
// legacy_variable_name was removed (= 14)
|
|
Order int64 `json:"order"`
|
|
Ephemeral bool `json:"ephemeral"`
|
|
}
|
|
|
|
type PreviewParameterStyling struct {
|
|
Placeholder *string `json:"placeholder,omitempty"`
|
|
Disabled *bool `json:"disabled,omitempty"`
|
|
Label *string `json:"label,omitempty"`
|
|
MaskInput *bool `json:"mask_input,omitempty"`
|
|
}
|
|
|
|
type PreviewParameterOption struct {
|
|
Name string `json:"name"`
|
|
Description string `json:"description"`
|
|
Value NullHCLString `json:"value"`
|
|
Icon string `json:"icon"`
|
|
}
|
|
|
|
type PreviewParameterValidation struct {
|
|
Error string `json:"validation_error"`
|
|
|
|
// All validation attributes are optional.
|
|
Regex *string `json:"validation_regex"`
|
|
Min *int64 `json:"validation_min"`
|
|
Max *int64 `json:"validation_max"`
|
|
Monotonic *string `json:"validation_monotonic"`
|
|
}
|
|
|
|
type DynamicParametersRequest struct {
|
|
// ID identifies the request. The response contains the same
|
|
// ID so that the client can match it to the request.
|
|
ID int `json:"id"`
|
|
Inputs map[string]string `json:"inputs"`
|
|
// OwnerID if uuid.Nil, it defaults to `codersdk.Me`
|
|
OwnerID uuid.UUID `json:"owner_id,omitempty" format:"uuid"`
|
|
}
|
|
|
|
type SecretRequirementStatus struct {
|
|
Env string `json:"env,omitempty"`
|
|
File string `json:"file,omitempty"`
|
|
HelpMessage string `json:"help_message"`
|
|
Satisfied bool `json:"satisfied"`
|
|
}
|
|
|
|
type DynamicParametersResponse struct {
|
|
ID int `json:"id"`
|
|
Diagnostics []FriendlyDiagnostic `json:"diagnostics"`
|
|
Parameters []PreviewParameter `json:"parameters"`
|
|
SecretRequirements []SecretRequirementStatus `json:"secret_requirements,omitempty"`
|
|
// TODO: Workspace tags
|
|
}
|
|
|
|
func (c *Client) TemplateVersionDynamicParameters(ctx context.Context, userID string, version uuid.UUID) (*wsjson.Stream[DynamicParametersResponse, DynamicParametersRequest], error) {
|
|
endpoint := fmt.Sprintf("/api/v2/templateversions/%s/dynamic-parameters", version)
|
|
if userID != Me {
|
|
uid, err := uuid.Parse(userID)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("invalid user ID: %w", err)
|
|
}
|
|
endpoint += fmt.Sprintf("?user_id=%s", uid.String())
|
|
}
|
|
|
|
conn, err := c.Dial(ctx, endpoint, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return wsjson.NewStream[DynamicParametersResponse, DynamicParametersRequest](conn, websocket.MessageText, websocket.MessageText, c.Logger()), nil
|
|
}
|