DEV Community

Cover image for AWS Amplify - Serverless Functions for External Data with Modern UI Frontend (React, Angular, Vue)
Sam Hamburger
Sam Hamburger

Posted on • Edited on • Originally published at sam-hamburger.netlify.com

AWS Amplify - Serverless Functions for External Data with Modern UI Frontend (React, Angular, Vue)

Cover photo by Kai Oberhäuser on Unsplash

As a frontend developer, using a serverless backend can help minimize backend development work, allowing you to focus almost entirely on client-side development. Although most cloud service providers have serverless capabilities, this article will be focused on AWS, using their Amplify framework to structure the backend.

Amplify Framework

From AWS Amplify:

AWS Amplify is a development platform for building secure, scalable mobile and web applications. It makes it easy for you to authenticate users, securely store data and user metadata, authorize selective access to data, integrate machine learning, analyze application metrics, and execute server-side code. Amplify covers the complete mobile application development workflow from version control, code testing, to production deployment, and it easily scales with your business from thousands of users to tens of millions. The Amplify libraries and CLI, part of the Amplify Framework, are open source and offer a pluggable interface that enables you to customize and create your own plugins.

Amplify is constantly evolving and improving, and currently contains features for:

  • Real-time data retrieval and storage via pre-structured (and modifiable) GraphQL and REST API queries, including auto authentication features and connections to Amazon DymanoDB databases out-of-the-box
  • Authentication APIs using Amazon Cognito user pools, including optional federated logins and sign-ups via social providers (Google, Facebook, Amazon, etc.)
  • Analytics for applications with built-in support for Amazon Pinpoint and Amazon Kinesis
  • AI and ML services for predictions, primarily for NLP and text recognition from images
  • Interactions for AI-powered chatbots
  • Storage buckets (public, protected, or private) via Amazon S3 for managing user and global content
  • PubSub for cloud-based message middleware, push notifications, etc.
  • AR and VR services

Setup, Configure and Invoke Lambda Function in a React + Amplify Project

For this tutorial, we will use a React web app, but you can choose other frameworks including React Native, Angular, Ionic, Vue, or Vanilla JavaScript at the Amplify Getting Started webpage.

1. Initialize a Client-Side Application

$ npx create-react-app amplify-sls-app
$ cd amplify-sls-app
$ npm i @aws-amplify/api

We install the @aws-amplify/api package for use later in this tutorial.

Note: ensure you use npx, and do not have create-react-app globally installed. If you do, you must run: npm uninstall -g create-react-app first, or else the npx command will not provide a starter app

2. Install Amplify CLI and Configure

This needs to be done once to connect and configure with your AWS account.

$ npm i -g @aws-amplify/cli
$ amplify configure

This will prompt you to sign into AWS console in your browser. Do so, and take a note of the region you are in (upper right hand corner). Follow the steps in the CLI, and click through all default options on the AWS console in your browser.

3. Initialize a New Amplify Project

This needs to be done for each new Amplify project, in the root of the project's directory (root of React app).

$ amplify init

Follow the steps, choosing default options where applicable. See my responses below:

  • Enter a name for the project ❯ amplify-sls-app
  • Enter a name for the environment ❯ dev
  • Choose your default editor: *❯ Visual Studio Code**
  • Choose the type of app that you're building ❯ javascript
  • What javascript framework are you using ❯ react
  • Source Directory Path: ❯ src
  • Distribution Directory Path: ❯ build
  • Build Command: ❯ npm run-script build
  • Start Command: ❯ npm run-script start
  • Do you want to use an AWS profile? ❯ Yes
  • Please choose the profile you want to use ❯ sls-profile

In the last step, choose the profile you configured with amplify configure, or use a previously created profile.

4. Create an API and a Lambda Function

To use an express server in your Lambda function, initialize an API for your Amplify app. You can create additional APIs in the future, but for now we will create a REST API named mainApi.

We will be hitting the Unsplash API to get a stock photo for a given query.

$ amplify add api

Follow below answers to the follow-up questions asked in the command-line:

  • Please select from one of the below mentioned services ❯ REST
  • Provide a friendly name for your resource to be used as a label for this category in the project ❯ mainApi
  • Provide a path (e.g., /items) ❯ /photo
  • Choose a Lambda source ❯ Create a new Lambda function
  • Provide a friendly name for your resource to be used as a label for this category in the project ❯ getPhoto
  • Provide the AWS Lambda function name ❯ getPhoto
  • Choose the function template that you want to use ❯ Serverless express function
  • Do you want to access other resources created in this project from your Lambda function? ❯ No
  • Do you want to edit the local lambda function now? ❯ No
  • Restrict API access ❯ No
  • Do you want to add another path? ❯ No

In the future, if you create a new function with amplify add function, ensure to run amplify update api to connect the new function with your previously created REST API.

5. Push Amplify Changes to AWS

To push your updated Amplify app including the new API and function, run:

$ amplify push

6. Verify Local Invocation of Lambda Function

Ensure all dependencies are installed locally for your function:

$ cd amplify/backend/function/getPhoto/src
$ npm install

To verify that your lambda can can be invoked, in any directory in the command-line, run:

$ amplify invoke function getPhoto

Choose all of the default responses.

This starts the function's express server, and you should be able to test a GET call to http://localhost:3000/photo (via Postman or Insomnia) and get a successful response.

7. Create an Unsplash Developer Account for Free API Access

Follow this to get an API App Key for Unsplash API: Join Unsplash.

8. Customize Lambda Function

Locally, your function code will be under amplify/backend/function/getPhoto.

In the src folder, you will see an index.js file, which acts as the function handler, and proxies requests through the express endpoints auto-generated in app.js in the same directory.

Although the CLI creates auto-generated endpoints in app.js, feel free to delete, modify or edit as needed.

Configure Unsplash

In the app.js file, under all of the require statements add with your Unsplash App Key:

const UNSPLASH_APP_ID = '<your-unsplash-app-id>';

Axios

In the amplify/backend/function/getPhoto/src directory, run:

$ npm i axios

In app.js, at the top, add:

const axios = require('axios');

Make Request

We will setup the photo to be pulled from Unsplash on the GET /photo endpoint. We will make it an async function, await the axios GET request and then send back the sample response, with a photo field added with the value of the response from the Unsplash request.

app.get('/photo', async function(req, res) {
  // Add your code here
  const photo = await axios
    .get('https://api.unsplash.com/search/photos', {
      params: {
        query: 'skiing',
        page: 1,
        per_page: 1,
        orientation: 'landscape',
      },
      headers: {
        Authorization: `Client-ID ${UNSPLASH_APP_ID}`,
      },
    })
    .catch(err => {
      console.log('Error happened during fetching!', err);
    });
  console.log(photo.data);
  res.json({ success: 'get call succeed!', urL: req.url, photo: photo.data });
});

I am doing a sample query for a 'skiing' photo. Additionally, I pass through a param to return a photo that has 'landscape' orientation. You can visit the Unsplash API Documentation for other requests.

Push to AWS

After any changes to your lambda function, you will want to run amplify push in order to keep your cloud application in-sync with your local modifications.

$ amplify push

9. Invoke Function via React App

To verify that your lambda can can be invoked, in any directory in the command-line, run:

$ amplify invoke function getPhoto

Test a GET call to http://localhost:3000/photo (via Postman or Insomnia).

The response should show you success message as well as the photo data from Unsplash in the photo field of the JSON response:

{
  "success": "get call succeed!",
  "urL": "/photo",
  "photo": {
    "total": 1319,
    "total_pages": 1319,
    "results": [
      {
        "id": "pmfJcN7RGiw",
        "created_at": "2019-03-04T06:24:23-05:00",
        "updated_at": "2020-02-14T00:03:35-05:00",
        "promoted_at": null,
        "width": 6000,
        "height": 4000,
        "color": "#200B13",
        "description": "Full speed ahead!",
        "alt_description": "man ice skiing on hill",
        "urls": {
          "raw": "https://images.unsplash.com/photo-1551698618-1dfe5d97d256?ixlib=rb-1.2.1&ixid=eyJhcHBfaWQiOjExNTM4Mn0",
          "full": "https://images.unsplash.com/photo-1551698618-1dfe5d97d256?ixlib=rb-1.2.1&q=85&fm=jpg&crop=entropy&cs=srgb&ixid=eyJhcHBfaWQiOjExNTM4Mn0",
          "regular": "https://images.unsplash.com/photo-1551698618-1dfe5d97d256?ixlib=rb-1.2.1&q=80&fm=jpg&crop=entropy&cs=tinysrgb&w=1080&fit=max&ixid=eyJhcHBfaWQiOjExNTM4Mn0",
          "small": "https://images.unsplash.com/photo-1551698618-1dfe5d97d256?ixlib=rb-1.2.1&q=80&fm=jpg&crop=entropy&cs=tinysrgb&w=400&fit=max&ixid=eyJhcHBfaWQiOjExNTM4Mn0",
          "thumb": "https://images.unsplash.com/photo-1551698618-1dfe5d97d256?ixlib=rb-1.2.1&q=80&fm=jpg&crop=entropy&cs=tinysrgb&w=200&fit=max&ixid=eyJhcHBfaWQiOjExNTM4Mn0"
        },
        "links": {
          "self": "https://api.unsplash.com/photos/pmfJcN7RGiw",
          "html": "https://unsplash.com/photos/pmfJcN7RGiw",
          "download": "https://unsplash.com/photos/pmfJcN7RGiw/download",
          "download_location": "https://api.unsplash.com/photos/pmfJcN7RGiw/download"
        },
...

10. Display Image in React App

In App.js (or feel free to create a new component), import useState and useEffect, as well as API from @aws-amplify/api, and ensure to configure it.

import React, {useState, useEffect} from 'react';

import API from '@aws-amplify/api';

import awsconfig from './aws-exports';

// Configure Amplify once in your app (importing API in subcomponents afterwords will be        auto-configured)
API.configure(awsconfig);

Change your App function to:

function App() {
  const [photoData, setPhotoData] = useState(null);

  useEffect(() => {
    const getPhoto = async () => {
      const photoResponse = await API.get('mainApi', '/photo');
      console.log(photoResponse);
      setPhotoData(photoResponse);
    };

    getPhoto();

    // adding the empty [] parameter will ensure this function only runs on the first mount of the component
  }, []);

  return (
    <div>
      {photoData && photoData.photo ? (
        <img src={photoData.photo.results[0].urls.small} />
      ) : (
        <p>Photo loading...</p>
      )}
    </div>
  );
}
  • The aws-exports.js file is auto-generated from the Amplify CLI, never edit it manually.
  • We use the useState hook to add state to our functional component
  • We use the useEffect hook with a second parameter of [] in order to run the async function inside only once

Run your React app in the root project directory and see your photo:

$ npm run start

Conclusion

The steps above can be expanded to access any external or internal APIs, and further expansion can accomplish some really cool functionalities for your app. Some use cases can be found here. Feel free to share this with your peers and comment with any suggestions or discussion points!

Top comments (0)