Lorem

Using Gatsby Serverless Functions as an abstracted API

Theyā€™re here! Gatsby Serverless Functions are here! Theyā€™re finally here! šŸ’ ā€” And I couldnā€™t be more excited!

A little while ago I wrote a post about using the Twitter API as a kind of CMS for my blog. You can read more about that here: Use Netlify Functions and the Twitter API v2 as a CMS for your Gatsby blog

In short I created my own mini API that is used by both my commercial portfolio: https://www.pauliescanlon.io and my blog: https://paulie.dev to display my current Twitter bio information on the home page(s).

Both my sites are built on top of my Gatsby theme gatsby-theme-terminal which I havenā€™t updated to Gatsby v3.x.x yet, and both sites are currently hosted by Netlifyā€¦ but I didnā€™t want that to hold me up from adopting Gatsby Serverless Functions.

Hereā€™s how I did it. šŸ‘‡

API

My API is actually a very simple Gatsby Site deployed on Gatsby Cloud. It displays the response from my endpoints in HTML <pre> tags. This isnā€™t entirely necessary but it allows me to test a request and preview the responses in an almost real and un-abstracted way.

You can also see the json response for each of my endpoints using the links below:

Functions

To use Gatsby Serverless Functions have a read of the Getting Started docs but iā€™ll break down the steps youā€™ll probably need to use Gatsby Serverless Functions anyway

Install Dependencies

Youā€™ll need Gatsby 3.4.0 or higher installed

npm install gatsby@latest

Enable Feature Flag

Gatsby Serverless Functions are still technically in beta so youā€™ll need to add FUNCTIONS as a feature flag in your gatsby-config.js

// gatsby-config.js

module.exports = {
  flags: {
    FUNCTIONS: true
  },
  plugins: [...]
}

šŸšØ News just in from Joel Sumner Smith | Product Manager @GatsbyšŸ‘‡

File System - API

Like with the File System Route API The path to Functions is the same on disk as it would be in browser: E.g src/api/some-function => http(s)://.../api/some-function

Endpoint

I have x2 two endpoints in my API they both work in the same way but hereā€™s the twitter-user endpoint

// api/twitter-user.js

const { twitter } = require('../clients'); // exports Twitter client

export default async function handler(req, res) {
  res.setHeader('Access-Control-Allow-Origin', '*'); // YOLO
  try {
    const { data } = await twitter.get('users/by/username/PaulieScanlon', {
      user: {
        fields:
          'created_at,description,entities,id,location,name,pinned_tweet_id,profile_image_url,protected,public_metrics,url,username,verified,withheld',
      },
    });
    res.status(200).json({
      user: data,
    });
  } catch {
    res.status(500).json({
      error: 'Ooops server error',
    });
  }
}

Request

I use fetch to hit my API and set the response in state which is then returned by Jsx.

I fetch the data in the ProfileInfo.js component in both my blog and my site

// Any React component

...

  const [twitter, setTwitter] = useState({ user: null })

  useEffect(() => {
    fetch('https://paulieapi.gatsbyjs.io/api/twitter-user')
      .then((response) => {
        if (response.status >= 200 && response.status <= 299) {
          return response.json()
        } else {
          throw Error(response.message)
        }
      })
      .then((response) => {
        // Save response in state hook
        setTwitter(response)
      })
      .catch((error) => {
        // Handle the error
      })
  }, [])

  return (
    <div>
      {twitter.user ? (
        <Fragment>
          <h2>Twitter User</h2>
          <pre>
            <code>{JSON.stringify(twitter.user, null, 2)}</code>
          </pre>
        </Fragment>
      ) : null}
    </div>
  )

...

ā€¦ and thatā€™s pretty much it. I might at some point do something with the GitHub data but I just wanted to share how iā€™m using Gatsby Serverless Functions in an abstracted and potentially easily incrementally adoptable way. ā€” jeez what a sentence! šŸ•ŗ