<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Will Boland</title>
    <description>The latest articles on DEV Community by Will Boland (@wboland33).</description>
    <link>https://dev.to/wboland33</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F410570%2F4d79ed1a-12fa-4a34-b3ea-d0554679ab88.jpg</url>
      <title>DEV Community: Will Boland</title>
      <link>https://dev.to/wboland33</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/wboland33"/>
    <language>en</language>
    <item>
      <title>My Experience with the Cloud Resume Challenge</title>
      <dc:creator>Will Boland</dc:creator>
      <pubDate>Fri, 19 Jun 2020 14:18:10 +0000</pubDate>
      <link>https://dev.to/wboland33/my-experience-with-the-cloud-resume-challenge-1ade</link>
      <guid>https://dev.to/wboland33/my-experience-with-the-cloud-resume-challenge-1ade</guid>
      <description>&lt;p&gt;I completed the &lt;a href="https://cloudresumechallenge.dev/"&gt;Cloud Resume Challenge&lt;/a&gt; and lived to tell about it.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--KasUqZq9--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/qe2t5429mftt9ylh74pa.jpg" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--KasUqZq9--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/qe2t5429mftt9ylh74pa.jpg" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;With no prior experience or education in technology, the tools required to successfully master this challenge were completely foreign to me. The &lt;a href="https://www.williamsboland.com/"&gt;finished product&lt;/a&gt; - which required several weeks of early morning study, research, coding, and troubleshooting - was difficult but fun to bring to life. Opening myself up to the tech world has certainly been a rewarding experience, so I'm grateful to Forrest for the challenge!&lt;/p&gt;

&lt;p&gt;This wrap-up post provides an overview of my experience with the challenge and a few tips for others tackling the project. For anyone like me intimidated by this challenge (or already stumbling through it): &lt;strong&gt;don't panic&lt;/strong&gt;. You'll be surprised by what you can accomplish!&lt;/p&gt;




&lt;h4&gt;
  
  
  Jump to Section
&lt;/h4&gt;

&lt;ul&gt;
&lt;li&gt;Why I Took the Challenge&lt;/li&gt;
&lt;li&gt;Challenge Overview&lt;/li&gt;
&lt;li&gt;Taking the AWS Certified Cloud Practitioner Exam&lt;/li&gt;
&lt;li&gt;Building the Resume Site&lt;/li&gt;
&lt;li&gt;Building the Visitor Count Functionality&lt;/li&gt;
&lt;li&gt;Creating Pipelines for CI/CD&lt;/li&gt;
&lt;li&gt;Summary&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Why I Took the Challenge &lt;a&gt;&lt;/a&gt;
&lt;/h2&gt;

&lt;p&gt;After spending the past nine years growing an &lt;a href="https://www.carlotz.com/"&gt;auto retail startup&lt;/a&gt; I founded with two remarkable business partners, I decided to make a change and recently transitioned to a board level role in pursuit of new ventures. While my background in finance and experience founding a company have honed my analytical and leadership skills, I've determined that improving my talent stack with technical skills will help me in my next venture, whether that's launching a new company or joining an existing one. In an effort to better understand cloud and voice technologies - areas that are particularly compelling to me - I began following a few interesting folks in the space on Twitter and stumbled upon &lt;a href="https://forrestbrazeal.com/"&gt;Forrest Brazeal&lt;/a&gt;, the benefactor (aka evil genius) behind the Cloud Resume Challenge. &lt;/p&gt;


&lt;blockquote class="ltag__twitter-tweet"&gt;

  &lt;div class="ltag__twitter-tweet__main"&gt;
    &lt;div class="ltag__twitter-tweet__header"&gt;
      &lt;img class="ltag__twitter-tweet__profile-image" src="https://res.cloudinary.com/practicaldev/image/fetch/s--6KKGBBk8--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://pbs.twimg.com/profile_images/1259236310990061575/8poErtri_normal.jpg" alt="Forrest Brazeal profile image"&gt;
      &lt;div class="ltag__twitter-tweet__full-name"&gt;
        Forrest Brazeal
      &lt;/div&gt;
      &lt;div class="ltag__twitter-tweet__username"&gt;
        &lt;a class="comment-mentioned-user" href="https://dev.to/forrestbrazeal"&gt;@forrestbrazeal&lt;/a&gt;

      &lt;/div&gt;
      &lt;div class="ltag__twitter-tweet__twitter-logo"&gt;
        &lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--P4t6ys1m--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://practicaldev-herokuapp-com.freetls.fastly.net/assets/twitter-f95605061196010f91e64806688390eb1a4dbc9e913682e043eb8b1e06ca484f.svg" alt="twitter logo"&gt;
      &lt;/div&gt;
    &lt;/div&gt;
    &lt;div class="ltag__twitter-tweet__body"&gt;
      Introducing the &lt;a href="https://twitter.com/hashtag/CloudResumeChallenge"&gt;#CloudResumeChallenge&lt;/a&gt;. I'm volunteering my network to help you get your first job in the cloud. But I can only share a certain kind of resume. Thread -&amp;gt; &lt;a href="https://t.co/FRUuzBupXr"&gt;forrestbrazeal.com/2020/04/23/the…&lt;/a&gt;
    &lt;/div&gt;
    &lt;div class="ltag__twitter-tweet__date"&gt;
      17:19 PM - 27 Apr 2020
    &lt;/div&gt;


    &lt;div class="ltag__twitter-tweet__actions"&gt;
      &lt;a href="https://twitter.com/intent/tweet?in_reply_to=1254822417203113986" class="ltag__twitter-tweet__actions__button"&gt;
        &lt;img src="/assets/twitter-reply-action.svg" alt="Twitter reply action"&gt;
      &lt;/a&gt;
      &lt;a href="https://twitter.com/intent/retweet?tweet_id=1254822417203113986" class="ltag__twitter-tweet__actions__button"&gt;
        &lt;img src="/assets/twitter-retweet-action.svg" alt="Twitter retweet action"&gt;
      &lt;/a&gt;
      75
      &lt;a href="https://twitter.com/intent/like?tweet_id=1254822417203113986" class="ltag__twitter-tweet__actions__button"&gt;
        &lt;img src="/assets/twitter-like-action.svg" alt="Twitter like action"&gt;
      &lt;/a&gt;
      142
    &lt;/div&gt;
  &lt;/div&gt;
&lt;/blockquote&gt;


&lt;p&gt;The project sounded fun, provided a practical complement to my conceptual learning, and offered exposure to people and companies doing work that aligns with my personal interest in cloud and voice technologies. Why not give it a shot?&lt;/p&gt;




&lt;h2&gt;
  
  
  Challenge Overview &lt;a&gt;&lt;/a&gt;
&lt;/h2&gt;

&lt;p&gt;The challenge in and of itself sounds straightforward: create a simple resume site with a visitor count feature using &lt;a href="https://aws.amazon.com/"&gt;Amazon Web Services (AWS)&lt;/a&gt;. Easy enough, right?&lt;/p&gt;

&lt;p&gt;Well, Forrest provides &lt;a href="https://cloudresumechallenge.dev/instructions/"&gt;specific instructions&lt;/a&gt; in the form of 16 distinct steps that must be completed to successfully meet the challenge. These steps might be a breeze for tech professionals with AWS experience, but for a novice like me...&lt;/p&gt;

&lt;p&gt;&lt;a href="https://i.giphy.com/media/3o6MblPatFjEETvBHa/source.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://i.giphy.com/media/3o6MblPatFjEETvBHa/source.gif" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Each step led me down a rabbit hole of tutorials, Googling, coding, and troubleshooting that took me on a roller coaster of emotions from frustration and despair when my fixes wouldn't work to elation when something finally clicked.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://i.giphy.com/media/l2JeamGwPGlc6K0lW/source.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://i.giphy.com/media/l2JeamGwPGlc6K0lW/source.gif" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  Taking the AWS CCP Exam &lt;a&gt;&lt;/a&gt;
&lt;/h2&gt;

&lt;p&gt;As a first step, I spent a few weeks preparing for the &lt;a href="https://aws.amazon.com/certification/certified-cloud-practitioner/"&gt;AWS Certified Cloud Practitioner&lt;/a&gt; exam. Given the scope of services offered by AWS, I was a little overwhelmed at first, but it eventually came together. Overall, I found getting certified before building the resume site helpful because it provided a strong understanding of various AWS services I used in completing the challenge.&lt;/p&gt;

&lt;p&gt;Below are the steps I took to prepare:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Took &lt;a href="https://aws.amazon.com/training/course-descriptions/cloud-practitioner-essentials/"&gt;AWS Cloud Essentials Course&lt;/a&gt;. This is a good starting point because it's AWS official, free, and takes about 6 hours.&lt;/li&gt;
&lt;li&gt;Reviewed the overview pages for the key AWS resources covered on the exam and took notes for future reference. This &lt;a href="https://www.youtube.com/watch?reload=9&amp;amp;v=KFhXKlX6p-o"&gt;YouTube video&lt;/a&gt; provides a helpful index of links.&lt;/li&gt;
&lt;li&gt;Took &lt;a href="https://acloud.guru/learn/aws-certified-cloud-practitioner"&gt;A Cloud Guru - CCP Course&lt;/a&gt;. This added a helpful layer to the AWS Cloud Essentials Course because it's specifically focused on helping pass the exam and includes some labs that got me playing around in my AWS account. They currently have a 7-day free trial you can use to access the course, though I ended up purchasing an annual subscription because I intend to pursue additional certifications.&lt;/li&gt;
&lt;li&gt;Read the following AWS guides and whitepapers:

&lt;ul&gt;
&lt;li&gt;&lt;a href="https://d0.awsstatic.com/whitepapers/aws-overview.pdf"&gt;Overview of Amazon Web Services&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://d1.awsstatic.com/whitepapers/architecture/AWS_Well-Architected_Framework.pdf"&gt;AWS Well-Architected Framework&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://d0.awsstatic.com/whitepapers/aws_pricing_overview.pdf"&gt;How AWS Pricing Works&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://aws.amazon.com/compliance/shared-responsibility-model/"&gt;AWS Shared Responsibility Model&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://aws.amazon.com/premiumsupport/plans/"&gt;Compare AWS Support Plans&lt;/a&gt;&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;li&gt;Did tons of practice questions and exams. This was the most helpful study tactic, as it helped me identify my weak areas so I could attack those with additional study. AWS offers some free questions, and the ACG - CCP course provided practice questions at the end of each section along with a comprehensive final exam. That said, I found the two paid resources below to be the most helpful. They each cost $15, and their content was quite different so complemented each other well.

&lt;ul&gt;
&lt;li&gt;&lt;a href="https://digitalcloud.training/aws-certified-cloud-practitioner-practice-tests/"&gt;Digital Cloud Training: CCP Practice Exams &amp;amp; Exam Simulator&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://portal.tutorialsdojo.com/courses/aws-certified-cloud-practitioner-practice-exams/"&gt;Tutorials Dojo: CCP Practice Exams&lt;/a&gt;&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;That might be overkill, but mission accomplished.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--Tht9vR_C--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/h7qmvap8rrb8ze78qlly.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--Tht9vR_C--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/h7qmvap8rrb8ze78qlly.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;em&gt;Step 1 down, 15 to go...&lt;/em&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  Building the Resume Site &lt;a&gt;&lt;/a&gt;
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Creating resume content in HTML/CSS
&lt;/h3&gt;

&lt;p&gt;Since the resume site needed to be written in HTML and styled with CSS, I took a helpful &lt;a href="https://www.khanacademy.org/computing/computer-programming/html-css"&gt;HTML/CSS course from Khan Academy&lt;/a&gt; to get a decent grasp on webpage fundamentals. The initial resume site I built from scratch was functional but had a face only a mother could love.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://i.giphy.com/media/5nkIn9AEfUQ6JtXL43/source.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://i.giphy.com/media/5nkIn9AEfUQ6JtXL43/source.gif" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Fortunately, I found a &lt;a href="https://www.sonjastrieder.com/"&gt;resume template&lt;/a&gt; I loved. Using &lt;a href="https://notepad-plus-plus.org/downloads/"&gt;Notepad++&lt;/a&gt; along with my newfound HTML/CSS knowledge, I adapted the template to my needs and made my resume site much easier on the eyes.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--IY-RKzzx--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/1jar514abhyml09afjkd.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--IY-RKzzx--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/1jar514abhyml09afjkd.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;em&gt;Steps 2 + 3 down, 13 to go...&lt;/em&gt;&lt;/p&gt;

&lt;h3&gt;
  
  
  Deploying static S3 site with custom domain and HTTPS
&lt;/h3&gt;

&lt;p&gt;Once I had my site content ready, I needed a place to host it publicly on the web. Enter &lt;a href="https://aws.amazon.com/s3/"&gt;Amazon Simple Storage Service (S3)&lt;/a&gt;. The first step is to create an S3 bucket and enable it to host a static website:&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--Iua_J4Rw--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/qwi2vc9bl1knw4at0tuf.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--Iua_J4Rw--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/qwi2vc9bl1knw4at0tuf.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;After uploading my HTML/CSS files, I had a working site, and with a URL like this, who who wouldn't want to visit?&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--Q8iVoU69--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/mn7czokvq7yzkce6orjc.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--Q8iVoU69--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/mn7czokvq7yzkce6orjc.PNG" alt="Alt Text"&gt;&lt;/a&gt; &lt;/p&gt;

&lt;p&gt;To make it a &lt;em&gt;little&lt;/em&gt; easier for visitors to find my site, I used &lt;a href="https://aws.amazon.com/route53/"&gt;Route 53&lt;/a&gt; to purchase a domain.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--a4PRK05c--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/m7p0sz0tpcsoiecuozcp.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--a4PRK05c--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/m7p0sz0tpcsoiecuozcp.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;From there, I used &lt;a href="https://aws.amazon.com/certificate-manager/"&gt;AWS Certificate Manager&lt;/a&gt; to secure the site with SSL so it could be served over HTTPS and then set up a distribution through &lt;a href="https://aws.amazon.com/cloudfront/"&gt;CloudFront&lt;/a&gt;, AWS' content delivery network.&lt;/p&gt;

&lt;p&gt;CloudFront uses cached content at AWS edge locations to improve user latency (ie, the time it takes for users around the world to access your site content). As such, when you update your site content down the road, it may take up to the default cache time to live of 24 hours before CloudFront begins serving your new content. If you want to serve the new content immediately, you'll need to invalidate your CloudFront distribution.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--uHrP2MQJ--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/ebq8il83003isuh3nrgf.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--uHrP2MQJ--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/ebq8il83003isuh3nrgf.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Some Tips if you're Troubleshooting:&lt;/strong&gt; &lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Make sure your S3 bucket has the same name as your custom domain.&lt;/li&gt;
&lt;li&gt;When you create your static site bucket, make sure it's public, and also make sure any objects your upload to the bucket are public.&lt;/li&gt;
&lt;li&gt;Check your DNS records and make sure the naming conventions match CloudFront.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;em&gt;Steps 4 + 5 + 6 down, 10 to go...&lt;/em&gt;&lt;/p&gt;

&lt;h3&gt;
  
  
  Placing visitor count on page using Javascript
&lt;/h3&gt;

&lt;p&gt;While my site was live and looking good, I still needed a way to get the required visitor count on my page. Specifically, I would need JavaScript to call the latest visitor count from an API endpoint that I set up in subsequent steps. &lt;a href="https://www.codecademy.com/learn/introduction-to-javascript"&gt;This tutorial&lt;/a&gt; was a helpful introduction to JavaScript, but it took Googling around a bit to find the right method to meet my specific use case. I experimented until I landed on something that would hopefully work, uploaded my JavaScript code, referenced it in my HTML, and moved onto the next steps. When I later got my API working, I came back to my JavaScript, pasted in the API endpoint, and &lt;em&gt;voila&lt;/em&gt;, I had a dynamic visitor count.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--thW9hyXk--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/1u52fkpe0aehj7dqlbma.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--thW9hyXk--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/1u52fkpe0aehj7dqlbma.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;a href="https://i.giphy.com/media/12NUbkX6p4xOO4/source.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://i.giphy.com/media/12NUbkX6p4xOO4/source.gif" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Tip if you're Troubleshooting:&lt;/strong&gt; &lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;For your JavaScript, take a look at FetchAPI and XMLHttpRequest.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;em&gt;Step 7 down, 9 to go...&lt;/em&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  Building the Visitor Count Functionality &lt;a&gt;&lt;/a&gt;
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Using AWS Lambda, DynamoDB, and API Gateway to create visitor count function
&lt;/h3&gt;

&lt;p&gt;We now enter the tricky phase. The steps I describe at a high level below do no justice to the cringe-worthy amount of tutorials, Googling, and trial &amp;amp; error I performed before I had everything working properly. The feeling of elation when everything started to click, however, was well worth it.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://i.giphy.com/media/Uno27COfoYlH2/source.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://i.giphy.com/media/Uno27COfoYlH2/source.gif" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Created a &lt;a href="https://aws.amazon.com/dynamodb/"&gt;DynamoDB&lt;/a&gt; table to update and store the number of visitors to my resume site. This was fairly straightforward once I figured out the primary key and attribute conventions.
&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--z4GtmLFN--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/0o5wq7emr4nancoby980.PNG" alt="Alt Text"&gt;
&lt;/li&gt;
&lt;li&gt;Built a &lt;a href="https://aws.amazon.com/lambda/"&gt;Lambda&lt;/a&gt; function using &lt;a href="https://www.python.org/"&gt;Python&lt;/a&gt; and the AWS SDK for Python called &lt;a href="https://boto3.amazonaws.com/v1/documentation/api/latest/index.html"&gt;boto3&lt;/a&gt;. When triggered, the Lambda function updates the number of visitors stored in my DynamoDB table by exactly 1. This was one of the most fun parts of the process, and I'm excited to continue learning Python.&lt;/li&gt;
&lt;li&gt;Implemented testing for my Python code. There are several ways to do this, but I used &lt;a href="https://docs.python.org/2/library/unittest.html"&gt;unittest&lt;/a&gt;.&lt;/li&gt;
&lt;li&gt;Created an API using &lt;a href="https://docs.aws.amazon.com/apigateway/index.html"&gt;API Gateway&lt;/a&gt; that sits in between my front-end website and my back-end Lambda function. Using the API's invoke URL, my JavaScript code calls my API whenever someone visits my website, and the API immediately triggers my Lambda function to update the number of visitors in my DynamoDB table and return the updated count to my API, which JavaScript then displays on my site. Super cool.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;&lt;strong&gt;Tip if you're Troubleshooting:&lt;/strong&gt; &lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;If you're struggling to get your API endpoint to work, try enabling Cors and adjusting your Lambda return so it passes the correct headers back in its response (you need to do both when using a proxy integration for your API).&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;em&gt;Steps 8 + 9 + 10 + 11 down, 5 to go...&lt;/em&gt;&lt;/p&gt;

&lt;h3&gt;
  
  
  Defining back-end AWS resources in SAM template and deploying using SAM CLI
&lt;/h3&gt;

&lt;p&gt;While you could configure the AWS resources above (DynamoDB, Lambda, and API Gateway) individually in the AWS console, a best practice would be to configure and deploy them as a unified application using the &lt;a href="https://github.com/awslabs/aws-sam-cli"&gt;SAM CLI&lt;/a&gt;. Figuring out how to define my AWS resources in a SAM template.yaml took some time, but once that code was cracked, deploying and updating using the SAM CLI was super easy and demonstrated the power of &lt;a href="https://en.wikipedia.org/wiki/Infrastructure_as_code"&gt;Infrastructure as Code&lt;/a&gt;. Watching my application come to life in &lt;a href="https://aws.amazon.com/cloudformation/"&gt;CloudFormation&lt;/a&gt; after deploying with AWS CLI was awesome!&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Tips if you're Troubleshooting:&lt;/strong&gt; &lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;When defining your Lambda function in the SAM template.yaml file, use the convention "filename.functionname" for your Handler property so your SAM template references your Python function appropriately. For example, for a Python file named "lambdafunction.py" with the Lambda function inside defined as "handler", your Handler property in the SAM template would be "lambdafunction.handler". This tripped me up initially.&lt;/li&gt;
&lt;li&gt;Finding the right API invoke URL can be tricky, but if you define it as an Output in your SAM template.yaml file, you can access it more easily post-deployment.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--1eLJ3wdE--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/dk6gu164srstx9a7nzk3.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--1eLJ3wdE--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/dk6gu164srstx9a7nzk3.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;em&gt;Step 12 down, 4 to go...&lt;/em&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  Creating Pipelines for CI/CD &lt;a&gt;&lt;/a&gt;
&lt;/h2&gt;

&lt;p&gt;This was one of the most fun and most useful steps in the process. While I had a working application tying together my front-end resume website and back-end visitor count function code, I would certainly need to make updates. Setting up a CI/CD pipeline would allow me to update my code from an online repository and automatically push those updates to my front-end website and back-end app/infrastructure.&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Created a &lt;a href="https://github.com/"&gt;GitHub&lt;/a&gt; account and set up two private code repositories, one for my front-end (HTML, CSS, JavaScript) and one for my back-end (Python function/tests, SAM templates), and uploaded my files.
&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--u26C25dU--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/0x4kxi1bmr918rd2jo4j.PNG" alt="Alt Text"&gt;
&lt;/li&gt;
&lt;li&gt;Set up a continuous deployment pipeline using &lt;a href="https://aws.amazon.com/codepipeline/"&gt;AWS CodePipeline&lt;/a&gt; using my front-end code repository on GitHub as the source. Now, whenever I make updates to my HTML/CSS/JavaScript code on GitHub, CodePipeline automatically deploys those changes to my S3 bucket.&lt;/li&gt;
&lt;li&gt;Set up a second pipeline using CodePipeline, this time using my back-end code repository as the source. This pipeline was trickier than the front-end because I added a build stage that would test and package my application before deploying to production, but now that this is setup, my updates to my back-end code in GitHub automatically test and deploy!&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--84nSaXWO--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/d30bw3rllvaqt2d4nhzy.PNG" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--84nSaXWO--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/i/d30bw3rllvaqt2d4nhzy.PNG" alt="Alt Text"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Tip if you're Troubleshooting:&lt;/strong&gt; &lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;If you're using CodePipeline for continuous deployment, be sure to check the "enable logs" box when setting up your pipeline, as it will provide more detailed insight into any pipeline errors/failures. My pipeline failed repeatedly during the test stage, leading me to spend even more time troubleshooting my buildspec.yml, which ultimately proved unnecessary, as once I enabled build logs in my pipeline settings, I was able to see exactly why my builds were failing: my CodePipeline service role did not have the proper permissions to interact with my DynamoDB table. Easy fix from there.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;em&gt;Steps 13 + 14 + 15 down, just this blog post to go!&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;&lt;a href="https://i.giphy.com/media/3o7buiQeyYFamzRoR2/source.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://i.giphy.com/media/3o7buiQeyYFamzRoR2/source.gif" alt="Alt Text"&gt;&lt;/a&gt; &lt;/p&gt;




&lt;h2&gt;
  
  
  Summary &lt;a&gt;&lt;/a&gt;
&lt;/h2&gt;

&lt;p&gt;Overall, the Cloud Resume Challenge was a fun, engaging project with "challenge" being the operative word. I learned a lot about cloud services, infrastructure-as-code, serverless applications, CI/CD, and some programming skills. I also learned a lot about myself, namely that a beginner's mindset is powerful and that I'm excited to keep going. Thanks again to Forrest Brazeal for putting this out there.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://i.giphy.com/media/JlpjgShzDsrMIyFu5U/source.gif" class="article-body-image-wrapper"&gt;&lt;img src="https://i.giphy.com/media/JlpjgShzDsrMIyFu5U/source.gif" alt="Alt Text"&gt;&lt;/a&gt; &lt;/p&gt;

&lt;p&gt;It's clear from Forrest's challenge and from the various resources I came across during the process that the tech community is open, supportive, and encouraging. To that end, if you're working through this challenge, good luck, and I hope this post was helpful. If you're still considering the challenge, I encourage you to give it a shot.&lt;/p&gt;

&lt;p&gt;If you would like to connect or learn more about my experience with the Cloud Resume Challenge, please comment here or find me on &lt;a href="https://www.linkedin.com/in/will-boland-b80b694/"&gt;LinkedIn&lt;/a&gt; or &lt;a href="https://twitter.com/WillBoland"&gt;Twitter&lt;/a&gt;.&lt;/p&gt;

</description>
      <category>aws</category>
      <category>serverless</category>
      <category>python</category>
      <category>beginners</category>
    </item>
  </channel>
</rss>
