mw casual engineeringcover image

Understanding AWS security policies

cloud AWS security

Some days ago I was (again) trying to fiddle with AWS security policies trying to create a user with only the rights to upload to a specific S3 "folder"1. I have already attempted this earlier before but with no success, in part because there is just so little documentation about how AWS policies work and also in part because of my failure to realize that the tool I use to access S3 - rclone also requires some permissions other than PutObject — namely ListBucket and GetObject. In this article, I will go over how to understand and make use of AWS (specifically S3) security policies.

AWS permission model

In the AWS world, clients use an access key to call various AWS APIs, and access keys belong to users. You can create users (and generate an initial access key) on the Users page of the IAM console.

A user's permission is controlled by the policies attached to them (or any of the groups that they belong to). There are a lot of pre-existing policies for simple cases, such as granting a user full control of a service, etc. For example, on the “add user” page, when you get to “Set permissions”, you can see a list of AWS pre-defined policies, and in it you can find AmazonS3FullAccess and AmazonS3ReadOnlyAccess, which is exactly what they sound like: allowing full read or read/write access to every bucket.

A screenshot showing the policies displayed under the "Set permissions" tab, with a search filter of "S3". The policies displayed are: AmazonDMSRedshiftS3Role, AmazonS3FullAccess, AmazonS3ReadOnlyAccess, and QuickSightAccessForS3StorageManagementAnalyticsReadOnly, all of which are of type AWS managed.

Creating policies

Usually, if you want something more fine-tuned, such as specifying which buckets are allowed to be accessed, you have to create your own policy. This can be done with the "Create policy" button here, or you can also go to the dedicated policies page to manage all your policies. The visual policy editor is probably good enough that you don't need to write a JSON policy yourself.

A screenshot of the initial state of the visual policy editor, with no permissions entered.

In your policy, you can have one or more permissions (known as "statements" in the JSON), each one can either grant some access or be an explicit deny, which explicitly declares that some operations should not be allowed. AWS first evaluates all explicit deny statements, and if any matches, the operation is denied whether or not some other statements allow the action. You can use the "add additional permissions" button to add more permissions (statements) to your policy.

Each permission targets a set of actions, which themselves correspond to various API calls. For example, if you choose "S3" as the "Service" and in "Actions" choose GetObject, this makes the permission allow (or deny) reading S3 files. If you choose the GetObject action, you need to then specify which files to allow/deny the user to read in the "Resources" section.

A screenshot of the "Resource" section for a newly created permission statement with action "GetObject". It reads: object: Specify object resource ARN for the GetObject action.

ARN is, as far as I understand, a fancy name for “the absolute path of a thing”. You can use ARN to refer to various things (such as S3 buckets, S3 files (aka objects), etc.) in AWS, and in this case, we need to provide an ARN for an S3 object, which has the form:

arn:aws:s3:::bucket_name/file_path

So, for example, if you have a file named data/data.txt in the myfiles bucket, its ARN would be arn:aws:s3:::myfiles/data/data.txt.

You can add multiple ARNs, and the permission is applied if any of them match. You can also use asterisks (*) in the ARN to denote "any string." Therefore, if you want to allow read access to everything under the data "folder" (and its subfolders), you would enter the ARN arn:aws:s3:::myfiles/data/*.

All of your ARNs will apply to all of the actions that you specified in the same permission. So, if you have a permission that grants PutObject and GetObject, and with an ARN of arn:aws:s3:::myfiles/data/data.txt, users with this permission would be able to read, write to, and create the file named data/data.txt. To allow/deny different ARNs for different actions, you can create multiple permission statements.

You can also add request conditions, which allow you to limit what the parameters for the API calls can be. For example, if you have a permission that grants ListBucket, having a request condition of s3:prefix StringLike data/* will limit the user to only being able to list the contents of the data "folder" (and its subfolders). You can see a list of request conditions (and resource types) for each action by clicking on the icon next to the actions checkbox. Some other interesting conditions include aws:CurrentTime and aws:SourceIp.

Example policies for S3

You should always test out these policies before using them in production!

You can import these policies by copying the JSON and pasting them into the JSON editor. After doing that, you may go back to the visual editor to change the statements.

Allow read/write to any objects that are not in the denied folder:

{
    "Version": "2012-10-17",
    "Statement": [
        {
            "Sid": "AllowAll",
            "Effect": "Allow",
            "Action": [
                "s3:PutObject",
                "s3:GetObject",
                "s3:DeleteObject"
            ],
            "Resource": [
                "arn:aws:s3:::secpolicy-test/*"
            ]
        },
        {
            "Sid": "ExecptDenied",
            "Effect": "Deny",
            "Action": [
                "s3:PutObject",
                "s3:GetObject",
                "s3:DeleteObject"
            ],
            "Resource": "arn:aws:s3:::secpolicy-test/denied/*"
        }
    ]
}

The first statement allows read, write and overwriting any files, but the second statement denies these operations for anything with a name starting denied/. Since there is no actual concept of "folders" in S3, you don't have to worry about the user being allowed to delete the denied folder. They can, however, create another regular file named denied and write/delete that file.

For some clients (such as rclone), you also need to grant the user s3:ListBucket and s3:ListAllMyBuckets for them to successfully upload. Hence for those clients the policy should look like this:

Update: In newer versions of rclone you need to pass --s3-no-check-bucket when uploading, otherwise it will attempt to call the bucket creation endpoint, and if that fails with a permission error rclone will fail.

{
    "Version": "2012-10-17",
    "Statement": [
        {
            "Sid": "AllowAll",
            "Effect": "Allow",
            "Action": [
                "s3:ListBucket",
                "s3:PutObject",
                "s3:GetObject",
                "s3:DeleteObject"
            ],
            "Resource": [
                "arn:aws:s3:::secpolicy-test",
                "arn:aws:s3:::secpolicy-test/*"
            ]
        },
        {
            "Sid": "AllowListAllMyBuckets",
            "Effect": "Allow",
            "Action": "s3:ListAllMyBuckets",
            "Resource": "*"
        },
        // Other statements e.g. deny certain subdirectories
    ]
}

Note that there are two resources specified in the AllowAll statement: the bucket secpolicy-test and the S3 objects * in secpolicy-test. Since {Put,Get,Delete}Object only takes an object resource, they will ignore the first specified resource. Similarly, ListBucket only takes a bucket resource, so it will ignore the second resource specified.

Allow user all access to everything under the user folder:

{
    "Version": "2012-10-17",
    "Statement": [
        {
            "Sid": "list",
            "Effect": "Allow",
            "Action": "s3:ListBucket",
            "Resource": "arn:aws:s3:::secpolicy-test",
            "Condition": {
                "ForAllValues:StringLike": {
                    "s3:prefix": "user/*"
                }
            }
        },
        {
            "Sid": "objectOps",
            "Effect": "Allow",
            "Action": [
                "s3:GetObject*",
                "s3:PutObject*",
                "s3:DeleteObject*"
            ],
            "Resource": "arn:aws:s3:::secpolicy-test/user/*"
        }
    ]
}

rclone specific: Allow rclone copy into /uploads

{
    "Version": "2012-10-17",
    "Statement": [
        {
            "Sid": "stat",
            "Effect": "Allow",
            "Action": [
                "s3:ListBucket",
                "s3:PutObject",
                "s3:GetObject"
            ],
            "Resource": [
                "arn:aws:s3:::secpolicy-test",
                "arn:aws:s3:::secpolicy-test/uploads/*"
            ]
        }
    ]
}

Due to the use of the HEAD request to determine existence, rclone needs {Put, Get}Object and ListBucket for uploads. When debugging, you can use -vv --dump headers to see what requests rclone is making.

Footnote

« home