AWS

Greenwood can be automatically deployed to AWS for static hosting using S3 and Cloudfront with GitHub Actions.

You can see a complete hybrid project example in our demonstration repo.

Static Hosting

In this section, we'll share the steps for up S3 and Cloudfront together for static web hosting.

  1. Configure S3 by following these steps
  2. Once you have followed those steps, run greenwood build in your project and upload the contents of the public/ directory to the bucket
  3. Finally, setup Cloudfront to use this bucket as an origin by following these steps:

Keep an eye out for prompts from AWS to enable IAM rules for your function and make sure to invalidate the Cloudfront distribution between tests, since error pages / responses will get cached.

You should now be able to access your site at http://{your-dist}.cloudfront.net/! 🏆

Now at this point, if you have any routes like /search/, you'll notice they are not working unless index.html is appended to the path. To enable routing (URL rewriting) for cleaner URLs, follow the Configure Trigger section of this guide to attach the Lambda as a Lambda@Edge function that will run on every incoming request.

Below is a sample Edge function for doing the rewrites:

export const handler = async (event, context, callback) => {
  const { request } = event.Records[0].cf;

  // re-write "clean" URLs to have index.html appended
  // to support routing for Cloudfront <> S3
  if (request.uri.endsWith("/")) {
    request.uri = `${request.uri}index.html`;
  }

  callback(null, request);
};

At this point, you'll probably want to use Route 53 to put your domain in front of your Cloudfront distribution.

Serverless

Coming soon!

There is no adapter plugin yet for serverless hosting, though it is on our roadmap.

GitHub Actions

If you're using GitHub, you can use GitHub Actions to automate the pushing of build files on commits to a GitHub repo. This action will also invalidate your Cloudfront cache on each publish.

  1. In your AWS account, create and / or add an AWS Secret Access Key (AWS_SECRET_ACCESS_KEY) and Access Key ID (AWS_SECRET_ACCESS_KEY_ID) and add them to your repository as GitHub secrets.

  2. We also recommend adding your bucket name as secret too, e.g. AWS_BUCKET_NAME

  3. At the root of your repo add a GitHub Action called .github/workflows/publish.yml and adapt as needed for your own branch, build commands, and package manager.

    name: Upload Website to S3
    
    on:
      push:
        branches:
          - main
    
    jobs:
      build:
        runs-on: ubuntu-latest
    
        # match to your version of NodeJS
        steps:
          - uses: actions/checkout@v4
          - uses: actions/setup-node@v4
            with:
              node-version: 22
    
          - name: Install Dependencies
            run: |
              npm ci
    
          # use your greenwood build script
          - name: Run Build
            run: |
              npm run build
    
          - name: Upload to S3 and invalidate CDN
            uses: opspresso/action-s3-sync@master
            env:
              AWS_ACCESS_KEY_ID: ${{ secrets.AWS_SECRET_ACCESS_KEY_ID }}
              AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
              # make sure this matches your bucket's region
              AWS_REGION: "us-east-1"
              FROM_PATH: "./public"
              # your target s3 bucket name goes here
              DEST_PATH: s3://${{ secrets.AWS_BUCKET_NAME }}
    

Now when you push changes to your repo, the action will run an the build files will automatically be uploaded.