Skip to content

Commit

Permalink
add serverless blog post
Browse files Browse the repository at this point in the history
  • Loading branch information
Kent C. Dodds committed Mar 18, 2019
1 parent aebf789 commit 23450a1
Show file tree
Hide file tree
Showing 10 changed files with 211 additions and 2 deletions.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
209 changes: 209 additions & 0 deletions content/blog/super-simple-start-to-serverless/index.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,209 @@
---
slug: 'super-simple-start-to-serverless'
title: 'Super simple start to serverless'
date: '2019-03-18'
author: 'Kent C. Dodds'
description:
'_Easily create server code without worrying about managing servers yourself_'
categories:
- 'node'
keywords:
- 'serverless'
- 'servers'
- 'nodejs'
- 'netlify'
- 'lambda'
- 'functions'
banner: './images/banner.jpg'
bannerCredit:
'Photo by [Klemen Vrankar](https://unsplash.com/photos/F_7Z1A6Jp_s)'
---

This last little while I've been doing a fair amount of work on
[kentcdodds.com](https://kentcdodds.com). One page that I've been working on is
the [contact page](https://kentcdodds.com/contact) where you can
[request enterprise training](https://kcd.im/request-a-workshop),
[schedule an hour of my time for consulting](https://kcd.im/consulting), or even
[leave your own testimonial of my work](https://kcd.im/testimonial) (I'd love it
if you did!).

My app is built with [GatsbyJS](https://www.gatsbyjs.org/) which has quickly
become one of my favorite pieces of technology. Gatsby is a static site
generator, so it generates a bunch of HTML, CSS, and JavaScript files in a
directory called `public` in my project and it leaves it to me to get those
files on the World Wide Web for you and your loved ones to enjoy.

So how do I get those files up there on the internet? I use
[Netlify](https://www.netlify.com/)! I've LOVED Netlify for years. It's bonkers
how quickly I can get a site up and running (with HTTPS) with Netlify and it's
integration with GitHub is unmatched. I love it.

Great, so now I've got static files served ultra-fast on the internet with
Netlify's CDN. But how do I make it so my contact form can actually send emails?
You may not be aware of this, but web browsers are not able to send emails
themselves. To do that you need a server. I don't know about you, but managing
servers and databases are among the top things on the list of stuff I do NOT
enjoy about web development.

You may have heard of "Serverless" or "Lambda Functions" or "Azure Functions"
etc. What's cool about this is you can write your server code without worrying
about managing servers or paying to have them sit around waiting for requests
all day long (which is particularly useful for a contact form like mine).

I remember years ago when those starting getting popular, I tried them out a bit
and was reminded why I don't enjoy working with backend/devops stuff. As cool as
these functions were conceptually, I just didn't feel like I was smart enough to
navigate the complexities of the offerings.

Luckily for us, almost exactly one year ago from the time I'm writing this,
Netlify took the complexity of AWS Lambda functions and made it super simple (I
think it's what they do best):

https://twitter.com/netlify/status/976126036491386880

I want to give you a really quick intro to using Netlify to deploy server code
in a way that is mind-blowingly straightforward to me.

## How to use Netlify Lambda Functions

Before you get started, you'll need a [GitHub](https://github.com) account, a
[Netlify](https://netlify.com) account, and a fresh repository. Once you've got
the project locally on your computer, then you can start with the following
steps:

### Step 1: Create a `netlify.toml` file

This file configures some stuff about our repository for netlify. Here's all you
need for this to work:

```toml
[build]
functions = "./functions"
```

All this is saying is: Hey Netlify, when you build my project, the functions are
located in the "functions" directory.

### Step 2: Create `functions/hello.js`

Now that we've told Netlify where to find our functions files, let's make one:

```javascript
// functions/hello.js
exports.handler = async event => {
const subject = event.queryStringParameters.name || 'World'
return {
statusCode: 200,
body: `Hello ${subject}!`,
}
}
```

Netlify uses AWS Lambda under the hood, so you can checkout documentation and
learning materials about creating AWS Lambda functions for what kind of code can
go in there. We'll start with something simple that returns a promise (note it's
an `async` function) which resolves to an object for the response we want for
this function. You can do all kinds of things in that function (like connect
with an SMTP mail server and send an email which is what I do in my contact
form).

### Step 3: Commit and push

![The above code in an editor](./images/code.png)

Now you can just do: `git add -A`, then `git commit -m 'init'`, and
`git push -u origin master` and you're set! Now let's go over to Netlify.

![The github repository](./images/repo.png)

### Step 4: Setup Netlify

Go to [your netlify sites](https://app.netlify.com/) and click "New site from
Git." From there you'll connect your Netlify account to your GitHub account and
it'll show you a list of repositories. Search for the one you want and click on
it.

![Create New Site page](./images/create-new-site.png)

Go ahead and leave the "Build command" and "Publish directory" fields empty and
ignore the advanced options. Just click "Deploy site."

![Overview page](./images/overview.png)

Now you're on the Overview page of your site. It should only take a few seconds
to deploy your site. You can see info about that on the "Deploys" page. While
you wait, you can edit the Site name to be something more memorable by clicking
"Site settings."

Once the deploy is finished, you can go to the "Functions" page:

![Functions page showing hello.js](./images/functions.png)

There you should see `hello.js`! If you click that it'll show you the function
log. Calls from your function to `console.log` should appear here with a
timestamp as well as a note for whenever your function is invoked:

![Function log page](./images/function-log.png)

Checkout the "Endpoint" and it shows your function is available on the World
Wide Web at `https://<your-site-name>.netlify.com/.netlify/functions/hello`.

Really quick I just want to explain something. In `netlify.toml` we told Netlify
that it can find the JavaScript for our functions in the `./functions` directory
(we could've called that whatever we wanted). Netlify takes each JavaScript file
in there and deploys it (as-is) to AWS lambda, then sets up an endpoint for
hitting that function for us under `/.netlify/functions` (this is not
configurable AFAIK).

Cool, so now we can hit that URL and boom, Hello Bob Ross!

![Function output saying Hello Bob Ross](./images/function-output.png)

Wahoo!

## Next steps

You can find my version of this project
[on GitHub](https://github.com/kentcdodds/super-simple-netlify-function-example)
and
[on Netlify](https://simple-fn-example.netlify.com/.netlify/functions/hello).

One important thing to note is that Netlify will send only the files for our
functions to AWS and no more. This means that if you try to `require` something
from a local `utils` directory or `node_modules` you're out of luck.

Also, testing this locally is... not easy...

**But wait!** Netlify engineers have built
[`netlify-lambda`](https://www.npmjs.com/package/netlify-lambda) to solve both
of these problems! I've taken the liberty of giving you an example of this same
function that uses `netlify-lambda` here
[on GitHub](https://github.com/kentcdodds/netlify-function-example).

Here are the main changes:

1. Added a `package.json` to configure npm/yarn to install `netlify-lambda` and
created `dev` and `build` scripts.
2. Added a `.gitignore` to ignore the `.netlify` directory which is where
`netlify-lambda` will place our compiled functions.
3. Added a `command` line in our `netlify.toml` file to have the build run the
command `npm run build` which will trigger `netlify-lambda` to build our
function into a single file for delivery to AWS Lambda.
4. Updated the `functions` line in our `netlify.toml` file to point to
`.netlify/functions/` which is where `netlify-lambda` will place our final
built functions.

Everything else is the same, but now we can actually require different files!

## Conclusion

Any time we need to make a change to our function, we simply make those changes,
commit them to GitHub and Netlify will automatically update that on AWS Lambda.
It's very fast and super empowering.

I love enabling technologies like this. I shouldn't have to build an entire
backend with a database and server farm to allow you to send me
[Enterprise Training Inquiries](https://kcd.im/request-a-workshop), and thanks
to Netlify, I don't have to. It's pretty slick!

Give it a try! I think you'll love it too.
4 changes: 2 additions & 2 deletions generate/blogpost.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ tinify.key = process.env.TINY_PNG_API_KEY

const padLeft0 = n => n.toString().padStart(2, '0')
const formatDate = d =>
`${d.getFullYear()}-${padLeft0(d.getMonth() + 1)}-${padLeft0(d.getDay())}`
`${d.getFullYear()}-${padLeft0(d.getMonth() + 1)}-${padLeft0(d.getDate())}`

const listify = a =>
a && a.trim().length
Expand Down Expand Up @@ -94,7 +94,7 @@ async function generateBlogPost() {
title,
date: formatDate(new Date()),
author: 'Kent C. Dodds',
description,
description: `_${description}_`,
categories: listify(categories),
keywords: listify(keywords),
banner: './images/banner.jpg',
Expand Down

0 comments on commit 23450a1

Please sign in to comment.