GitHub Pages+Vercel双部署?小心publicPath/base让你前功尽弃!

  1. 问题现象
  • 本地做了两个项目,将项目推送到github中,想要同时部署在github pages和vercel上。结果第一个项目在github pages上可以正常访问,在vercel上访问失败。另一个项目在vercel上可以正常访问,在githubpages中却访问失败。报错内容都是访问时CSS加载失败。

  • 一通研究,发现是publicPath/base搞的鬼。

    • 我的第一个项目是使用vite+vue3+element ui搭建的To-Do List项目,刚开始在vite.config.js中设置了base,为了方便在Github Pages上部署。
    javascript 复制代码
    // vite.config.js
    import { defineConfig } from 'vite'
    import vue from '@vitejs/plugin-vue'
    
    export default defineConfig({
        plugins: [vue()],
        base: '/vue-todo-list/'
    })
    • 我的第二个项目是使用vue2和@vue/cli5搭建的天气展示项目。在vue.config.js根据NODE_ENV动态设置了publicPath的值。
    javascript 复制代码
    // vue.config.js
    const { defineConfig } = require('@vue/cli-service')
    module.exports = defineConfig({
      transpileDependencies: true,
      publicPath: process.env.NODE_ENV === 'production' 
        ? '/weather-app/' 
        : '/'
    })
  1. 原理分析
  • 在 GitHub Pages 或Vercel上项目空白的问题,很可能是因为构建的静态资源路径与 GitHub Pages 或Vercel的部署方式不匹配导致的。
    • 不同项目对于如何处理静态资源路径问题的配置方式不同
      • Vue CLI 项目通常在 vue.config.js 文件中配置 publicPath 。
      • Vite 项目通常在 vite.config.js 文件中配置 base 。
    • 不同平台的部署方式不同
      • GitHub Pages通常将项目部署到项目仓库子目录下,此时需要将publicPathbase设置为 /<REPO_NAME>/
      • Vercel通常将项目部署到根目录下,此时需要将publicPathbase设置为 /
      • 在开发环境下,通常是 /
    • 需要设置正确的 publicPathbase,以便 GitHub Pages和Vercel 都能够正确地找到CSS、JavaScript、图片等静态资源。
  1. 解决方案
  • 动态配置publicPathbase(环境变量+条件判断)。
    • Vite项目

      1. 修改vite.config.js,根据DEPLOY_TARGET是否为'github-pages'来决定base的值
      javascript 复制代码
       // vite.config.js
       import { defineConfig } from 'vite'
       import vue from '@vitejs/plugin-vue'
      
       const isGitHubPages = process.env.DEPLOY_TARGET === 'github-pages'
       const base = isGitHubPages ? '/vue-todo-list/' : '/'
      
       export default defineConfig({
         plugins: [vue()],
         base
       })
      1. 修改package.json,在部署到Github-pages上需要执行的"deploy"命令中添加DEPLOY_TARGET=github-pages。相当于在部署到Github-pages前先修改DEPLOY_TARGET的值。
      json 复制代码
       // package.json
       {
        // ...,
        "scripts": {
          "dev": "vite",
          "build": "vite build",
          "preview": "vite preview",
          "deploy": "DEPLOY_TARGET=github-pages npm run build && gh-pages -d dist"
        },
        // ...
      }
    • Vue CLI项目
      *

      1. 修改vue.config.js,根据NODE_ENV以及DEPLOY_TARGET的值来决定publicPath的值
      javascript 复制代码
         // vue.config.js
         const { defineConfig } = require('@vue/cli-service')
         module.exports = defineConfig({
           transpileDependencies: true,
           publicPath: (() => {
             if (process.env.NODE_ENV === 'production') {
               if (process.env.DEPLOY_TARGET === 'github-pages') {
                 return '/weather-app/'
               } else if (process.env.DEPLOY_TARGET === 'vercel') {
                 return '/'
               }
             }
             return '/' // 默认开发环境
           })()
         })
      1. 修改package.json,在部署到Github-pages上需要执行的"deploy"命令中添加DEPLOY_TARGET=github-pages。相当于在执行build命令前先修改DEPLOY_TARGET的值。
      json 复制代码
       // package.json
       {
        // ...,
        "scripts": {
          "serve": "vue-cli-service serve",
          "build": "vue-cli-service build",
          "lint": "vue-cli-service lint",
          "deploy": "DEPLOY_TARGET = github-pages bash deploy.sh"
        },
        // ...
       }
      1. 在Vercel里的Build and Deployment Setting中修改Build Command,在npm run build之前添加DEPLOY_TARGET=vercel。相当于在执行build命令前先修改DEPLOY_TARGET的值。Framework Preset选择了Vue.js时,点击Build Command右侧的Override即可修改。
相关推荐
渔舟唱晚@4 小时前
大模型数据流处理实战:Vue+NDJSON的Markdown安全渲染架构
vue.js·大模型·数据流
鱼樱前端7 小时前
Vue3+d3-cloud+d3-scale+d3-scale-chromatic实现词云组件
前端·javascript·vue.js
q_19132846957 小时前
基于Springboot+Vue的办公管理系统
java·vue.js·spring boot·后端·intellij idea
满分观测网友z7 小时前
vue的<router-link>的to里面的query和params的区别
前端·javascript·vue.js
BillKu7 小时前
Vue3 + TypeSrcipt 防抖、防止重复点击实例
前端·javascript·vue.js
鱼樱前端7 小时前
Vue3结合three和babylonjs实现3D数字展厅效果
前端·vue.js
Themberfue7 小时前
Vue ⑥-路由
前端·javascript·vue.js
陪我一起学编程8 小时前
关于nvm与node.js
vue.js·后端·npm·node.js
thels_9 小时前
记录一个用了很久的git提交到github和gitee比较方便的方法
git·gitee·github
周全全9 小时前
基于 Vue 和 Spring Boot 实现滑块验证码的机器验证
前端·vue.js·spring boot