0
Fork 0
mirror of https://github.com/withastro/astro.git synced 2024-12-30 22:03:56 -05:00
astro/packages/integrations/netlify
2023-07-17 14:56:14 +00:00
..
src [ci] format 2023-07-17 14:56:14 +00:00
test feat(@astrojs/netlify): edge middleware support (#7632) 2023-07-17 15:53:10 +01:00
CHANGELOG.md [ci] release (#7624) 2023-07-13 21:43:11 +08:00
package.json feat(@astrojs/netlify): edge middleware support (#7632) 2023-07-17 15:53:10 +01:00
README.md [ci] format 2023-07-17 14:56:14 +00:00
tsconfig.json feat(@astrojs/netlify): edge middleware support (#7632) 2023-07-17 15:53:10 +01:00

@astrojs/netlify

This adapter allows Astro to deploy your SSR site to Netlify.

Learn how to deploy your Astro site in our Netlify deployment guide.

Why Astro Netlify

If you're using Astro as a static site builder—its behavior out of the box—you don't need an adapter.

If you wish to use server-side rendering (SSR), Astro requires an adapter that matches your deployment runtime.

Netlify is a deployment platform that allows you to host your site by connecting directly to your GitHub repository. This adapter enhances the Astro build process to prepare your project for deployment through Netlify.

Installation

Add the Netlify adapter to enable SSR in your Astro project with the following astro add command. This will install the adapter and make the appropriate changes to your astro.config.mjs file in one step.

# Using NPM
npx astro add netlify
# Using Yarn
yarn astro add netlify
# Using PNPM
pnpm astro add netlify

If you prefer to install the adapter manually instead, complete the following two steps:

  1. Install the Netlify adapter to your projects dependencies using your preferred package manager. If youre using npm or arent sure, run this in the terminal:

      npm install @astrojs/netlify
    
  2. Add two new lines to your astro.config.mjs project configuration file.

    // astro.config.mjs
    import { defineConfig } from 'astro/config';
    import netlify from '@astrojs/netlify/functions';
    
    export default defineConfig({
      output: 'server',
      adapter: netlify(),
    });
    

Edge Functions

Netlify has two serverless platforms, Netlify Functions and Netlify's experimental Edge Functions. With Edge Functions your code is distributed closer to your users, lowering latency.

To deploy with Edge Functions, use netlify/edge-functions in the Astro config file instead of netlify/functions.

// astro.config.mjs
import { defineConfig } from 'astro/config';
import netlify from '@astrojs/netlify/edge-functions';

export default defineConfig({
  output: 'server',
  adapter: netlify(),
});

Per-page functions

The Netlify adapter builds to a single function by default. Astro 2.7 added support for splitting your build into separate entry points per page. If you use this configuration, the Netlify adapter will generate a separate function for each page. This can help reduce the size of each function so they are only bundling code used on that page.

// astro.config.mjs
import { defineConfig } from 'astro/config';
import netlify from '@astrojs/netlify/functions';

export default defineConfig({
  output: 'server',
  adapter: netlify(),
  build: {
    split: true,
  },
});

Static sites

For static sites you usually don't need an adapter. However, if you use redirects configuration (experimental) in your Astro config, the Netlify adapter can be used to translate this to the proper _redirects format.

import { defineConfig } from 'astro/config';
import netlify from '@astrojs/netlify/static';

export default defineConfig({
  adapter: netlify(),

  redirects: {
    '/blog/old-post': '/blog/new-post',
  },
  experimental: {
    redirects: true,
  },
});

Once you run astro build there will be a dist/_redirects file. Netlify will use that to properly route pages in production.

Note

You can still include a public/_redirects file for manual redirects. Any redirects you specify in the redirects config are appended to the end of your own.

Edge Middleware with Astro middleware

The @astrojs/netlify/functions adapter can automatically create an edge function that will act as "Edge Middleware", from an Astro middleware in your code base.

This is an opt-in feature and the build.excludeMiddleware option needs to be set to true:

// astro.config.mjs
import { defineConfig } from 'astro/config';
import netlify from '@astrojs/netlify/functions';
export default defineConfig({
  output: 'server',
  adapter: netlify(),
  build: {
    excludeMiddleware: true,
  },
});

Optionally, you can create a file recognized by the adapter named netlify-edge-middleware.(js|ts) in the srcDir folder to create Astro.locals.

Typings require the https://edge.netlify.com types.

Netlify edge functions run in a Deno environment, so you would need to import types using URLs.

You can find more in the Netlify documentation page

// src/netlify-edge-middleware.ts
import type { Context } from 'https://edge.netlify.com';

export default function ({ request, context }: { request: Request; context: Context }): object {
  // do something with request and context
  return {
    title: "Spider-man's blog",
  };
}

The data returned by this function will be passed to Astro middleware.

The function:

  • must export a default function;
  • must return an object;
  • accepts an object with a request and context as properties;
  • request is typed as Request;
  • context is typed as Context;

Limitations and constraints

When you opt-in to this feature, there are a few constraints to note:

  • The Edge middleware will always be the first function to receive the Request and the last function to receive Response. This is an architectural constraint that follows the boundaries set by Netlify.
  • Only request and context may be used to produce an Astro.locals object. Operations like redirects, etc. should be delegated to Astro middleware.
  • Astro.locals must be serializable. Failing to do so will result in a runtime error. This means that you cannot store complex types like Map, function, Set, etc.

Usage

Read the full deployment guide here.

After performing a build the netlify/ folder will contain Netlify Functions in the netlify/functions/ folder.

Now you can deploy. Install the Netlify CLI and run:

netlify deploy --build

The Netlify Blog post on Astro and the Netlify Documentation provide more information on how to use this integration to deploy to Netlify.

Configuration

To configure this adapter, pass an object to the netlify() function call in astro.config.mjs - there's only one possible configuration option:

dist

We build to the dist directory at the base of your project. To change this, use the dist option:

// astro.config.mjs
import { defineConfig } from 'astro/config';
import netlify from '@astrojs/netlify/functions';

export default defineConfig({
  output: 'server',
  adapter: netlify({
    dist: new URL('./dist/', import.meta.url),
  }),
});

And then point to the dist in your netlify.toml:

[functions]
directory = "dist/functions"

builders

Netlify On-demand Builders are serverless functions used to build and cache page content on Netlifys Edge CDN. You can enable these functions with the builders option:

// astro.config.mjs
import { defineConfig } from 'astro/config';
import netlify from '@astrojs/netlify/functions';

export default defineConfig({
  output: 'server',
  adapter: netlify({
    builders: true,
  }),
});

On-demand Builders are only available with the @astrojs/netlify/functions adapter and are not compatible with Edge Functions.

binaryMediaTypes

This option is only needed for the Functions adapter and is not needed for Edge Functions.

Netlify Functions requires binary data in the body to be base64 encoded. The @astrojs/netlify/functions adapter handles this automatically based on the Content-Type header.

We check for common mime types for audio, image, and video files. To include specific mime types that should be treated as binary data, include the binaryMediaTypes option with a list of binary mime types.

// src/pages/image.jpg.ts

import fs from 'node:fs';

export function get() {
  const buffer = fs.readFileSync('../image.jpg');

  // Return the buffer directly, @astrojs/netlify will base64 encode the body
  return new Response(buffer, {
    status: 200,
    headers: {
      'content-type': 'image/jpeg',
    },
  });
}

Examples

Troubleshooting

For help, check out the #support channel on Discord. Our friendly Support Squad members are here to help!

You can also check our Astro Integration Documentation for more on integrations.

Contributing

This package is maintained by Astro's Core team. You're welcome to submit an issue or PR!

Changelog

See CHANGELOG.md for a history of changes to this integration.