Skip to content

Commit 38fd510

Browse files
authored
Add fsx ontap_volume resource (#21889)
* new volume resource * changelog and formatting * main merge * clear lint * fixed tests
1 parent 1e039f6 commit 38fd510

File tree

8 files changed

+1177
-0
lines changed

8 files changed

+1177
-0
lines changed

.changelog/21889.txt

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
```release-note:new-resource
2+
aws_fsx_ontap_volume
3+
```

internal/provider/provider.go

+1
Original file line numberDiff line numberDiff line change
@@ -1191,6 +1191,7 @@ func Provider() *schema.Provider {
11911191
"aws_fsx_lustre_file_system": fsx.ResourceLustreFileSystem(),
11921192
"aws_fsx_ontap_file_system": fsx.ResourceOntapFileSystem(),
11931193
"aws_fsx_ontap_storage_virtual_machine": fsx.ResourceOntapStorageVirtualMachine(),
1194+
"aws_fsx_ontap_volume": fsx.ResourceOntapVolume(),
11941195
"aws_fsx_windows_file_system": fsx.ResourceWindowsFileSystem(),
11951196

11961197
"aws_gamelift_alias": gamelift.ResourceAlias(),

internal/service/fsx/find.go

+39
Original file line numberDiff line numberDiff line change
@@ -131,3 +131,42 @@ func FindStorageVirtualMachineByID(conn *fsx.FSx, id string) (*fsx.StorageVirtua
131131

132132
return storageVirtualMachines[0], nil
133133
}
134+
135+
func FindVolumeByID(conn *fsx.FSx, id string) (*fsx.Volume, error) {
136+
input := &fsx.DescribeVolumesInput{
137+
VolumeIds: []*string{aws.String(id)},
138+
}
139+
140+
var volumes []*fsx.Volume
141+
142+
err := conn.DescribeVolumesPages(input, func(page *fsx.DescribeVolumesOutput, lastPage bool) bool {
143+
if page == nil {
144+
return !lastPage
145+
}
146+
147+
volumes = append(volumes, page.Volumes...)
148+
149+
return !lastPage
150+
})
151+
152+
if tfawserr.ErrCodeEquals(err, fsx.ErrCodeVolumeNotFound) {
153+
return nil, &resource.NotFoundError{
154+
LastError: err,
155+
LastRequest: input,
156+
}
157+
}
158+
159+
if err != nil {
160+
return nil, err
161+
}
162+
163+
if len(volumes) == 0 || volumes[0] == nil {
164+
return nil, tfresource.NewEmptyResultError(input)
165+
}
166+
167+
if count := len(volumes); count > 1 {
168+
return nil, tfresource.NewTooManyResultsError(count, input)
169+
}
170+
171+
return volumes[0], nil
172+
}

internal/service/fsx/ontap_volume.go

+340
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,340 @@
1+
package fsx
2+
3+
import (
4+
"fmt"
5+
"log"
6+
"time"
7+
8+
"github.com/aws/aws-sdk-go/aws"
9+
"github.com/aws/aws-sdk-go/service/fsx"
10+
"github.com/hashicorp/aws-sdk-go-base/tfawserr"
11+
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/resource"
12+
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
13+
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/validation"
14+
"github.com/hashicorp/terraform-provider-aws/internal/conns"
15+
tftags "github.com/hashicorp/terraform-provider-aws/internal/tags"
16+
"github.com/hashicorp/terraform-provider-aws/internal/tfresource"
17+
"github.com/hashicorp/terraform-provider-aws/internal/verify"
18+
)
19+
20+
func ResourceOntapVolume() *schema.Resource {
21+
return &schema.Resource{
22+
Create: resourceOntapVolumeCreate,
23+
Read: resourceOntapVolumeRead,
24+
Update: resourceOntapVolumeUpdate,
25+
Delete: resourceOntapVolumeDelete,
26+
Importer: &schema.ResourceImporter{
27+
State: schema.ImportStatePassthrough,
28+
},
29+
30+
Timeouts: &schema.ResourceTimeout{
31+
Create: schema.DefaultTimeout(30 * time.Minute),
32+
Update: schema.DefaultTimeout(30 * time.Minute),
33+
Delete: schema.DefaultTimeout(30 * time.Minute),
34+
},
35+
36+
Schema: map[string]*schema.Schema{
37+
"arn": {
38+
Type: schema.TypeString,
39+
Computed: true,
40+
},
41+
"file_system_id": {
42+
Type: schema.TypeString,
43+
Computed: true,
44+
},
45+
"flexcache_endpoint_type": {
46+
Type: schema.TypeString,
47+
Computed: true,
48+
},
49+
"junction_path": {
50+
Type: schema.TypeString,
51+
Required: true,
52+
ValidateFunc: validation.StringLenBetween(1, 255),
53+
},
54+
"name": {
55+
Type: schema.TypeString,
56+
Required: true,
57+
ForceNew: true,
58+
ValidateFunc: validation.StringLenBetween(1, 203),
59+
},
60+
"ontap_volume_type": {
61+
Type: schema.TypeString,
62+
Computed: true,
63+
},
64+
"security_style": {
65+
Type: schema.TypeString,
66+
Optional: true,
67+
Default: "UNIX",
68+
ValidateFunc: validation.StringInSlice(fsx.StorageVirtualMachineRootVolumeSecurityStyle_Values(), false),
69+
},
70+
"size_in_megabytes": {
71+
Type: schema.TypeInt,
72+
Required: true,
73+
ValidateFunc: validation.IntBetween(0, 2147483647),
74+
},
75+
"storage_efficiency_enabled": {
76+
Type: schema.TypeBool,
77+
Required: true,
78+
},
79+
"storage_virtual_machine_id": {
80+
Type: schema.TypeString,
81+
Required: true,
82+
ValidateFunc: validation.StringLenBetween(21, 21),
83+
},
84+
"tiering_policy": {
85+
Type: schema.TypeList,
86+
Optional: true,
87+
DiffSuppressFunc: verify.SuppressMissingOptionalConfigurationBlock,
88+
MaxItems: 1,
89+
Elem: &schema.Resource{
90+
Schema: map[string]*schema.Schema{
91+
"cooling_period": {
92+
Type: schema.TypeInt,
93+
Optional: true,
94+
ValidateFunc: validation.IntBetween(2, 183),
95+
},
96+
"name": {
97+
Type: schema.TypeString,
98+
Optional: true,
99+
Computed: true,
100+
ValidateFunc: validation.StringInSlice(fsx.TieringPolicyName_Values(), false),
101+
},
102+
},
103+
},
104+
},
105+
"tags": tftags.TagsSchema(),
106+
"tags_all": tftags.TagsSchemaComputed(),
107+
"uuid": {
108+
Type: schema.TypeString,
109+
Computed: true,
110+
},
111+
"volume_type": {
112+
Type: schema.TypeString,
113+
Default: fsx.VolumeTypeOntap,
114+
Optional: true,
115+
ValidateFunc: validation.StringInSlice(fsx.VolumeType_Values(), false),
116+
},
117+
},
118+
CustomizeDiff: verify.SetTagsDiff,
119+
}
120+
}
121+
122+
func resourceOntapVolumeCreate(d *schema.ResourceData, meta interface{}) error {
123+
conn := meta.(*conns.AWSClient).FSxConn
124+
defaultTagsConfig := meta.(*conns.AWSClient).DefaultTagsConfig
125+
tags := defaultTagsConfig.MergeTags(tftags.New(d.Get("tags").(map[string]interface{})))
126+
127+
input := &fsx.CreateVolumeInput{
128+
Name: aws.String(d.Get("name").(string)),
129+
VolumeType: aws.String(d.Get("volume_type").(string)),
130+
OntapConfiguration: &fsx.CreateOntapVolumeConfiguration{
131+
JunctionPath: aws.String(d.Get("junction_path").(string)),
132+
SizeInMegabytes: aws.Int64(int64(d.Get("size_in_megabytes").(int))),
133+
StorageEfficiencyEnabled: aws.Bool(d.Get("storage_efficiency_enabled").(bool)),
134+
StorageVirtualMachineId: aws.String(d.Get("storage_virtual_machine_id").(string)),
135+
},
136+
}
137+
138+
if v, ok := d.GetOk("security_style"); ok {
139+
input.OntapConfiguration.SecurityStyle = aws.String(v.(string))
140+
}
141+
142+
if v, ok := d.GetOk("tiering_policy"); ok {
143+
input.OntapConfiguration.TieringPolicy = expandFsxOntapVolumeTieringPolicy(v.([]interface{}))
144+
}
145+
146+
if len(tags) > 0 {
147+
input.Tags = Tags(tags.IgnoreAWS())
148+
}
149+
150+
log.Printf("[DEBUG] Creating FSx ONTAP Volume: %s", input)
151+
result, err := conn.CreateVolume(input)
152+
153+
if err != nil {
154+
return fmt.Errorf("error creating FSx Volume: %w", err)
155+
}
156+
157+
d.SetId(aws.StringValue(result.Volume.VolumeId))
158+
159+
if _, err := waitVolumeCreated(conn, d.Id(), d.Timeout(schema.TimeoutCreate)); err != nil {
160+
return fmt.Errorf("error waiting for FSx Volume(%s) create: %w", d.Id(), err)
161+
}
162+
163+
return resourceOntapVolumeRead(d, meta)
164+
165+
}
166+
167+
func resourceOntapVolumeRead(d *schema.ResourceData, meta interface{}) error {
168+
conn := meta.(*conns.AWSClient).FSxConn
169+
defaultTagsConfig := meta.(*conns.AWSClient).DefaultTagsConfig
170+
ignoreTagsConfig := meta.(*conns.AWSClient).IgnoreTagsConfig
171+
172+
volume, err := FindVolumeByID(conn, d.Id())
173+
174+
if !d.IsNewResource() && tfresource.NotFound(err) {
175+
log.Printf("[WARN] FSx ONTAP Volume (%s) not found, removing from state", d.Id())
176+
d.SetId("")
177+
return nil
178+
}
179+
180+
if err != nil {
181+
return fmt.Errorf("error reading FSx ONTAP Volume (%s): %w", d.Id(), err)
182+
}
183+
184+
ontapConfig := volume.OntapConfiguration
185+
if ontapConfig == nil {
186+
return fmt.Errorf("error describing FSx ONTAP Volume (%s): empty ONTAP configuration", d.Id())
187+
}
188+
189+
d.Set("arn", volume.ResourceARN)
190+
d.Set("name", volume.Name)
191+
d.Set("file_system_id", volume.FileSystemId)
192+
d.Set("junction_path", ontapConfig.JunctionPath)
193+
d.Set("ontap_volume_type", ontapConfig.OntapVolumeType)
194+
d.Set("security_style", ontapConfig.SecurityStyle)
195+
d.Set("size_in_megabytes", ontapConfig.SizeInMegabytes)
196+
d.Set("storage_efficiency_enabled", ontapConfig.StorageEfficiencyEnabled)
197+
d.Set("storage_virtual_machine_id", ontapConfig.StorageVirtualMachineId)
198+
d.Set("uuid", ontapConfig.UUID)
199+
d.Set("volume_type", volume.VolumeType)
200+
201+
if err := d.Set("tiering_policy", flattenFsxOntapVolumeTieringPolicy(ontapConfig.TieringPolicy)); err != nil {
202+
return fmt.Errorf("error setting tiering_policy: %w", err)
203+
}
204+
205+
//Volume tags do not get returned with describe call so need to make a separate list tags call
206+
tags, tagserr := ListTags(conn, *volume.ResourceARN)
207+
208+
if tagserr != nil {
209+
return fmt.Errorf("error reading Tags for FSx ONTAP Volume (%s): %w", d.Id(), err)
210+
} else {
211+
tags = tags.IgnoreAWS().IgnoreConfig(ignoreTagsConfig)
212+
}
213+
214+
//lintignore:AWSR002
215+
if err := d.Set("tags", tags.RemoveDefaultConfig(defaultTagsConfig).Map()); err != nil {
216+
return fmt.Errorf("error setting tags: %w", err)
217+
}
218+
219+
if err := d.Set("tags_all", tags.Map()); err != nil {
220+
return fmt.Errorf("error setting tags_all: %w", err)
221+
}
222+
223+
return nil
224+
}
225+
226+
func resourceOntapVolumeUpdate(d *schema.ResourceData, meta interface{}) error {
227+
conn := meta.(*conns.AWSClient).FSxConn
228+
229+
if d.HasChange("tags_all") {
230+
o, n := d.GetChange("tags_all")
231+
232+
if err := UpdateTags(conn, d.Get("arn").(string), o, n); err != nil {
233+
return fmt.Errorf("error updating FSx ONTAP Volume (%s) tags: %w", d.Get("arn").(string), err)
234+
}
235+
}
236+
237+
if d.HasChangesExcept("tags_all", "tags") {
238+
input := &fsx.UpdateVolumeInput{
239+
ClientRequestToken: aws.String(resource.UniqueId()),
240+
VolumeId: aws.String(d.Id()),
241+
OntapConfiguration: &fsx.UpdateOntapVolumeConfiguration{},
242+
}
243+
244+
if d.HasChange("junction_path") {
245+
input.OntapConfiguration.JunctionPath = aws.String(d.Get("junction_path").(string))
246+
}
247+
248+
if d.HasChange("security_style") {
249+
input.OntapConfiguration.SecurityStyle = aws.String(d.Get("security_style").(string))
250+
}
251+
252+
if d.HasChange("size_in_megabytes") {
253+
input.OntapConfiguration.SizeInMegabytes = aws.Int64(int64(d.Get("size_in_megabytes").(int)))
254+
}
255+
256+
if d.HasChange("storage_efficiency_enabled") {
257+
input.OntapConfiguration.StorageEfficiencyEnabled = aws.Bool(d.Get("storage_efficiency_enabled").(bool))
258+
}
259+
260+
if d.HasChange("tiering_policy") {
261+
input.OntapConfiguration.TieringPolicy = expandFsxOntapVolumeTieringPolicy(d.Get("tiering_policy").([]interface{}))
262+
}
263+
264+
_, err := conn.UpdateVolume(input)
265+
266+
if err != nil {
267+
return fmt.Errorf("error updating FSx ONTAP Volume (%s): %w", d.Id(), err)
268+
}
269+
270+
if _, err := waitVolumeUpdated(conn, d.Id(), d.Timeout(schema.TimeoutUpdate)); err != nil {
271+
return fmt.Errorf("error waiting for FSx ONTAP Volume (%s) update: %w", d.Id(), err)
272+
}
273+
}
274+
275+
return resourceOntapVolumeRead(d, meta)
276+
}
277+
278+
func resourceOntapVolumeDelete(d *schema.ResourceData, meta interface{}) error {
279+
conn := meta.(*conns.AWSClient).FSxConn
280+
281+
log.Printf("[DEBUG] Deleting FSx ONTAP Volume: %s", d.Id())
282+
_, err := conn.DeleteVolume(&fsx.DeleteVolumeInput{
283+
VolumeId: aws.String(d.Id()),
284+
})
285+
286+
if tfawserr.ErrCodeEquals(err, fsx.ErrCodeVolumeNotFound) {
287+
return nil
288+
}
289+
290+
if err != nil {
291+
return fmt.Errorf("error deleting FSx ONTAP Volume (%s): %w", d.Id(), err)
292+
}
293+
294+
if _, err := waitVolumeDeleted(conn, d.Id(), d.Timeout(schema.TimeoutDelete)); err != nil {
295+
return fmt.Errorf("error waiting for FSx ONTAP Volume (%s) delete: %w", d.Id(), err)
296+
}
297+
298+
return nil
299+
}
300+
301+
func expandFsxOntapVolumeTieringPolicy(cfg []interface{}) *fsx.TieringPolicy {
302+
if len(cfg) < 1 {
303+
return nil
304+
}
305+
306+
conf := cfg[0].(map[string]interface{})
307+
308+
out := fsx.TieringPolicy{}
309+
310+
//Cooling period only accepts a minimum of 2 but int will return 0 not nil if unset
311+
//Therefore we only set it if it is 2 or more
312+
if v, ok := conf["cooling_period"].(int); ok && v >= 2 {
313+
out.CoolingPeriod = aws.Int64(int64(v))
314+
}
315+
316+
if v, ok := conf["name"].(string); ok {
317+
out.Name = aws.String(v)
318+
}
319+
320+
return &out
321+
}
322+
323+
func flattenFsxOntapVolumeTieringPolicy(rs *fsx.TieringPolicy) []interface{} {
324+
if rs == nil {
325+
return []interface{}{}
326+
}
327+
328+
minCoolingPeriod := 2
329+
330+
m := make(map[string]interface{})
331+
if rs.CoolingPeriod != nil && *rs.CoolingPeriod >= int64(minCoolingPeriod) {
332+
m["cooling_period"] = aws.Int64Value(rs.CoolingPeriod)
333+
}
334+
335+
if rs.Name != nil {
336+
m["name"] = aws.StringValue(rs.Name)
337+
}
338+
339+
return []interface{}{m}
340+
}

0 commit comments

Comments
 (0)