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: add juju_access_offer resource #633

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

amandahla
Copy link
Contributor

@amandahla amandahla commented Nov 19, 2024

Description

This is a PR for adding the juju_access_offer resource based on schema described in #627

Type of change

<What type of a change is this? Please keep one or more relevant lines from below and delete the others.>

  • Add new resource

Environment

  • Juju controller version: 3.5

  • Terraform version: 1.7.4

QA steps

Manual QA steps should be done to test this PR.

terraform {
  required_providers {
    juju = {
      version = "~> 0.15.0"
      source  = "registry.terraform.io/juju/juju"
    }
  }
}

provider "juju" {}

resource "juju_model" "development" {
  name = "dev"
}

resource "juju_user" "amanda" {
  name     = "amanda"
  password = "amanda"
}


resource "juju_application" "prometheus" {
  name = "prometheus-k8s"

  model = juju_model.development.name

  charm {
    name    = "prometheus-k8s"
    channel = "latest/edge"
  }

  units = 1
}

resource "juju_offer" "prometheus" {
  model            = juju_model.development.name
  application_name = juju_application.prometheus.name
  endpoint         = "receive-remote-write"
}

resource "juju_access_offer" "prometheus" {
  offer_url = juju_offer.prometheus.url
  consume   = [juju_user.amanda.name]
}

Additional notes

<Please add relevant notes & information, links to mattermost chats, other related issues/PRs, anything to help understand and QA the PR.>

@amandahla amandahla marked this pull request as draft November 19, 2024 21:23
@amandahla amandahla marked this pull request as ready for review November 28, 2024 21:21
@amandahla amandahla changed the title WIP feat: add juju_access_offer resource feat: add juju_access_offer resource Nov 28, 2024
@amandahla amandahla force-pushed the feat/offer-access branch 2 times, most recently from 29c1c47 to df0e20b Compare December 4, 2024 13:40
Copy link
Member

@alesstimec alesstimec left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

some initial comments. happy to talk about this, if needed.

internal/provider/resource_access_offer.go Show resolved Hide resolved
internal/provider/resource_access_offer.go Show resolved Hide resolved
internal/provider/resource_access_offer.go Show resolved Hide resolved
}

// Get information from ID
offerURL, _, _ := retrieveAccessOfferDataFromID(ctx, state.ID, state.Users, &resp.Diagnostics)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why ignore the access level from offer ID. i think you should use it to filter out users that don't have the matching access level.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Implementation has changed but the provider still gets all users from the Offer, creates a map and add them to the state.

continue
}
users = append(users, offerUserDetail.UserName)
if access != "" && access != string(offerUserDetail.Access) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

again, i don't think this is quite right. imagine an application offer where alice has admin access, bob has consume access and eve has read access, which is quite a valid state for an application offer. the way this condition is written, this would produce an error here, because we see three different access levels for the same offer.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Implementation has changed. Could you review it again?

internal/provider/resource_access_offer.go Show resolved Hide resolved
internal/provider/resource_access_offer.go Show resolved Hide resolved
internal/provider/resource_access_offer.go Show resolved Hide resolved
internal/provider/resource_access_offer.go Show resolved Hide resolved
return nil
}

// This function revokes access to an offer.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This function revokes a specific level of access to an offer.

I think (correct me if i'm wrong), but access levels for offers work a bit differently in juju.
e.g. if a user has admin access to an offer and we revoke admin access for that user, the user is left with consume access to the offer.. if we revoke consume access from an admin user, that user is left with read access. and if we revoke read access, user is left with no access to the offer.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@alesstimec you are correct in how revoke on offer works.

The provider should decide which level to grant or revoke, not the internal juju code making the API call.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@hmlanigan yes, but with the current schema, where juju_access_offer only lists users for a single access level, this is nearly impossible, since you can have a juju_access_offer resource for the consume level and another one for read level and the two don't "know" about eachother.

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also the godoc is not formatted correctly.

.github/workflows/test_integration.yml Outdated Show resolved Hide resolved

A resource that represent a Juju Access Offer.


Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

todo: add examples/resource/juju_access_offer directory with resource.tf and import.sh files, content will be added to this markdown file during make install.

Comment on lines 25 to 24
"github.com/juju/juju/core/crossmodel"
"github.com/juju/names/v5"
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

todo: move the github.com/juju imports to the second stanza, in sorted order.

resp.Diagnostics.AddError(
"ImportState Failure",
fmt.Sprintf("Malformed AccessOffer ID %q, "+
"please use format '<offer URL>:<access>:<user1,user1>'", IDstr),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thought: we're having issues with cross controller CMR today in this provider, however an offer URL with a controller uses a colon, e.g. foo:admin/another-model.postgresql. A different delineator would be a good idea.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now the ID is just the offer URL.

Comment on lines 240 to 253
// Get user/access info from Offer
response, err := a.client.Offers.ReadOffer(&juju.ReadOfferInput{
OfferURL: offerURL,
})
if err != nil {
resp.Diagnostics.AddError("Client Error", fmt.Sprintf("Unable to read offer, got error: %s", err))
return
}
a.trace(fmt.Sprintf("read juju offer %q", offerURL))
offerUsers := response.Users
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This data is available in state:

        var state accessOfferResourceOffer

 	// Get the current state
 	resp.Diagnostics.Append(req.State.Get(ctx, &state)...)

Doing it this way has the potential to change access levels for the offer which were not set via this resource.

return nil
}

// This function revokes access to an offer.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@alesstimec you are correct in how revoke on offer works.

The provider should decide which level to grant or revoke, not the internal juju code making the API call.

offerUserNames[offerUser.UserName] = struct{}{}
if !slices.Contains(planUsers, offerUser.UserName) {
a.trace(fmt.Sprintf("revoke user %q for offer %q", offerUser.UserName, offerURL))
err := a.client.Offers.RevokeOffer(&juju.GrantRevokeOfferInput{
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If a user has been removed from the resource, then to remove their privileges, you'd Revoke read. Afaik, it's not necessary to step through the levels.

I've also suggested not allowing the admin user as a valid user to avoid problems there.

Comment on lines 433 to 441
_, err = client.ApplicationOffer(input.OfferURL)
if err != nil {
return err
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No need to do this. RevokeOffer will validate that the offer exists.

Comment on lines 411 to 417
_, err = client.ApplicationOffer(input.OfferURL)
if err != nil {
return err
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No need to do this. GrantOffer will validate that the offer exists.

users := []string{}
var access string
for _, offerUserDetail := range response.Users {
if offerUserDetail.UserName == "everyone@external" || offerUserDetail.UserName == "admin" {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

todo: please document in the schema description the admin user will not be handled via this resource. Suggest failing the validation of the user set as well if admin is specified.

type GrantRevokeOfferInput struct {
User string
Access string
OfferURL string
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this should be URL's, you can revoke many at once

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since we discussed changing the schema in the last terraform office hours, I think this no longer apply, WDYT?

return nil
}

// This function revokes access to an offer.
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also the godoc is not formatted correctly.

return err
}

err = client.RevokeOffer(input.User, input.Access, input.OfferURL)
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should probably be (if we can make the schema look nice) input.OfferURLs

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since we discussed changing the schema in the last terraform office hours, I think this no longer apply, WDYT?

@amandahla amandahla force-pushed the feat/offer-access branch 2 times, most recently from 3fa9531 to 35e236a Compare December 12, 2024 20:55
internal/juju/offers.go Show resolved Hide resolved
err = client.GrantOffer(user, input.Access, input.OfferURL)
if err != nil {
// ignore if user was already granted
if strings.Contains(err.Error(), "user already has") {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i don't think the juju api client returns an error in that case... maybe @hmlanigan knows about this

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I got this error while testing and the python library does something similar:
https://github.com/juju/python-libjuju/blob/1907f7af91195712badc8358d405424fc85f3ee0/juju/controller.py#L698

internal/juju/offers.go Outdated Show resolved Hide resolved
err = client.RevokeOffer(user, input.Access, input.OfferURL)
if err != nil {
// ignorer if user was already revoked
if strings.Contains(err.Error(), "not found") {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

same question as above - i'm not sure what error is returned in the case where user does not have access to an offer

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

internal/provider/resource_access_offer.go Outdated Show resolved Hide resolved
@alesstimec alesstimec requested a review from hmlanigan January 8, 2025 13:53
@amandahla amandahla force-pushed the feat/offer-access branch 4 times, most recently from d47a190 to 41675ff Compare January 9, 2025 14:36
@alesstimec alesstimec requested a review from kian99 January 10, 2025 07:59
@kian99 kian99 requested a review from SimoneDutto January 10, 2025 09:09
internal/juju/offers.go Outdated Show resolved Hide resolved
internal/provider/resource_access_offer.go Show resolved Hide resolved
internal/provider/resource_access_offer.go Show resolved Hide resolved
}
}

// Revoke read (remove access) of all state users
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why revoke access from all users and not just those users whose access has been removed?
And then add access for new users.
Is it just for simplicity sake?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Discussed here :)

#633 (comment)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Gotcha, ty. I'd suggest adding a comment explaining thinking. Coming across this without explanation I'd be surprised.

continue
}

if _, ok := users[offerUserDetail.Access]; !ok {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Have you tested the import functionality? Because the import doesn't accept a list of users/access levels, so if any users already have access the Read() which I believe is called on import, will fail.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I didn't follow. Why would it fail?

resource.TestCheckNoResourceAttr(resourceName, "read.*"),
),
},
{ // Destroy the resource and validate it can be imported correctly
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Before the last test case, you can consider adding a test to import the offer access.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't this already tested with ImportStateVerify: true and ImportState: true?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants