GitXplorerGitXplorer
a

iam-policy-document-tester

public
8 stars
1 forks
0 issues

Commits

List of commits on branch main.
Verified
e390a13c2623fcd04ba1663af7a75392d176bfb4

Update README.md

aalexwlchan committed 4 years ago
Verified
b6b1fcb4cdf48b6a3ecf903894b0f19a71ca914b

Update README.md

aalexwlchan committed 4 years ago
Verified
7d323949f9664100a6a8361b54a51ce7d21bbe4f

Update README.md

aalexwlchan committed 4 years ago
Verified
ff6eeb984887f2736a37bd4b2ef9619bceb64549

Update README.md

aalexwlchan committed 4 years ago
Verified
dcf226c6f718cd21dc583550662a5552286840af

Update README.md

aalexwlchan committed 4 years ago
Verified
b7995fa531ecd6a2940192340ef7b248a19f807b

Update README.md

aalexwlchan committed 4 years ago

README

The README file for this repository.

iam-policy-document-tester

Create short-lived, temporary roles for experimenting with AWS IAM policy documents.

This is a Python function for rapidly testing and experimenting with AWS IAM policy documents. Here's what it looks like:

with temporary_iam_credentials(admin_role_arn, policy_document) as credentials:
    # Do stuff with your new credentials, which have the permissions defined by the
    # IAM policy document.

The function temporary_iam_credentials() gives you a set of temporary AWS credentials, which have the permissions defined by the IAM policy document. You can make API calls using those credentials, and check they behave correctly -- that API calls are allowed or denied as appropriate.

When you're done, it cleans up after itself, so there are no temporary resources left hanging around in your account.

I use this in two ways:

  • To dramatically speed up the flow for developing IAM policy documents. It gives me a fast write-test-debug loop for making changes; much faster than if I was using a more full-featured deployment tool like Terraform or CloudFormation.

  • To create a temporary set of tightly-scoped permissions for a risky operation, which act as an extra guard rail.

    This is why I originally wrote the code: I had an admin role that had blanket "Deny" permissions (on an important S3 bucket), but had full IAM permissions, and so could create new roles with arbitrary permissions. I used this function to create a temporary set of credentials which only had permission to delete specific objects, minimising the risk that I'd inadvertently delete the wrong thing.

    If you want to select a subset of the permissions in a role you already have (rather than creating a brand new role/permissions set), consider using IAM session policies. You can apply a policy document when you call AssumeRole, and that policy applies for the duration of your session. (hat tip Ben Kehoe)

How does it work?

The function creates a temporary IAM role, and attaches your policy document as an inline policy. (I considered creating a temporary IAM user, but roles have a 5x limit on the size of inline policies.)

Then it gives your admin role permission to assume the temporary role, assumes it, and gets some credentials using STS. It hands back those credentials for you to use.

When you're done, it cleans up the temporary role and associated policies, so there's nothing left hanging around in your account.

Interesting ideas: what did I learn?

Here are some of the interesting things I learnt while writing this code:

  • Context managers are great for temporary resources. Context managers are a Python feature that let you create a resource, and ensure it gets cleaned up afterwards. An example you've probably used is the open function for files:

    with open('spam.txt', 'r') as f:
        print(f.read())

    The file will be closed when you're done, even if an exception is thrown inside the with block.

    I'm using contextlib.contextmanager to create a couple of my own context managers for temporary IAM resources, so those resources can always be cleaned up afterwards. It goes something like:

    import contextlib
    
    @contextlib.contextmanager
    def temporary_iam_resource(*args, **kwargs):
        # Code to create resource
        resource = create_iam_resource(*args, **kwargs)
        try:
            yield resource
        finally:
            # Code to clean up resource
            delete_iam_resource(resource)
  • ExitStack is a good way to handle nested context managers. This script creates several temporary resources, and the nested context managers start to get unwieldy:

    with temporary_iam_role() as role1:
        with temporary_iam_role() as role2:
            with temporary_iam_role_policy(role1, policy_document):
                with temporary_iam_role_policy(role2, another_policy_document):
                    ...

    I recently read about ExitStack in a blog post by Nikolaus Rath, which gives a way to nest context managers in a cleaner way:

    with contextlib.ExitStack() as es:
        role1 = es.enter_context(temporary_iam_role())
        role2 = es.enter_context(temporary_iam_role())
        es.enter_context(temporary_iam_role_policy(role1, policy_document))
        es.enter_context(temporary_iam_role_policy(role2, another_policy_document))

    Not only does it reduce the amount of indentation, it also lines things up vertically so it's easier to see the similarities between different lines.

  • Changes in IAM take a while to propagate. In particular:

    • There's a delay between creating a role and being able to assume it
    • There's a delay between creating a role, and credentials that use that role being usable

    I have a hard-coded 15 second delay in my function, because that's what worked in my testing.

    These delays shouldn't be surprising -- IAM is a global, distributed system, and changes won't propagate instantly -- but it's the first time I've encountered them, because I don't usually create a role and immediately try to use it.

  • You can use EC2's DescribeRegions API to test IAM credentials. The API call has a DryRun flag, which tells you if the request was authorised without actually making it, and I saw several examples that suggested using it.

    I did try it here, but it wasn't a reliable source of "is this role ready yet?" Sometimes a DescribeRegions call would succeed, then the next call would fail, then the next call would succeed. Consistency in distributed systems is hard.

Motivation: why did I write this?

I work on an archival storage service, which keeps a copy of every object in two S3 buckets (our "permanent storage"). It's important that objects in these buckets are never inadvertently modified or deleted.

Developers have several IAM roles that we use, which give us different permissions within the account (e.g. read-only, billing, developer, admin). Although the latter two roles can usually do almost anything in an account, we have a blanket "Deny" rule that prevents those roles from modifying anything in these permanent storage buckets -- so we can't corrupt the archive by accident.

However, sometimes we do want to delete objects -- for example, objects that were stored in the wrong place.

When we do this, I don't want to remove the blanket "Deny" rule, because that puts the archive at risk -- objects that I don't want to change are now vulnerable to an errant DeleteObject call. Instead, I wanted to create a fine-grained rule that said "let me delete these three objects, but nothing else".

A "Deny" always beats an "Allow" in an IAM policy document, so I can't modify our developer roles to give us these permissions -- but that developer role can create new IAM roles with arbitrary permissions! I wrote this function to create a temporary role with these permissions, which we could then assume to run the deletion. There's less risk of accidentally deleting something that we weren't planning to delete, because there shouldn't be an IAM policy anywhere that allows its deletion.

(We trust each other to use the "create a role that can do anything" permission responsibly. The blanket "Deny" in the roles we use day-to-day is about preventing careless mistakes, not preventing us from ever deleting something.)

It's not until I finished that I realised this could be more general-purpose, and used to experiment with IAM policy documents.

Usage: how can somebody else use this?

Read the code in iam_tester.py, then look at the worked example in example.py. Hopefully there's enough there to get started.

You need an IAM role with administrator access (in particular, one that can create, update and destroy IAM roles).

License

MIT.