Expo Router v3: API Routes, bundle splitting, speed improvements, and more

Jan 23, 2024 by

Avatar of Evan Bacon

Evan Bacon

Welcome to Expo Router v3, our most powerful release yet! Today we're introducing beta support for the newest Expo platform: Servers. With this, Expo Router is now the first universal, full-stack React framework!

  • API Routes (beta): Build universal server endpoints for your app and website.
  • Bundle splitting (web): Route-based bundle splitting on web for faster page loads.
  • Speed improvements: 2x faster static web builds, 30% smaller base JS bundle, added .mjs support.
  • Testing library: You can now test and reproduce complex navigation flows with Jest.
  • Web <Link /> props: Configure and style <Link /> components with the new target, push, and className props.

Get started with Expo Router v3 today in one line:

Terminal
npx create-expo-app@latest -t tabs@50

If you're new here, Expo Router uses a file-based approach to app development which enables you to build more powerful apps than ever before, with less boilerplate code. The key features so far have been autocomplete and type safety for navigation, SEO and accessibility for web, automatic universal linking, lazy bundling, and more!

Introducing API routes

A cover image for Expo Router v3

Note: API Routes are still in beta during SDK 50.

Based on our API Routes RFC — API Routes are a zero-config system for creating server endpoints with a unified build process. This is the first step toward making Expo Router a full-stack React framework.

Adding a +api.js extension to a route will ensure it's only rendered on the server. API routes are hosted from the same dev server as the website and app in development and must be deployed to a dynamic hosting service in production.

import { ExpoRequest, ExpoResponse } from 'expo-router/server';

export function GET() {
  return ExpoResponse.json({ hello: 'world' });
}

export function POST(request: ExpoRequest) {
  const { prompt } = await request.json();

  // Do something with the prompt
  return ExpoResponse.json({
    /* ... */
  });
}

You can export any of the following functions GET, POST, PUT, PATCH, DELETE, HEAD, and OPTIONS from an API route. The function executes when the corresponding HTTP method is matched. Unsupported methods will automatically return 405: Method not allowed.

Learn more and get started today in API Routes. Additionally, you can download an example app that uses OpenAI to generate text from a GPT-3 model with:

Terminal
npx create-expo-app@latest -e with-openai

The new server architecture will be used to render universal React Server Components in an upcoming release.

Relative Fetch requests

To better support API Routes, we've added the ability to perform relative fetch requests on native by setting the production server URL in the app.json:

{
  "plugins": [
    [
      "expo-router",
      {
        "origin": "https://my-app.dev/"
      }
    ]
  ]
}

This will enable making relative requests with the fetch API, in both development and production environments:

async function fetchHello() {
  // Requests from `http://localhost:8081/hello` in development and `https://my-app.dev/hello` in production.
  const response = await fetch('/hello');
  const data = await response.json();
  // Alerts "Hello world"
  alert('Hello ' + data.hello);
}

In order to use this in production, the server must be hosted publicly. Learn more about hosting production servers.

Supporting 404s with +not-found

API Routes are a special type of route that is only matched after standard routes have been matched. If you were previously using a top-level catch-all like app/[...missing].js to handle 404s and missing routes, then no API Route would ever be matched.

To account for API Routes, we've added an official convention to match all 404 / Not Found routes. By creating a +not-found.js route you can match all remaining requests after API routes have been processed. This is supported on all native platforms, and web in server-mode. When this route is matched, a 404 status code will also be returned on web. Learn more about +not-found routes.

Route-based bundle splitting

Bundle splitting in Expo Router v3

Expo CLI now supports bundle splitting on async imports (e.g. await import("./route")) when bundling for the web platform. We've extended this behavior with Expo Router to automatically split on routes in the app directory.

Expo Router also eagerly loads chunks to prevent network waterfalls on initial requests. Learn more in "Async Routes".

We built this entire feature to be completely universal, but due to the complex nature of native caching we've opted for web-only support in Expo Router v3. Support for splitting bundles on native platforms will be included with React Server Component support in the future.

Configurable app directory

You can now change the /app directory to be any directory in your project. This is useful for testing and white-labeling projects with multiple sub-apps. Learn more about the root directory.

{
  "plugins": [["expo-router", { "root": "./routes" }]]
}

Avoid changing the root directory as this complicates the build process and may cause unexpected development issues. Opt to use the app and src/app directories instead for a consistent and tested experience. Learn more about the src/app directory.

Static font optimization

Static font optimization in Expo Router v3

Fonts loaded with expo-font are now automatically extracted and preloaded on web when using static or server output. This enables fonts to start loading before the JavaScript has finished, leading to better initial styles. This system also enables you to statically render your app even if there's a top-level render guard. Learn more.

import { useFonts } from 'expo-font';

export default function RootLayout() {
  // `loaded` will be `true` in static websites as the font was eagerly loaded with the HTML before this JS was executed.
  const [loaded] = useFonts({
    inter: require('@/fonts/inter.ttf'),
  });

  if (!loaded) {
    // This will no longer be called on static web, meaning the entire boundary will be statically rendered to searchable HTML.
    return null;
  }

  return <Stack />;
}

New push and navigate behaviors

To fix issues with pushing screens in complex routing scenarios, we've changed the router.push() API to always push new routes, whereas the previous version would pop occasionally. You can use the new router.navigate() API to obtain this previous behavior. Learn more about the imperative routing API.

Expo Router testing library

To provide robust test coverage for Expo Router, we created a set of Jest utilities that could quickly emulate entire navigation structures. This testing library is now available for public consumption. Learn more in "Testing Expo Router".

import { renderRouter, screen } from 'expo-router/testing-library';

it('my-test', async () => {
  const MockComponent = jest.fn(() => <View />);

  renderRouter(
    {
      index: MockComponent,
      'folder/a': MockComponent,
      '(group)/b': MockComponent,
    },
    {
      initialUrl: '/folder/a',
    }
  );

  expect(screen).toHavePathname('/folder/a');
});

We use this internally to prevent regressions against the majority of reported issues. We recommend using this system to create minimal reproducible test cases before reporting issues with Expo Router.

The Link component now supports target, rel, and download props on web. Link also now has className support which works as-is on web and can be used with tools like Nativewind to add Tailwind support on all platforms.

<Link target="_blank" className="text-blue-300" href="/home" />

Link components currently navigate to the nearest route matching the href prop, you can now force them to always push a new route by passing the new push prop.

// Navigate to the closest route
<Link href="/" />

// Push "/" as a new route
<Link push href="/" />

Faster builds and smaller bundles

npx expo export -p web is over 2x faster for static websites. An average v2 project exported in ~23s, v3 exports in ~11s. These savings scale proportional to the project size.

The base JS bundle size for production websites is now 30% smaller (from 1.48mb to 1.05mb). The initial bundle size is further decreased by enabling the new bundle splitting functionality on web.

The URL and URLSearchParams standards are built-in. It was previously necessary to polyfill the web standard URL API in order to use many cross-platform libraries available on npm, where developers tend to assume that the URL API is available. We believe that URL is an important enough primitive that it deserves to be built in to the Expo core runtime, and so we now ship our own implementation in the expo package. By doing this, we were able to remove all the various duplicate helper libraries that were used to parse URLs, this further reduced the base bundle size. Learn more about the URL API.

We'll continue to reduce the bundle size by reworking parts of React Navigation and improving tree shaking in Expo CLI.

Stability and support

In Expo Router v3, we've moved the source code and issue tracking to the expo/expo monorepo. During the migration, we fixed and addressed the majority of issues and bugs regarding Expo Router and added lots more documentation and tests.

Configuration requirements like the Babel plugin have been folded into babel-preset-expo and Expo CLI, this also enables many Metro web features like expo-constants features for non-router users. Additional Expo Router functionality has been integrated across the SDK with packages like Splash Screen, Linking, and Font. We've also removed the need for any custom Yarn resolutions and upstreamed a number of bug fixes to Metro and React Native.

Overall, Expo Router is now more powerful, reliable, and seamless than ever before.

Other Highlights

  • Server-hosted dynamic routes on web. The new server output mode supports server navigation to dynamic routes on web. Previously, you could only perform client-side navigation to routes like app/[id].tsx but the server API is capable of redirecting requests to any route in your project automatically. This is not supported with standard static output.
  • Universal Fast Refresh. We've fixed universal Fast Refresh upstream so you no longer need resolutions on react-refresh (be sure to remove them if you have any). The same Fast Refresh implementation now works across all platforms universally and is far more reliable!
  • Experimental base URL support. Expo Router now supports deploying to subdomains with experiments.baseUrl——this applies to all platforms so you may want to configure it with an environment variable in app.config.js. With the addition of this feature, you can now deploy static Expo Router websites to GitHub Pages. We plan to stabilize this API in SDK 51.
  • Improved Tailwind/PostCSS on web. PostCSS with Expo's Metro web will no longer be blocked on caching. This means you can use full Tailwind + PostCSS on web and integrate with fantastic UI packages like Shadcn UI (web-only). Metro CSS is now enabled by default! Here's an example of using Expo Router with Nativewind v4.
  • Support for system links. You can now link to popular external URLs like mailto: and sms: which don't follow the standard :// convention of other URLs.
  • Added mjs and cjs support. All modules are converted to commonjs in the bundler because ESM is not supported on native, but you can now import .mjs modules as expected without modifying the metro.config.js.
  • Custom Metro resolvers and transforms. Users can now extend the Metro resolver and modify the transformer using the Babel caller, enabling better control over the bundling process. Learn more in the new Expo Metro docs.
  • Improved Typed Routes. Typed routes ensure better stability over time by automatically generating TypeScript types for your project. You can now generate types in CI with npx expo customize tsconfig.json. Learn more about Typed Routes in Expo Router.
  • Improved monorepo support. Projects no longer need expo-yarn-workspaces to enable monorepo support in their app. The majority of standard monorepo functionality is built-in to Expo CLI and Expo Metro Config.
  • Better source maps. Source map exports in production web are now supported, we've renamed the npx expo export flag --dump-sourcemap to --source-maps——Hermes source maps now work more reliably.
  • Improved error messages and code removal. Expo CLI now provides full stack traces for component-based errors, tree shakes all unused platform-specific code, and transforms faster when bundling for Hermes.
  • @expo/webpack-config is deprecated in favor of Expo CLI's Metro web. This means that Webpack support will continue to work in SDK 50, but it will not be actively developed, and it will be removed in a future release. Read the "Webpack support in Expo CLI is now deprecated" blog for the full story, and learn about migrating away from Webpack to Metro.
  • CSS is enabled by default with Metro web. CSS is not supported on Android and iOS, but on web you can use all CSS features by importing CSS files. Learn more.
  • tsconfigPaths is now enabled in @expo/metro-config by default: this means that all you need to do to add path aliases is configure the paths property in your tsconfig.json. For example, "@/*": ["src/*"] will allow you to write code like import Button from '@/components/Button'; anywhere in your codebase and have it resolve to the correct location within src. They're also now supported in jest-expo. Learn more.

Notable breaking changes

  • expo-router/babel has been removed. Delete this plugin from your babel.config.js file, and be sure to clear the Metro cache before restarting your dev server——this means running npx expo start --clear.
  • router.push default behavior changed. router.push is now router.navigate and the new router.push will always push routes. This is technically a bug fix, but it may cause unexpected changes in complex navigation behavior.
  • react-native-gesture-handler is no longer added automatically. You can now choose to optionally add gesture handler if you wish to use the <Drawer /> navigator. We recommend avoiding this dependency on web platforms as it will increase bundle size substantially and mostly be unused on web. Learn more about the Drawer navigator.
  • src directory changed to build. We now ship transpiled JavaScript to production in the expo-router/build/* directory. This is a breaking change if you were imported internals from Expo Router in your project.

Known issues

➡️ Upgrading your app

Here's how to upgrade your app to Expo Router v3 from v2:

  • Upgrade your app to SDK 50: follow the instructions in the SDK 50 release notes.

  • Update the babel.config.js:

    • Remove the expo-router/babel plugin in favor of babel-preset-expo preset.
    module.exports = function (api) {
      api.cache(true);
      return {
        presets: ['babel-preset-expo'],
    -    plugins: ['expo-router/babel']
      };
    };
    
    • Be sure to clear the Metro cache before restarting your dev server:
    Terminal
    npx expo start --clear
  • Possibly the largest behavior change in any version of Expo Router——router.push is now router.navigate and the new router.push will always push routes. This is technically a bug fix, but it may cause unexpected changes in complex navigation behavior.

  • react-native-gesture-handler is no longer added automatically and must be injected if you wish to use the <Drawer /> navigator. We recommend avoiding this dependency on web platforms as it will increase bundle size substantially and mostly be unused on web. Learn more about the Drawer navigator.

  • Enable Async Routes to use the new bundle splitting functionality on web. Async Routes may have issues on Android with Reanimated, you can disable the feature per-platform if needed.

  • If you have a top-level catch-all route like [...missing].js, ensure you rename this to +not-found.js if you plan to use API Routes. Otherwise, you'll only see the "not found" route when you ping an API endpoint.

  • If you have custom splash screen handling, change the import of SplashScreen in expo-router to expo-splash-screen.

  • If you were using the hrefAttrs prop on the Link component for adding additional web props, migrate to top-level props by the same name, e.g. hrefAttrs={{ target: '_blank' }} should now be target="_blank"——this applies to target, rel, download——all of which are web-only and automatically shimmed on native.

  • If you're using the react-native-web style escape hatch (style={{ $$css: true, _: "myclass" }}) to set className on Link components for web, migrate to the top-level className prop, e.g. <Link className="myclass" />

  • Questions? We'll be hosting an SDK 50 launch live-stream on January 31st, join us on YouTube.

Compatibility

Ensure you use libraries that are versioned to work with Expo SDK 50:

| Library | Version | | -------------------------- | ------- | | expo | ^50.0.0 | | expo-router | ^3.0.0 | | react | 18.2.0 | | react-native | ~0.73.2 | | react-native-web | ~0.19.6 | | @react-navigation/native | ^6.0.2 |

You can validate versions automatically with Expo CLI:

Terminal
npx expo install --check

Thanks to everyone who contributed to the release!

Along with everyone who's adopted this exciting new technology, we'd like to thank the following people for their contributions to Expo Router v3: @sync @kudo @muneebahmedayub @gabrieldonadel @tsapeta @quinlanj @douglowder @marklawlor @kitten @byCedric @EvanBacon

Credits

Avatar of Evan Bacon

Evan Bacon as project lead

Avatar of Mark Lawlor

Mark Lawlor for router development, testing, support, and documentation

Avatar of Aman Mittal

Aman Mittal for documentation

Avatar of Cedric van Putten

Cedric van Putten for dev tools development, debugging, and support

Avatar of Phil Pluckthun

Phil Pluckthun for bundler development, and documentation