I have a HTML content loaded from Cloudfront.
Cloudfront adds a header (cloudfront-viewer-country
) with the user's country in some of his Lambda@Edge functions.
I tried to send the user's country to the browser and read it using a JS code.
When I added the code
response.headers['cloudfront-viewer-country'] = request.headers["cloudfront-viewer-country"];
to the function origin_reponse, the response from Cloudfront contained the header cloudfront-viewer-country
with the user's country but this data is not accessible to the JS code on the page.
I also tried to send the country to the browser using a cookie:
let finalCookieArray = [];
const cookiePath = '/';
finalCookieArray.push('cloudfront-viewer-country1='+ countryCode +'; SameSite=Strict; Path=' + cookiePath + '}');
response['headers']['set-cookie'] = [{
'key': 'Set-Cookie',
'value': finalCookieArray
}];
But the problem in this method is that Set-Cookie doesn't work from the function origin_reponse.
Finally I tried to use the code above in the function viewer_reponse but the request sent to the this function doesn't contain the header cloudfront-viewer-country
.
Is there any way to send the country added by cloudfront to the browser in a way that it will be accessible through JS code (preferably using only 1 Lambda@Edge function)?
As you realised, you can only set cookies using viewer_reponse, and you can only access CloudFront-Viewer-Country in origin_reponse or origin_request. So it can not work with cookies. I guess the intention is to pass the information of country to the browser.
Several options:
if your origin is a server that can change the content based on headers, you could configure CF to cache based on CloudFront-Viewer-Country, and pass this header to the origin server, which will serve a different content for each country. It reduces the cache hits, but you can have the country name where you want without relying on browser logic.
if your origin is S3 or any static server, you could use an origin_request to modify the URL and add ?c=[country] to it. And then read that parameter from the url using Javascript.
See here how to modify: https://aws.amazon.com/blogs/networking-and-content-delivery/handling-redirectsedge-part1/
And how to read using JS:
This can be tricky as you have to cache based on URL parameters and that can really reduce the cache hits.
if you do not mind an extra AJAX call (and having the information async), you could define a specific url in CloudFront, that calls an origin response lambda function. This function will intercept the response and replace it with a JSON array containing the contry code. This allows you to maintain the cache hits on the main content.
one other way would be to use geo detection API to do that for you: https://ipstack.com/ https://ipapi.com/
If you do still want to set a cookie with the country, you can do it with JavaScript once you get the information with one of the above-mentioned methods.