Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(storage): add storage support to application resource #494

Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 11 additions & 9 deletions docs/resources/application.md
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ resource "juju_application" "this" {
}
}

resource "juju_application" "placement_example" {
resource "juju_application" "placement_and_storage_example" {
name = "placement-example"
model = juju_model.development.name
charm {
Expand All @@ -45,6 +45,10 @@ resource "juju_application" "placement_example" {
units = 3
placement = "0,1,2"

storage = {
files = "101M"
}

config = {
external-hostname = "..."
}
Expand Down Expand Up @@ -78,7 +82,8 @@ resource "juju_application" "placement_example" {
latest revision.
* If the charm revision or channel are not updated, then no changes will take
place (juju does not have an "un-attach" command for resources).
- `storage` (Attributes Set) Configure storage constraints for the juju application. (see [below for nested schema](#nestedatt--storage))
- `storage` (Attributes Set) Storage used by the application. (see [below for nested schema](#nestedatt--storage))
- `storage_directives` (Map of String) Storage directives (constraints) for the juju application. The map key is the label of the storage defined by the charm, the map value is the storage directive in the form <pool>,<count>,<size>.
- `trust` (Boolean) Set the trust for the application.
- `units` (Number) The number of application units to deploy for the charm.

Expand Down Expand Up @@ -127,15 +132,12 @@ Optional:
<a id="nestedatt--storage"></a>
### Nested Schema for `storage`

Required:

- `label` (String) The specific storage option defined in the charm.

Optional:
Read-Only:

- `count` (Number) The number of volumes.
- `pool` (String) Name of the storage pool to use. E.g. ebs on aws.
- `size` (String) The size of each volume. E.g. 100G.
- `label` (String) The specific storage option defined in the charm.
- `pool` (String) Name of the storage pool to use.
- `size` (String) The size of each volume.

## Import

Expand Down
6 changes: 5 additions & 1 deletion examples/resources/juju_application/resource.tf
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ resource "juju_application" "this" {
}
}

resource "juju_application" "placement_example" {
resource "juju_application" "placement_and_storage_example" {
anvial marked this conversation as resolved.
Show resolved Hide resolved
name = "placement-example"
model = juju_model.development.name
charm {
Expand All @@ -30,6 +30,10 @@ resource "juju_application" "placement_example" {
units = 3
placement = "0,1,2"

storage = {
files = "101M"
}

config = {
external-hostname = "..."
}
Expand Down
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ require (
)

require (
github.com/dustin/go-humanize v1.0.1
github.com/hashicorp/terraform-plugin-framework v1.9.0
github.com/hashicorp/terraform-plugin-framework-validators v0.12.0
github.com/hashicorp/terraform-plugin-go v0.23.0
Expand Down Expand Up @@ -69,7 +70,6 @@ require (
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
github.com/distribution/reference v0.5.0 // indirect
github.com/docker/distribution v2.8.3+incompatible // indirect
github.com/dustin/go-humanize v1.0.1 // indirect
github.com/emicklei/go-restful/v3 v3.11.0 // indirect
github.com/fatih/color v1.16.0 // indirect
github.com/go-logr/logr v1.4.1 // indirect
Expand Down
143 changes: 120 additions & 23 deletions internal/juju/applications.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ import (
"github.com/juju/juju/core/network"
"github.com/juju/juju/environs/config"
"github.com/juju/juju/rpc/params"
jujustorage "github.com/juju/juju/storage"
jujuversion "github.com/juju/juju/version"
"github.com/juju/names/v5"
"github.com/juju/retry"
Expand All @@ -59,6 +60,17 @@ func (ae *applicationNotFoundError) Error() string {
return fmt.Sprintf("application %s not found", ae.appName)
}

var StorageNotFoundError = &storageNotFoundError{}

// StorageNotFoundError
type storageNotFoundError struct {
storageName string
}

func (se *storageNotFoundError) Error() string {
return fmt.Sprintf("storage %s not found", se.storageName)
}

type applicationsClient struct {
SharedClient
controllerVersion version.Number
Expand Down Expand Up @@ -125,21 +137,22 @@ func ConfigEntryToString(input interface{}) string {
}

type CreateApplicationInput struct {
ApplicationName string
ModelName string
CharmName string
CharmChannel string
CharmBase string
CharmSeries string
CharmRevision int
Units int
Trust bool
Expose map[string]interface{}
Config map[string]string
Placement string
Constraints constraints.Value
EndpointBindings map[string]string
Resources map[string]int
ApplicationName string
ModelName string
CharmName string
CharmChannel string
CharmBase string
CharmSeries string
CharmRevision int
Units int
Trust bool
Expose map[string]interface{}
Config map[string]string
Placement string
Constraints constraints.Value
EndpointBindings map[string]string
Resources map[string]int
StorageConstraints map[string]jujustorage.Constraints
}

// validateAndTransform returns transformedCreateApplicationInput which
Expand All @@ -156,6 +169,7 @@ func (input CreateApplicationInput) validateAndTransform(conn api.Connection) (p
parsed.trust = input.Trust
parsed.units = input.Units
parsed.resources = input.Resources
parsed.storage = input.StorageConstraints

appName := input.ApplicationName
if appName == "" {
Expand Down Expand Up @@ -241,6 +255,7 @@ type transformedCreateApplicationInput struct {
trust bool
endpointBindings map[string]string
resources map[string]int
storage map[string]jujustorage.Constraints
}

type CreateApplicationResponse struct {
Expand All @@ -266,6 +281,7 @@ type ReadApplicationResponse struct {
Principal bool
Placement string
EndpointBindings map[string]string
Storage map[string]jujustorage.Constraints
Resources map[string]int
}

Expand All @@ -282,10 +298,11 @@ type UpdateApplicationInput struct {
Unexpose []string
Config map[string]string
//Series string // Unsupported today
Placement map[string]interface{}
Constraints *constraints.Value
EndpointBindings map[string]string
Resources map[string]int
Placement map[string]interface{}
Constraints *constraints.Value
EndpointBindings map[string]string
StorageConstraints map[string]jujustorage.Constraints
Resources map[string]int
}

type DestroyApplicationInput struct {
Expand Down Expand Up @@ -318,6 +335,9 @@ func (c applicationsClient) CreateApplication(ctx context.Context, input *Create
return nil, err
}

// print transformedInput.storage
c.Tracef("transformedInput.storage", map[string]interface{}{"transformedInput.storage": transformedInput.storage})
anvial marked this conversation as resolved.
Show resolved Hide resolved

applicationAPIClient := apiapplication.NewClient(conn)
if applicationAPIClient.BestAPIVersion() >= 19 {
err = c.deployFromRepository(applicationAPIClient, transformedInput)
Expand Down Expand Up @@ -364,6 +384,7 @@ func (c applicationsClient) deployFromRepository(applicationAPIClient *apiapplic
Revision: &transformedInput.charmRevision,
Trust: transformedInput.trust,
Resources: resources,
Storage: transformedInput.storage,
anvial marked this conversation as resolved.
Show resolved Hide resolved
})
return errors.Join(errs...)
}
Expand Down Expand Up @@ -504,6 +525,7 @@ func (c applicationsClient) legacyDeploy(ctx context.Context, conn api.Connectio
Config: appConfig,
Cons: transformedInput.constraints,
Resources: resources,
Storage: transformedInput.storage,
Placement: transformedInput.placement,
EndpointBindings: transformedInput.endpointBindings,
}
Expand Down Expand Up @@ -734,7 +756,7 @@ func (c applicationsClient) ReadApplicationWithRetryOnNotFound(ctx context.Conte
Func: func() error {
var err error
output, err = c.ReadApplication(input)
if errors.As(err, &ApplicationNotFoundError) {
if errors.As(err, &ApplicationNotFoundError) || errors.As(err, &StorageNotFoundError) {
anvial marked this conversation as resolved.
Show resolved Hide resolved
return err
} else if err != nil {
// Log the error to the terraform Diagnostics to be
Expand All @@ -759,6 +781,18 @@ func (c applicationsClient) ReadApplicationWithRetryOnNotFound(ctx context.Conte
if len(machines) != output.Units {
return fmt.Errorf("ReadApplicationWithRetryOnNotFound: need %d machines, have %d", output.Units, len(machines))
}

// NOTE: Application can always have storage. However, they
anvial marked this conversation as resolved.
Show resolved Hide resolved
// will not be listed right after the application is created. So
// we need to wait for the storages to be ready. And we need to
anvial marked this conversation as resolved.
Show resolved Hide resolved
// check if all storage constraints have pool equal "" and size equal 0
// to drop the error.
for _, storage := range output.Storage {
if storage.Pool == "" || storage.Size == 0 {
return fmt.Errorf("ReadApplicationWithRetryOnNotFound: no storages found in output")
anvial marked this conversation as resolved.
Show resolved Hide resolved
}
}

// NOTE: An IAAS subordinate should also have machines. However, they
// will not be listed until after the relation has been created.
// Those happen with the integration resource which will not be
Expand All @@ -785,6 +819,50 @@ func (c applicationsClient) ReadApplicationWithRetryOnNotFound(ctx context.Conte
return output, retryErr
}

func transformToStorageConstraints(
storageDetailsSlice []params.StorageDetails,
filesystemDetailsSlice []params.FilesystemDetails,
volumeDetailsSlice []params.VolumeDetails,
) map[string]jujustorage.Constraints {
storage := make(map[string]jujustorage.Constraints)
for _, storageDetails := range storageDetailsSlice {
// switch base on storage kind
storageCounters := make(map[string]uint64)
switch storageDetails.Kind.String() {
case "filesystem":
for _, fd := range filesystemDetailsSlice {
if fd.Storage.StorageTag == storageDetails.StorageTag {
// Cut 'storage-' prefix from the storage tag and `-NUMBER` suffix
anvial marked this conversation as resolved.
Show resolved Hide resolved
storageLabel := getStorageLabel(storageDetails)
storageCounters[storageLabel]++
storage[storageLabel] = jujustorage.Constraints{
Pool: fd.Info.Pool,
Size: fd.Info.Size,
Count: storageCounters[storageLabel],
}
}
}
case "block":
for _, vd := range volumeDetailsSlice {
if vd.Storage.StorageTag == storageDetails.StorageTag {
storageLabel := getStorageLabel(storageDetails)
storageCounters[storageLabel]++
storage[storageLabel] = jujustorage.Constraints{
Pool: vd.Info.Pool,
Size: vd.Info.Size,
Count: storageCounters[storageLabel],
}
}
}
}
}
return storage
}

func getStorageLabel(storageDetails params.StorageDetails) string {
anvial marked this conversation as resolved.
Show resolved Hide resolved
return strings.TrimSuffix(strings.TrimPrefix(storageDetails.StorageTag, "storage-"), "-0")
anvial marked this conversation as resolved.
Show resolved Hide resolved
}

func (c applicationsClient) ReadApplication(input *ReadApplicationInput) (*ReadApplicationResponse, error) {
conn, err := c.GetConnection(&input.ModelName)
if err != nil {
Expand Down Expand Up @@ -815,10 +893,20 @@ func (c applicationsClient) ReadApplication(input *ReadApplicationInput) (*ReadA

appInfo := apps[0].Result

// TODO: Investigate why we're getting the full status here when
// application status is needed.
status, err := clientAPIClient.Status(nil)
// We are currently retrieving the full status, which might be more information than necessary.
// The main focus is on the application status, particularly including the storage status.
// It might be more efficient to directly query for the application status and its associated storage status.
// TODO: Investigate if there's a way to optimize this by only fetching the required information.
status, err := clientAPIClient.Status(&apiclient.StatusArgs{
Patterns: []string{},
IncludeStorage: true,
})
if err != nil {
if strings.Contains(err.Error(), "filesystem for storage instance") || strings.Contains(err.Error(), "volume for storage instance") {
// Retry if we get this error. It means the storage is not ready yet.
return nil, &storageNotFoundError{input.AppName}
}
c.Errorf(err, "failed to get status")
return nil, err
}
var appStatus params.ApplicationStatus
Expand All @@ -827,6 +915,12 @@ func (c applicationsClient) ReadApplication(input *ReadApplicationInput) (*ReadA
return nil, fmt.Errorf("no status returned for application: %s", input.AppName)
}

storages := transformToStorageConstraints(status.Storage, status.Filesystems, status.Volumes)
// Print storage to console
for k, v := range storages {
c.Tracef("StorageConstraints constraints", map[string]interface{}{"storage": k, "constraints": v})
}

anvial marked this conversation as resolved.
Show resolved Hide resolved
allocatedMachines := set.NewStrings()
for _, v := range appStatus.Units {
if v.Machine != "" {
Expand Down Expand Up @@ -993,6 +1087,7 @@ func (c applicationsClient) ReadApplication(input *ReadApplicationInput) (*ReadA
Principal: appInfo.Principal,
Placement: placement,
EndpointBindings: endpointBindings,
Storage: storages,
Resources: resourceRevisions,
}

Expand Down Expand Up @@ -1069,6 +1164,8 @@ func (c applicationsClient) UpdateApplication(input *UpdateApplicationInput) err
return err
}

setCharmConfig.StorageConstraints = input.StorageConstraints

err = applicationAPIClient.SetCharm(model.GenerationMaster, *setCharmConfig)
if err != nil {
return err
Expand Down
Loading