Enabling POST/PUT/DELETE on AWS CloudFront?
Asked Answered
O

1

13

In AWS CloudFront I set this within: "Allowed HTTP Methods" in the "Default Cache Behavior Settings" area: GET, HEAD, OPTIONS, PUT, POST, PATCH, DELETE

My CloudFront is linked to an AWS S3 bucket. So I set the AWS S3 CORS configuration to:

<?xml version="1.0" encoding="UTF-8"?>
<CORSConfiguration xmlns="http://s3.amazonaws.com/doc/2006-03-01/">
    <CORSRule>
        <AllowedOrigin>*</AllowedOrigin>
        <AllowedMethod>GET</AllowedMethod>
        <AllowedMethod>PUT</AllowedMethod>
        <AllowedMethod>POST</AllowedMethod>
        <AllowedMethod>DELETE</AllowedMethod>
        <AllowedMethod>HEAD</AllowedMethod>
        <MaxAgeSeconds>3000</MaxAgeSeconds>
        <AllowedHeader>Authorization</AllowedHeader>
    </CORSRule>
</CORSConfiguration>

My current AWS S3 bucket policy is:

{
    "Version": "2008-10-17",
    "Statement": [
        {
            "Sid": "AllowPublicRead",
            "Effect": "Allow",
            "Principal": {
                "AWS": "*"
            },
            "Action": "s3:GetObject",
            "Resource": "arn:aws:s3:::<bucket_name_here>/*"
        }
    ]
}

Unfortunately when run through curl I get:

$ curl -I -s -X POST -H "Origin: www.example.com" [hash_here].cloudfront.net
HTTP/1.1 405 Method Not Allowed
Content-Type: application/xml
Transfer-Encoding: chunked
Connection: keep-alive
Access-Control-Allow-Origin: *
Access-Control-Allow-Methods: GET, PUT, POST, DELETE, HEAD
Access-Control-Max-Age: 3000
Vary: Origin, Access-Control-Request-Headers, Access-Control-Request-Method
Allow: GET, DELETE, HEAD, PUT
Date: Sun, 01 Mar 2015 14:12:26 GMT
Server: AmazonS3
X-Cache: Error from cloudfront
Via: 1.1 5896eef8502a96757950c7d389f2015c.cloudfront.net (CloudFront)
X-Amz-Cf-Id: uBK_gStEvSTWypvU8_YYjtfjC2UzdR3Ff_cDLitMaeUBNZ9AgrSkJg==
Osmen answered 7/3, 2015 at 0:14 Comment(7)
From the response you have posted it looks like the error is returned by S3, not CloudFront. Did you properly setup bucket access permissions so CloudFront could POST there?Corpuscle
I think I did, everything I did is outlined in my question. If I'm missing a step, then point out where and I'll accept the working solution as answer :)Osmen
Take a look here: docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/…. It talks about permissions to read object, but I think you might need to explicitly enable write permission for Origin Access Identity you use for CloudFrontCorpuscle
Samuel, from I can see you did not set up correct bucket policy for your your be*****-frontend S3 bucket. Currently it allows only public reads, while you need to allow posting to it (docs.aws.amazon.com/AmazonS3/latest/dev/…). The easiest way to do this automatically would be to set up Origin Access Identity in your CloudFront configuration; CloudFront will automatically add required permissions to your bucket policy. Cheers, your CloudFront team.Suffragette
Actually @DmitryGuyvoronsky I seem to be approaching this problem incorrectly. I have JavaScript doing GET/POST/PUT/DELETE to a custom REST API on EC2 with failover via ELB. Should I use Route 53 to direct calls to /api/* to ELB, and all other paths to CloudFront? - Or should I use CloudFront's "Custom Origins" (ref) to forward onto my /api/*?Osmen
@SamuelMarks: I think decision is yours, and either approach is fine. You can set up Cache Behaviors so that CloudFront will proxy POST/PUT requests to your origin EC2 server without caching them. This setup has 2 advantages. Firstly, you can use single domain name for your viewers (say, example.com/api is proxied to your EC2 server, while rest of example.com/* is seved as static files from S3 bucket). Secondly, your viewers are likely to have better connection times since CloudFront will try to keep TCP connection open to EC2 and reuse it between viewer requests.Suffragette
Just to throw another option into the mix, I had my cloudfront distribution pointing at an "Origin Path" (folder from the root of the bucket) to make permissions simpler. This also caused the 405 error and so had to create a new bucket instead. It's the Allow: GET, DELETE, HEAD, PUT that gives that away in curl testing.Uncharted
C
5

It may be too late to answer this question. One of the causes of this issue is the Default Root Object . The POST request must be made to the cloudfront root url ("/"). Check the 'Default Root Object' settings in cloudfront. Its value must be empty, and not index.html.

Cling answered 13/10, 2015 at 10:21 Comment(2)
That didn't make a difference in my case, unfortunatelyMortenson
This did fix my issue. If it isn't the answer for everyone, it is at least something that should be in the checklist.Sandbox

© 2022 - 2024 — McMap. All rights reserved.