AWS React SSR architectures

Oleksii Kosynskyi

by

Hello

Oleksii Kosynskyi

Senior Software Engineer

at ChainSafe (web3.js)

 

 

 

 

 

 

 

avkos

 

avkosinski

 

avkosinski

SSR

What is it?

Why SSR?

- SEO optimization

- Fast page load

Why cloud?

- SSL

- Cache settings

- Distributed execution (regions)

- Scaling

Lambda implementation

import {renderToString} from 'react-dom/server'


export const handler = async (event) => {
       
        const props = await getSomeDataFromDB(event)
        
        const appString = renderToString(<App {...props}/>)
        
        return {
            statusCode: 200,
            headers: {"Content-Type": "text/html"},
            body: insertAppToIndexHTML(appString)
        };
}

ApiGateway

SSR

LambdaEdge

SSR

LambdaEdge

CloudFront

LambdaEdge

Lambda

Max execution time

5s

15 min

Max response size

1Mb

-

Limitation

LambdaEdge

Lambda

1 mln requests

Free Tier

1 mln requests and 400 000 Gb-seconds executions per month

0,60 USD

-

Gb-seconds

0,00005001 USD

0,20 USD

0,0000166667 USD

Price

Lambda EDGE

Lambda API GATEWAY

Compress

CloudFront

Compress

Lambda Edge

import zlib from 'zlib'
...
const gzip = (html) => new Promise((resolve, reject) => {
    const input = Buffer.from(html);
    zlib.deflate(input, (err, res) => {
        if (err) {
            return reject(err)
        }
        resolve(res.toString('base64'))
    });
})

...

await gzip(html)
...

Lambda Edge Response

{
   status: "200",
   statusDescription: "OK",
   bodyEncoding: 'base64',
   headers: {
      "cache-control": [
          {
               key: "Cache-Control",
               value: "max-age=100",
          },
      ],
      "content-type": [
          {
               key: "Content-Type",
               value: "text/html",
          },
      ],
      "content-encoding": [
          {
               key: 'Content-Encoding',
               value: 'deflate'
          }
      ]
  },
  body,
}

Set up resources

Need to upload files directly from S3 without using Lambda

{
    test: /\.(svg|jpeg|jpg|gif|png)$/,
    use: [{
      loader: 'file-loader',
      options: {
                
                 publicPath: url => url, 
                 
                 emitFile: false, 
                 
                 name(resourcePath) {
                      const filename = path.basename(resourcePath); 
                    
                      return clientManifest.files[`static/media/${filename}`]; 
                 },
              },
    }],
},
...
plugins: [
   new webpack.DefinePlugin({
      PUBLIC_URL: JSON.stringify(process.env.PUBLIC_URL),
   })
]

IMAGE_INLINE_SIZE_LIMIT=0

Set up deploy

AWS CDK

SSR API GATEWAY STACK

export class AppSsrApiStack extends Stack {
    constructor(scope: Construct, id: string, props?: StackProps) {
    
        super(scope, id, props);
        
        const staticConstruct = new StaticConstruct(this, 'StaticStack', {
            buildPath: '../app/build-prod/',
            bucketName: 'static-ssr-api'
        })
        
        new SsrApiConstruct(this, 'SsrApiStack', {
            region: this.region,
            staticS3Bucket: staticConstruct.staticS3Bucket,
            originAccessIdentity: staticConstruct.originAccessIdentity
        })
        
    }
}
export class StaticConstruct extends Construct {
    public staticS3Bucket: s3.Bucket
    public originAccessIdentity: cloudfront.OriginAccessIdentity

    constructor(scope: Construct, id: string, props: TProps) {
    
        super(scope, id);

        this.staticS3Bucket = new s3.Bucket(...);

        this.originAccessIdentity = new cloudfront.OriginAccessIdentity(...);
        
        this.staticS3Bucket.grantRead(this.originAccessIdentity);

        new s3deploy.BucketDeployment(...);
        
    }
}
export class SsrApiConstruct extends Construct {
    constructor(scope: Construct, id: string, props: TProps) {
        
        super(scope, id);
        
        const ssrLambdaApi = new lambda.Function(...)
        
        const ssrApi = new apigw.LambdaRestApi(...)
        
        const ssrApiDistribution = new cloudfront.CloudFrontWebDistribution(...);
        
    }
}
            {
                originConfigs: [{
                        s3OriginSource: {...S3Props},
                        behaviors: [
                            {
                                pathPattern: '/static/*.*',
                            },
                            {
                                pathPattern: '/static/js/*.*',
                            },
                            {
                                pathPattern: '/static/css/*.*',
                            },
                            {
                                pathPattern: '/favicon.ico',
                            },
                            {
                                pathPattern: '/manifest.json',
                            }
                        ]
                    }, {
                        customOriginSource: { ...apiGWProps },
                        behaviors: [
                            {
                                isDefaultBehavior: true,
                            }
                        ]
                    }
                ]
            }
  

SSR LAMBDA-EDGE STACK

export class AppSsrEdgeStack extends Stack {
    constructor(scope: Construct, id: string, props?: StackProps) {
    
        super(scope, id, props);
        
        const staticConstruct = new StaticConstruct(this, 'StaticEdgeStack', {
            buildPath: '../app/build-prod/',
            bucketName: 'static-ssr-edge'
        })
        
        new SsrEdgeConstruct(this, 'SsrEdgeStack', {
            staticS3Bucket: staticConstruct.staticS3Bucket,
            originAccessIdentity: staticConstruct.originAccessIdentity
        })
        
    }
}
export class StaticConstruct extends Construct {
    public staticS3Bucket: s3.Bucket
    public originAccessIdentity: cloudfront.OriginAccessIdentity

    constructor(scope: Construct, id: string, props: TProps) {
    
        super(scope, id);

        this.staticS3Bucket = new s3.Bucket(...);

        this.originAccessIdentity = new cloudfront.OriginAccessIdentity(...);
        
        this.staticS3Bucket.grantRead(this.originAccessIdentity);

        new s3deploy.BucketDeployment(...);
        
    }
}
export class SsrEdgeConstruct extends Construct {
    constructor(scope: Construct, id: string, props: TProps) {
    
        super(scope, id);

        const ssrLambdaEdge = new lambda.Function(...);

        const ssrEdgeFunctionVersion = new lambda.Version(...);

        const ssrEdgeDistribution = new cloudfront.CloudFrontWebDistribution(...);
        
    }
}
           originConfigs: [{
                   s3OriginSource: { ...S3Props },
                   behaviors: [
                       {
                           pathPattern: '/static/*.*',
                       },
                       {
                           pathPattern: '/static/js/*.*',
                       },
                       {
                           pathPattern: '/static/css/*.*',
                       },
                       {
                           pathPattern: '/favicon.ico',
                       },
                       {
                           pathPattern: '/manifest.json',
                       }
                   ]
               }, {
                   s3OriginSource: { ...S3Props },
                   behaviors: [{
                           isDefaultBehavior: true,
                           lambdaFunctionAssociations: [{
                                   eventType: cloudfront.LambdaEdgeEventType.ORIGIN_REQUEST,
                                   lambdaFunction: ssrEdgeFunctionVersion
                           }]
                       }]
               }]
  

Deploy to cloud

Demo - Time

SSR-APIGW

SSR-EDGE

SSR CHECK

Test different regions

SSR LAMBDA-EDGE

SSR LAMBDA-EDGE

(Second test)

SSR LAMBDA API GW

Set up local start

AWS CDK

yarn cdk synth AppSsrApiStack 

sam local start-api -t ./cdk.out/AppSsrApiStack.template.json

Set up local start

Serverless Framework

service: lambda-ssr
frameworkVersion: '2 || 3'
app: lambda-ssr

custom:
  serverless-offline:
    httpPort: 3005
    includeModules: true # add excluded modules to the bundle
    noPrependStageInUrl: true #remove stage /dev from url
  webpack:
    webpackConfig: 'webpack.config.js' # Name of webpack configuration file
    includeModules:
    packager: 'yarn' # Packager that will be used to package your external modules
    excludeFiles: /**/*.test.js # Provide a glob for files to ignore

provider:
  name: aws
  lambdaHashingVersion: '20201221'
  runtime: nodejs14.x

plugins:
  - serverless-webpack
  - serverless-offline
functions:
  ssr:
    name: lambda-ssr
    handler: ssr.apiHandler
    events:
      - http:
          method: GET
          path: /
          cors: true
      - http:
          method: ANY
          path: /{proxy+}
          cors: true

Static files

import {apiHandler as apiHandlerSsr} from './ssr'
import express from 'express'
import serverlessExpress from '@vendia/serverless-express'

export const apiHandler = apiHandlerSsr

const app = express();
app.use(express.static(__dirname + '/../../build-local/static'));
app.use(express.static(__dirname + '/../../build-local'));

export const staticHandler = serverlessExpress({app})
  static:
    name: lambda-static
    handler: ssr.staticHandler
    events:
      - http:
          method: GET
          path: /static/{proxy+}
          cors: true
      - http:
          method: GET
          path: /favicon.ico
          cors: true
      - http:
          method: GET
          path: /manifest.json
          cors: true
      - http:
          method: GET
          path: /logo192.png
          cors: true
watch 'yarn build' ./src
sls offline

2

1

Local start

Local start

GitHub

Helpful links

AWS CDK

https://cdkworkshop.com/20-typescript.html

 

REACT APP ENVs

https://create-react-app.dev/docs/advanced-configuration/

 

SERVERLESS - NEXTJS

https://github.com/serverless-nextjs/serverless-next.js

?

React Serverless SSR

By Alexey Kosinski

React Serverless SSR

  • 430