0
0
mirror of https://github.com/bpg/terraform-provider-proxmox.git synced 2025-06-30 02:31:10 +00:00
terraform-provider-proxmox/fwprovider/cluster/acme/datasource_acme_account.go
Pavel Boldyrev 2a356014a1
misc(code): move fwprovider files around (#1866)
Signed-off-by: Pavel Boldyrev <627562+bpg@users.noreply.github.com>
2025-03-29 19:02:41 +00:00

191 lines
5.6 KiB
Go

/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/
package acme
import (
"context"
"fmt"
"github.com/hashicorp/terraform-plugin-framework/attr"
"github.com/hashicorp/terraform-plugin-framework/datasource"
"github.com/hashicorp/terraform-plugin-framework/datasource/schema"
"github.com/hashicorp/terraform-plugin-framework/types"
"github.com/bpg/terraform-provider-proxmox/fwprovider/config"
"github.com/bpg/terraform-provider-proxmox/proxmox/cluster/acme/account"
)
// Ensure the implementation satisfies the expected interfaces.
var (
_ datasource.DataSource = &acmeAccountDatasource{}
_ datasource.DataSourceWithConfigure = &acmeAccountDatasource{}
)
// NewACMEAccountDataSource is a helper function to simplify the provider implementation.
func NewACMEAccountDataSource() datasource.DataSource {
return &acmeAccountDatasource{}
}
// acmeAccountDatasource is the data source implementation for ACME accounts.
type acmeAccountDatasource struct {
client *account.Client
}
type accountDataModel struct {
Contact []types.String `tfsdk:"contact"`
CreatedAt types.String `tfsdk:"created_at"`
Status types.String `tfsdk:"status"`
}
func (m *accountDataModel) attrTypes() map[string]attr.Type {
return map[string]attr.Type{
"contact": types.ListType{ElemType: types.StringType},
"created_at": types.StringType,
"status": types.StringType,
}
}
// accountModel is the model used to represent an ACME account.
type accountModel struct {
// Name is the ACME account config file name.
Name types.String `tfsdk:"name"`
// Account is the ACME account information.
Account types.Object `tfsdk:"account"`
// Directory is the URL of the ACME CA directory endpoint.
Directory types.String `tfsdk:"directory"`
// Location is the location of the ACME account.
Location types.String `tfsdk:"location"`
// URL of CA TermsOfService - setting this indicates agreement.
TOS types.String `tfsdk:"tos"`
}
// Metadata returns the data source type name.
func (d *acmeAccountDatasource) Metadata(
_ context.Context,
req datasource.MetadataRequest,
resp *datasource.MetadataResponse,
) {
resp.TypeName = req.ProviderTypeName + "_acme_account"
}
// Schema returns the schema for the data source.
func (d *acmeAccountDatasource) Schema(
_ context.Context,
_ datasource.SchemaRequest,
resp *datasource.SchemaResponse,
) {
resp.Schema = schema.Schema{
Description: "Retrieves information about a specific ACME account.",
Attributes: map[string]schema.Attribute{
"name": schema.StringAttribute{
Description: "The identifier of the ACME account to read.",
Optional: true,
},
"account": schema.SingleNestedAttribute{
Description: "The ACME account information.",
Computed: true,
Attributes: map[string]schema.Attribute{
"contact": schema.ListAttribute{
Description: "An array of contact email addresses.",
ElementType: types.StringType,
Computed: true,
},
"created_at": schema.StringAttribute{
Description: "The timestamp of the account creation.",
Computed: true,
},
"status": schema.StringAttribute{
Description: "The status of the account.",
MarkdownDescription: "The status of the account. Can be one of `valid`, `deactivated` or `revoked`.",
Computed: true,
},
},
},
"directory": schema.StringAttribute{
Description: "The directory URL of the ACME account.",
Computed: true,
},
"location": schema.StringAttribute{
Description: "The location URL of the ACME account.",
Computed: true,
},
"tos": schema.StringAttribute{
Description: "The URL of the terms of service of the ACME account.",
Computed: true,
},
},
}
}
// Configure adds the provider-configured client to the data source.
func (d *acmeAccountDatasource) Configure(
_ context.Context,
req datasource.ConfigureRequest,
resp *datasource.ConfigureResponse,
) {
if req.ProviderData == nil {
return
}
cfg, ok := req.ProviderData.(config.DataSource)
if !ok {
resp.Diagnostics.AddError(
"Unexpected DataSource Configure Type",
fmt.Sprintf("Expected config.DataSource, got: %T", req.ProviderData),
)
return
}
d.client = cfg.Client.Cluster().ACME().Account()
}
// Read retrieves the ACME account information.
func (d *acmeAccountDatasource) Read(ctx context.Context, req datasource.ReadRequest, resp *datasource.ReadResponse) {
var state accountModel
resp.Diagnostics.Append(req.Config.Get(ctx, &state)...)
if resp.Diagnostics.HasError() {
return
}
name := state.Name.ValueString()
accountData, err := d.client.Get(ctx, name)
if err != nil {
resp.Diagnostics.AddError(
fmt.Sprintf("Unable to read ACME account '%s'", name),
err.Error(),
)
return
}
contactList := make([]types.String, len(accountData.Account.Contact))
for i, contact := range accountData.Account.Contact {
contactList[i] = types.StringValue(contact)
}
data := &accountDataModel{
Contact: contactList,
CreatedAt: types.StringValue(accountData.Account.CreatedAt),
Status: types.StringValue(accountData.Account.Status),
}
accountObject, diags := types.ObjectValueFrom(ctx, data.attrTypes(), data)
resp.Diagnostics.Append(diags...)
state.Account = accountObject
state.Directory = types.StringValue(accountData.Directory)
state.Location = types.StringValue(accountData.Location)
state.TOS = types.StringValue(accountData.TOS)
resp.Diagnostics.Append(resp.State.Set(ctx, &state)...)
}