question-mark
Stuck on an issue?

Lightrun Answers was designed to reduce the constant googling that comes with debugging 3rd party libraries. It collects links to all the places you might be looking at while hunting down a tough bug.

And, if you’re still stuck at the end, we’re happy to hop on a call to see how we can help out.

Unable to serve static files with Vite + SSR

See original GitHub issue

Prerequisites

  • I have written a descriptive issue title
  • I have searched existing issues to ensure the bug has not already been reported

Fastify version

4.3.0

Plugin version

6.4.1

Node.js version

14.18.1

Operating system

macOS

Operating system version (i.e. 20.04, 11.3, 10)

BigSur 11.6.6

Description

Seems like fastify-static doesn’t serve files when I try to make PROD build with Vite + SSR. Incorporating sirv does work but when I switch to fastify-static I get

{
  "message": "Route GET:/ not found",
  "error": "Not Found",
  "statusCode": 404
}

Steps to Reproduce

  1. npm init vite-plugin-ssr --skip-git
  2. pick any name
  3. pick react-ts
  4. replace package.json with
  "scripts": {
    "dev": "npm run server",
    "prod": "npm run build && npm run server:prod",
    "build": "vite build",
    "server": "ts-node ./server",
    "server:prod": "cross-env NODE_ENV=production ts-node ./server"
  },
  "dependencies": {
    "@fastify/static": "6.4.1",
    "@types/node": "18.6.1",
    "@types/react": "18.0.15",
    "@types/react-dom": "18.0.6",
    "@vitejs/plugin-react": "2.0.0",
    "cross-env": "7.0.3",
    "fastify": "4.3.0",
    "react": "18.2.0",
    "react-dom": "18.2.0",
    "sirv": "2.0.2",
    "ts-node": "10.9.1",
    "typescript": "4.7.4",
    "vite": "3.0.3",
    "vite-plugin-ssr": "0.4.12"
  },
  "devDependencies": {
    "pino-pretty": "8.1.0"
  }
}
  1. replace ./server/index.ts with
import { createServer as createViteServer } from 'vite';
import { renderPage } from 'vite-plugin-ssr';
import path from 'path';
import fastifyStatic from '@fastify/static';
import sirv from 'sirv';

const isProduction = process.env.NODE_ENV === 'production';
const root = path.resolve();
const assets = sirv(`${root}/dist/client`, {
  immutable: true,
  dev: !isProduction
});

async function startServer () {
  const app = fastify({
    logger: {
      transport:
        !isProduction
          ? {
            target: 'pino-pretty',
            options: {
              translateTime: 'HH:MM:ss Z',
              ignore: 'pid,hostname'
            }
          }
          : undefined
    },
  });

  try {
    if (isProduction) {
      await app.register(fastifyStatic, { root: `${root}/dist/client` });
      // app.addHook('onRequest', (req, reply, done) => {assets(req.raw, reply.raw, done);});
    } else {
      const viteDevMiddleware = await createViteServer({
        root,
        server: {
          middlewareMode: true,
        },
      });

      app.addHook('onRequest', (req, reply, done) => {
        viteDevMiddleware.middlewares(req.raw, reply.raw, done);
      });
    }

    app.get('*', async (req, reply) => {
      const pageContextInit = { url: req.url };
      const pageContext = await renderPage(pageContextInit);
      const { httpResponse } = pageContext;

      if (!httpResponse) return;

      const { body, statusCode, contentType } = httpResponse;

      reply.status(statusCode).type(contentType).send(body);
    });

    const port = (process.env.PORT || 3000) as number;

    await app.listen({ port });

    console.log(`Server running at http://localhost:${port}`);
  } catch (error) {
    app.log.fatal(error);
  }
}

startServer();
  1. npm run prod
  2. visit http://localhost:3000/
  3. See 404
  4. go back to ./server/idex.ts and comment line 33 (await app.register(fastifyStatic, { root: ${root}/dist/client })😉 and uncomment line 34 (app.addHook(‘onRequest’, (req, reply, done) => {assets(req.raw, reply.raw, done);})😉
  5. restart process and visit http://localhost:3000/
  6. You can see application

Expected Behavior

I should be able to see application when using fastify-static

Issue Analytics

  • State:closed
  • Created a year ago
  • Comments:7 (5 by maintainers)

github_iconTop GitHub Comments

2reactions
mcollinacommented, Jul 27, 2022
1reaction
mcollinacommented, Jul 27, 2022

Here is a fix for your prototype:

diff --git a/package.json b/package.json
index fecc423..8567812 100644
--- a/package.json
+++ b/package.json
@@ -4,7 +4,7 @@
     "prod": "npm run build && npm run server:prod",
     "build": "vite build",
     "server": "ts-node ./server",
-    "server:prod": "cross-env NODE_ENV=production ts-node ./server"
+    "server:prod": "npm run build && cross-env NODE_ENV=production ts-node ./server"
   },
   "dependencies": {
     "@fastify/static": "6.4.1",
diff --git a/server/index.ts b/server/index.ts
index f908051..1e81437 100644
--- a/server/index.ts
+++ b/server/index.ts
@@ -31,13 +31,8 @@ async function startServer () {
   try {
     if (isProduction) {
       const staticAssetsPath = `${root}/dist/client`;
-      const assets = sirv(staticAssetsPath, {
-        immutable: true,
-        dev: !isProduction
-      });
 
-      // await app.register(fastifyStatic, { root: staticAssetsPath})
-      app.addHook('onRequest', (req, reply, done) => {assets(req.raw, reply.raw, done);});
+      await app.register(fastifyStatic, { root: staticAssetsPath, wildcard: false })
     } else {
       const viteDevMiddleware = await createViteServer({
         root,
@@ -56,7 +51,8 @@ async function startServer () {
       trpcOptions: { router: appRouter, createContext },
     });
 
-    app.get('*', async (req, reply) => {
+
+    app.get('/*', async (req, reply) => {
       const pageContextInit = { url: req.url };
       const pageContext = await renderPage(pageContextInit);
       const { httpResponse } = pageContext;
@@ -65,7 +61,7 @@ async function startServer () {
 
       const { body, statusCode, contentType } = httpResponse;
 
-      reply.status(statusCode).type(contentType).send(body);
+      return reply.status(statusCode).type(contentType).send(body);
     });
 
     const port = (process.env.PORT || 3000) as number;

However there is a lurking bug in find-my-way.

Read more comments on GitHub >

github_iconTop Results From Across the Web

Can't serve static during dev assets, even inside advertised ...
The simple greeting works fine, but trying localhost:3000/robots.txt fails , as does localhost:3000/public/robots.txt . npx vite build does ...
Read more >
Static Asset Handling - Vite
Importing a static asset will return the resolved public URL when it is served: js import imgUrl from './img.png' document.getElementById('hero-img').src = ...
Read more >
Deploying a Static Site - Vite
These guides provide instructions for performing a static deployment of your Vite site. Vite also supports Server Side Rendering. SSR refers to front-end ......
Read more >
Server-Side Rendering - Vite
A typical SSR application will have the following source file structure: - index.html - server.js # main application server - src/ - main.js...
Read more >
Troubleshooting - Vite
Syntax Error / Type Error happens​​ Vite cannot handle and does not support code that only runs on non-strict mode (sloppy mode). This...
Read more >

github_iconTop Related Medium Post

No results found

github_iconTop Related StackOverflow Question

No results found

github_iconTroubleshoot Live Code

Lightrun enables developers to add logs, metrics and snapshots to live code - no restarts or redeploys required.
Start Free

github_iconTop Related Reddit Thread

No results found

github_iconTop Related Hackernoon Post

No results found

github_iconTop Related Tweet

No results found

github_iconTop Related Dev.to Post

No results found

github_iconTop Related Hashnode Post

No results found