From 4947e8d6e2aff614a9b32d1b63c0441c9a27b9a5 Mon Sep 17 00:00:00 2001 From: Kei Tsuchiya Date: Fri, 5 Apr 2024 18:09:45 +0900 Subject: [PATCH 1/2] set AbortIncompleteMultipartUpload action as first rule and set default value --- aws/private-s3-bucket/bucket_options.tf | 77 +++---------------------- aws/private-s3-bucket/variables.tf | 25 ++++---- 2 files changed, 19 insertions(+), 83 deletions(-) diff --git a/aws/private-s3-bucket/bucket_options.tf b/aws/private-s3-bucket/bucket_options.tf index ba35453..c7ea022 100644 --- a/aws/private-s3-bucket/bucket_options.tf +++ b/aws/private-s3-bucket/bucket_options.tf @@ -61,79 +61,22 @@ resource "aws_s3_bucket_ownership_controls" "b" { object_ownership = local.object_ownership } } + # lifecycle resource "aws_s3_bucket_lifecycle_configuration" "b" { - count = length(var.lifecycle_rule) > 0 ? 1 : 0 bucket = aws_s3_bucket.b.id - # First(0-th) rule: aws_s3_bucket_lifecycle_configuration must have at least 1 rule block explicitly rule { - id = var.lifecycle_rule[0].id - status = var.lifecycle_rule[0].enabled ? "Enabled" : "Disabled" - dynamic "filter" { - for_each = var.lifecycle_rule[0].prefix != null || length(var.lifecycle_rule[0].tags) > 0 ? [1] : [] - content { - # Only prefix or only 1 tag: bare - # both prefix and tag, multiple tags: inside `and` block - prefix = length(var.lifecycle_rule[0].tags) == 0 ? var.lifecycle_rule[0].prefix : null - dynamic "tag" { - for_each = var.lifecycle_rule[0].prefix == null && length(var.lifecycle_rule[0].tags) == 1 ? [1] : [] - content { - key = keys(var.lifecycle_rule[0].tags)[0] - value = values(var.lifecycle_rule[0].tags)[0] - } - } - dynamic "and" { - for_each = (var.lifecycle_rule[0].prefix != null && length(var.lifecycle_rule[0].tags) >= 1) || length(var.lifecycle_rule[0].tags) >= 2 ? [1] : [] - content { - prefix = var.lifecycle_rule[0].prefix - tags = var.lifecycle_rule[0].tags - } - } - } - } - dynamic "abort_incomplete_multipart_upload" { - for_each = var.lifecycle_rule[0].abort_incomplete_multipart_upload_days != null ? [1] : [] - content { - days_after_initiation = var.lifecycle_rule[0].abort_incomplete_multipart_upload_days - } - } - dynamic "transition" { - for_each = var.lifecycle_rule[0].transition - content { - date = transition.value.date - days = transition.value.days - storage_class = transition.value.storage_class - } - } - dynamic "expiration" { - for_each = var.lifecycle_rule[0].expiration - content { - date = expiration.value.date - days = expiration.value.days - expired_object_delete_marker = expiration.value.expired_object_delete_marker - } - } - dynamic "noncurrent_version_transition" { - for_each = var.lifecycle_rule[0].noncurrent_version_transition - content { - noncurrent_days = noncurrent_version_transition.value.days - newer_noncurrent_versions = noncurrent_version_transition.value.versions - storage_class = noncurrent_version_transition.value.storage_class - } - } - dynamic "noncurrent_version_expiration" { - for_each = var.lifecycle_rule[0].noncurrent_version_expiration - content { - noncurrent_days = noncurrent_version_expiration.value.days - newer_noncurrent_versions = noncurrent_version_expiration.value.versions - } + id = "Abort incomplete multipart upload" + status = "Enabled" + + abort_incomplete_multipart_upload { + days_after_initiation = 3 } } - # Rest (1st and after) rules dynamic "rule" { - for_each = toset(slice(var.lifecycle_rule, 1, length(var.lifecycle_rule))) + for_each = toset(var.lifecycle_rule) content { id = rule.value.id status = rule.value.enabled ? "Enabled" : "Disabled" @@ -159,12 +102,6 @@ resource "aws_s3_bucket_lifecycle_configuration" "b" { } } } - dynamic "abort_incomplete_multipart_upload" { - for_each = rule.value.abort_incomplete_multipart_upload_days != null ? [1] : [] - content { - days_after_initiation = rule.value.abort_incomplete_multipart_upload_days - } - } dynamic "transition" { for_each = rule.value.transition content { diff --git a/aws/private-s3-bucket/variables.tf b/aws/private-s3-bucket/variables.tf index 0c319b0..5f583b7 100644 --- a/aws/private-s3-bucket/variables.tf +++ b/aws/private-s3-bucket/variables.tf @@ -54,32 +54,31 @@ variable "grant" { variable "lifecycle_rule" { type = list(object({ - id = string - enabled = bool - prefix = string - abort_incomplete_multipart_upload_days = number - tags = map(string) - transition = list(object({ + id = string + enabled = optional(bool, true) + prefix = optional(string) + tags = optional(map(string), {}) + transition = optional(list(object({ date = optional(string) days = optional(number) storage_class = string - })) + })), []) # Note for expiration, noncurrent_version_transition, noncurrent_version_expiration # define as list for simplicity, though expected only a single object - expiration = list(object({ + expiration = optional(list(object({ date = optional(string) days = optional(number) expired_object_delete_marker = optional(bool, false) - })) - noncurrent_version_transition = list(object({ + })), []) + noncurrent_version_transition = optional(list(object({ days = number versions = optional(number) storage_class = string - })) - noncurrent_version_expiration = list(object({ + })), []) + noncurrent_version_expiration = optional(list(object({ days = number versions = optional(number) - })) + })), []) })) description = "S3 lifecycle rule" default = [] From 9c05b536c6343e9925ca6832fa6e25dfa6cd1648 Mon Sep 17 00:00:00 2001 From: Kei Tsuchiya Date: Fri, 5 Apr 2024 18:10:57 +0900 Subject: [PATCH 2/2] update README --- aws/private-s3-bucket/README.md | 5 ++--- aws/private-s3-bucket/main.tf | 3 +-- 2 files changed, 3 insertions(+), 5 deletions(-) diff --git a/aws/private-s3-bucket/README.md b/aws/private-s3-bucket/README.md index 32f9a58..411ccbe 100644 --- a/aws/private-s3-bucket/README.md +++ b/aws/private-s3-bucket/README.md @@ -62,7 +62,7 @@ logging = [ Full featured example. NOTE: - * abort\_incomplete\_multipart\_upload\_days is exclusive against tags + * abort\_incomplete\_multipart\_upload\_days is always set as 3 days * expiration, noncurrent\_version\_{transition,expiration} can be set up to once ```hcl @@ -75,7 +75,6 @@ lifecycle_rule = [ a = "b" c = "d" } - abort_incomplete_multipart_upload_days = null transition = [ { date = null @@ -198,7 +197,7 @@ No modules. | [disable\_private](#input\_disable\_private) | If true, disable private bucket feature | `bool` | `false` | no | | [disable\_sse](#input\_disable\_sse) | If true, disable server side encryption | `bool` | `false` | no | | [grant](#input\_grant) | S3 grants |
list(object({
id = string
type = string
permissions = list(string)
uri = string
}))
| `[]` | no | -| [lifecycle\_rule](#input\_lifecycle\_rule) | S3 lifecycle rule |
list(object({
id = string
enabled = bool
prefix = string
abort_incomplete_multipart_upload_days = number
tags = map(string)
transition = list(object({
date = optional(string)
days = optional(number)
storage_class = string
}))
# Note for expiration, noncurrent_version_transition, noncurrent_version_expiration
# define as list for simplicity, though expected only a single object
expiration = list(object({
date = optional(string)
days = optional(number)
expired_object_delete_marker = optional(bool, false)
}))
noncurrent_version_transition = list(object({
days = number
versions = optional(number)
storage_class = string
}))
noncurrent_version_expiration = list(object({
days = number
versions = optional(number)
}))
}))
| `[]` | no | +| [lifecycle\_rule](#input\_lifecycle\_rule) | S3 lifecycle rule |
list(object({
id = string
enabled = optional(bool, true)
prefix = optional(string)
tags = optional(map(string), {})
transition = optional(list(object({
date = optional(string)
days = optional(number)
storage_class = string
})), [])
# Note for expiration, noncurrent_version_transition, noncurrent_version_expiration
# define as list for simplicity, though expected only a single object
expiration = optional(list(object({
date = optional(string)
days = optional(number)
expired_object_delete_marker = optional(bool, false)
})), [])
noncurrent_version_transition = optional(list(object({
days = number
versions = optional(number)
storage_class = string
})), [])
noncurrent_version_expiration = optional(list(object({
days = number
versions = optional(number)
})), [])
}))
| `[]` | no | | [logging](#input\_logging) | S3 access logging |
list(object({
target_bucket = string
target_prefix = string
}))
| `[]` | no | | [mfa\_delete](#input\_mfa\_delete) | Enable MFA delete, this requires the versioning feature | `bool` | `false` | no | | [object\_lock\_configuration](#input\_object\_lock\_configuration) | S3 Object Lock Configuration. You can only enable S3 Object Lock for new buckets. If you need to turn on S3 Object Lock for an existing bucket, please contact AWS Support. |
list(object({
rule = object({
default_retention = object({
mode = string
days = number
years = number
})
})
}))
| `[]` | no | diff --git a/aws/private-s3-bucket/main.tf b/aws/private-s3-bucket/main.tf index 4473d99..34df65c 100644 --- a/aws/private-s3-bucket/main.tf +++ b/aws/private-s3-bucket/main.tf @@ -62,7 +62,7 @@ * Full featured example. * * NOTE: -* * abort_incomplete_multipart_upload_days is exclusive against tags +* * abort_incomplete_multipart_upload_days is always set as 3 days * * expiration, noncurrent_version_{transition,expiration} can be set up to once * * ```hcl @@ -75,7 +75,6 @@ * a = "b" * c = "d" * } -* abort_incomplete_multipart_upload_days = null * transition = [ * { * date = null