Skip to content

Commit

Permalink
feat(pki): Add support for ACME configuration
Browse files Browse the repository at this point in the history
  • Loading branch information
Viper61 committed Feb 26, 2024
1 parent 23bd879 commit 801f4b5
Show file tree
Hide file tree
Showing 8 changed files with 462 additions and 15 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ FEATURES:
* Add support to `enable_templating` in `vault_pki_secret_backend_config_urls` ([#2147](https://github.com/hashicorp/terraform-provider-vault/pull/2147)).
* Add support for `skip_import_rotation` and `skip_static_role_import_rotation` in `ldap_secret_backend_static_role` and `ldap_secret_backend` respectively. Requires Vault 1.16+ ([#2128](https://github.com/hashicorp/terraform-provider-vault/pull/2128)).
* Improve logging to track full API exchanges between the provider and Vault ([#2139](https://github.com/hashicorp/terraform-provider-vault/pull/2139))
* Add support for ACME configuration with the `vault_pki_secret_backend_config_acme` resource. Requires Vault 1.14+ ([#2157](https://github.com/hashicorp/terraform-provider-vault/pull/2157)).

## 3.25.0 (Feb 14, 2024)

Expand Down
24 changes: 16 additions & 8 deletions internal/consts/consts.go
Original file line number Diff line number Diff line change
Expand Up @@ -391,6 +391,13 @@ const (
FieldWrappingToken = "wrapping_token"
FieldWithWrappedAccessor = "with_wrapped_accessor"
FieldAIAPath = "aia_path"
FieldEnabled = "enabled"
FieldAllowedIssuers = "allowed_issuers"
FieldAllowedRoles = "allowed_roles"
FieldAllowRoleExtKeyUsage = "allow_role_ext_key_usage"
FieldDefaultDirectoryPolicy = "default_directory_policy"
FieldDnsResolver = "dns_resolver"
FieldEabPolicy = "eab_policy"

/*
common environment variables
Expand Down Expand Up @@ -456,14 +463,15 @@ const (
/*
Vault version constants
*/
VaultVersion190 = "1.9.0"
VaultVersion110 = "1.10.0"
VaultVersion111 = "1.11.0"
VaultVersion112 = "1.12.0"
VaultVersion113 = "1.13.0"
VaultVersion114 = "1.14.0"
VaultVersion115 = "1.15.0"
VaultVersion116 = "1.16.0"
VaultVersion190 = "1.9.0"
VaultVersion110 = "1.10.0"
VaultVersion111 = "1.11.0"
VaultVersion112 = "1.12.0"
VaultVersion113 = "1.13.0"
VaultVersion114 = "1.14.0"
VaultVersion1141 = "1.14.1"
VaultVersion115 = "1.15.0"
VaultVersion116 = "1.16.0"

/*
Vault auth methods
Expand Down
15 changes: 8 additions & 7 deletions internal/provider/meta.go
Original file line number Diff line number Diff line change
Expand Up @@ -35,13 +35,14 @@ const (
var (
MaxHTTPRetriesCCC int

VaultVersion110 = version.Must(version.NewSemver(consts.VaultVersion110))
VaultVersion111 = version.Must(version.NewSemver(consts.VaultVersion111))
VaultVersion112 = version.Must(version.NewSemver(consts.VaultVersion112))
VaultVersion113 = version.Must(version.NewSemver(consts.VaultVersion113))
VaultVersion114 = version.Must(version.NewSemver(consts.VaultVersion114))
VaultVersion115 = version.Must(version.NewSemver(consts.VaultVersion115))
VaultVersion116 = version.Must(version.NewSemver(consts.VaultVersion116))
VaultVersion110 = version.Must(version.NewSemver(consts.VaultVersion110))
VaultVersion111 = version.Must(version.NewSemver(consts.VaultVersion111))
VaultVersion112 = version.Must(version.NewSemver(consts.VaultVersion112))
VaultVersion113 = version.Must(version.NewSemver(consts.VaultVersion113))
VaultVersion114 = version.Must(version.NewSemver(consts.VaultVersion114))
VaultVersion1141 = version.Must(version.NewSemver(consts.VaultVersion1141))
VaultVersion115 = version.Must(version.NewSemver(consts.VaultVersion115))
VaultVersion116 = version.Must(version.NewSemver(consts.VaultVersion116))

TokenTTLMinRecommended = time.Minute * 15
)
Expand Down
4 changes: 4 additions & 0 deletions vault/provider.go
Original file line number Diff line number Diff line change
Expand Up @@ -565,6 +565,10 @@ var (
Resource: UpdateSchemaResource(pkiSecretBackendCrlConfigResource()),
PathInventory: []string{"/pki/config/crl"},
},
"vault_pki_secret_backend_config_acme": {
Resource: UpdateSchemaResource(pkiSecretBackendConfigACMEResource()),
PathInventory: []string{"/pki/config/acme"},
},
"vault_pki_secret_backend_config_ca": {
Resource: UpdateSchemaResource(pkiSecretBackendConfigCAResource()),
PathInventory: []string{"/pki/config/ca"},
Expand Down
231 changes: 231 additions & 0 deletions vault/resource_pki_secret_backend_config_acme.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,231 @@
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0

package vault

import (
"context"
"fmt"
"log"
"regexp"
"strings"

"github.com/hashicorp/terraform-plugin-sdk/v2/diag"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema"
"github.com/hashicorp/terraform-plugin-sdk/v2/helper/validation"

"github.com/hashicorp/terraform-provider-vault/internal/consts"
"github.com/hashicorp/terraform-provider-vault/internal/provider"
"github.com/hashicorp/terraform-provider-vault/util"
)

var (
pkiSecretBackendFromConfigACMERegex = regexp.MustCompile("^(.+)/config/acme$")
pkiAcmeFields = []string{
consts.FieldEnabled,
consts.FieldDefaultDirectoryPolicy,
consts.FieldAllowedRoles,
consts.FieldAllowRoleExtKeyUsage,
consts.FieldAllowedIssuers,
consts.FieldEabPolicy,
consts.FieldDnsResolver,
}
)

func pkiSecretBackendConfigACMEResource() *schema.Resource {
return &schema.Resource{
CreateContext: provider.MountCreateContextWrapper(pkiSecretBackendConfigACMECreate, provider.VaultVersion113),
ReadContext: provider.ReadContextWrapper(pkiSecretBackendConfigACMERead),
UpdateContext: pkiSecretBackendConfigACMEUpdate,
DeleteContext: pkiSecretBackendConfigACMEDelete,
Importer: &schema.ResourceImporter{
StateContext: func(_ context.Context, d *schema.ResourceData, meta interface{}) ([]*schema.ResourceData, error) {
id := d.Id()
if id == "" {
return nil, fmt.Errorf("no path set for import, id=%q", id)
}

parts := strings.Split(util.NormalizeMountPath(id), "/")
if err := d.Set("backend", parts[0]); err != nil {
return nil, err
}

return []*schema.ResourceData{d}, nil
},
},

Schema: map[string]*schema.Schema{
consts.FieldBackend: {
Type: schema.TypeString,
Required: true,
ForceNew: true,
Description: "Full path where PKI backend is mounted.",
// standardise on no beginning or trailing slashes
StateFunc: func(v interface{}) string {
return strings.Trim(v.(string), "/")
},
},
consts.FieldEnabled: {
Type: schema.TypeBool,
Required: true,
Description: "Specifies whether ACME is enabled.",
},
consts.FieldDefaultDirectoryPolicy: {
Type: schema.TypeString,
Optional: true,
Computed: true,
Description: "Specifies the policy to be used for non-role-qualified ACME requests.",
ValidateFunc: validation.StringMatch(
regexp.MustCompile(`^(forbid|sign-verbatim|role:.+|external-policy(?:\:.+)?)$`), ""),
},
consts.FieldAllowedRoles: {
Type: schema.TypeList,
Optional: true,
Computed: true,
Description: "Specifies which roles are allowed for use with ACME.",
Elem: &schema.Schema{
Type: schema.TypeString,
},
},
consts.FieldAllowRoleExtKeyUsage: {
Type: schema.TypeBool,
Optional: true,
Description: "Specifies whether the ExtKeyUsage field from a role is used.",
},
consts.FieldAllowedIssuers: {
Type: schema.TypeList,
Optional: true,
Computed: true,
Description: "Specifies which issuers are allowed for use with ACME.",
Elem: &schema.Schema{
Type: schema.TypeString,
},
},
consts.FieldEabPolicy: {
Type: schema.TypeString,
Optional: true,
Computed: true,
Description: "Specifies the policy to use for external account binding behaviour.",
ValidateFunc: validation.StringInSlice(
[]string{"not-required", "new-account-required", "always-required"}, false),
},
consts.FieldDnsResolver: {
Type: schema.TypeString,
Optional: true,
Description: "DNS resolver to use for domain resolution on this mount. " +
"Must be in the format <host>:<port>, with both parts mandatory.",
ValidateFunc: validation.StringMatch(
regexp.MustCompile(`^([0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}:[0-9]{1,5})?$`), ""),
},
},
}
}

func pkiSecretBackendConfigACMECreate(ctx context.Context, d *schema.ResourceData, meta interface{}) diag.Diagnostics {
client, e := provider.GetClient(d, meta)
if e != nil {
return diag.FromErr(e)
}

backend := d.Get(consts.FieldBackend).(string)
path := fmt.Sprintf("%s/config/acme", backend)

resp, err := client.Logical().ReadWithContext(ctx, path)
if err != nil {
return diag.Errorf("error reading acme config at %s, err=%s", path, err)
}

if resp == nil {
return diag.Errorf("no acme config found at path %s", path)
}

d.SetId(path)

return pkiSecretBackendConfigACMEUpdate(ctx, d, meta)
}

func pkiSecretBackendConfigACMEUpdate(ctx context.Context, d *schema.ResourceData, meta interface{}) diag.Diagnostics {
client, e := provider.GetClient(d, meta)
if e != nil {
return diag.FromErr(e)
}

path := d.Id()

var patchRequired bool
data := map[string]interface{}{}
for _, k := range pkiAcmeFields {
if d.HasChange(k) {
data[k] = d.Get(k)
patchRequired = true
}
}

if patchRequired {
_, err := client.Logical().WriteWithContext(ctx, path, data)
if err != nil {
return diag.Errorf("error writing data to %q, err=%s", path, err)
}
}

return pkiSecretBackendConfigACMERead(ctx, d, meta)
}

func pkiSecretBackendConfigACMERead(ctx context.Context, d *schema.ResourceData, meta interface{}) diag.Diagnostics {
client, e := provider.GetClient(d, meta)
if e != nil {
return diag.FromErr(e)
}

path := d.Id()
if path == "" {
return diag.Errorf("no path set, id=%q", d.Id())
}

// get backend from full path
backend, err := pkiSecretBackendFromConfigACME(path)
if err != nil {
return diag.FromErr(err)
}

log.Printf("[DEBUG] Reading %s from Vault", path)
resp, err := client.Logical().ReadWithContext(ctx, path)
if err != nil {
return diag.Errorf("error reading from Vault: %s", err)
}

if resp == nil {
d.SetId("")

return nil
}

// set backend and issuerRef
if err := d.Set(consts.FieldBackend, backend); err != nil {
return diag.FromErr(err)
}

for _, k := range pkiAcmeFields {
if err := d.Set(k, resp.Data[k]); err != nil {
return diag.Errorf("error setting state key %q for acme config, err=%s",
k, err)
}
}

return nil
}

func pkiSecretBackendConfigACMEDelete(ctx context.Context, d *schema.ResourceData, meta interface{}) diag.Diagnostics {
return nil
}

func pkiSecretBackendFromConfigACME(path string) (string, error) {
if !pkiSecretBackendFromConfigACMERegex.MatchString(path) {
return "", fmt.Errorf("no backend found")
}
res := pkiSecretBackendFromConfigACMERegex.FindStringSubmatch(path)
if len(res) != 2 {
return "", fmt.Errorf("unexpected number of matches (%d) for backend", len(res))
}
return res[1], nil
}
Loading

0 comments on commit 801f4b5

Please sign in to comment.