纯前端实现更新检测

通过判断打包后的html文件中的js入口是否发生变化,进而实现前端的代码更新

为了使打包后的文件带有hash值,需要对vite打包进行配置

javascript 复制代码
import { defineConfig } from 'vite';
import vue from '@vitejs/plugin-vue';
import { resolve } from 'path';
import AutoImport from 'unplugin-auto-import/vite'
import Components from 'unplugin-vue-components/vite'
import { ElementPlusResolver } from 'unplugin-vue-components/resolvers'

export default defineConfig({
  base: './',
  plugins: [
    vue(), AutoImport({
      resolvers: [ElementPlusResolver()],
    }),
    Components({
      resolvers: [ElementPlusResolver()],
    }),],
  envDir: './', // .env所在目录
  build: {
    target: 'es2015',
    outDir: 'dist',
    rollupOptions: {
      output: {
        manualChunks(id) { 
          if (id.includes('node_modules')) {
            return id.toString().split('node_modules/')[1].split('/')[0].toString();
          }
        },
        // 文件附带上hash
        entryFileNames: '[name]-[hash].js',
        chunkFileNames: '[name]-[hash].js',
        assetFileNames: '[name]-[hash].[ext]'
      }
    },
    minify: 'esbuild', 
  },
  resolve: {
    alias: {
      "@": resolve(__dirname, 'src'),
    },
    extensions: ['.vue', '.js']
  },
  server: {
    proxy: {
      '/api': {
        target: 'http://localhost:8083',
        changeOrigin: true,
        rewrite: (path) => path.replace(/^\/api/, "")
      }
    }
  },
});

update.js的主要功能是定期检查网页中的脚本标签是否有更新,并在检测到新版本时提示用户刷新页面以确保平台正常使用。

javascript 复制代码
// update.js
import { ElMessageBox } from 'element-plus';

let timer = undefined;
const url = import.meta.env.VITE_UPDATE_URL;

function cmpSets(set1, set2) {
    if (set1.size !== set2.size) return false;
    for (let item of set1) {
        if (!set2.has(item)) return false;
    }
    return true;
}
// 更新提示框
function updateNotice() {
    ElMessageBox({
        title: '更新提示!',
        message: "检测到新版本,请立即刷新以确保平台正常使用",
        confirmButtonText: '确定',
        type: 'warning',
    }).finally(() => {
        window.location.reload();
    });
}

// 获取页面中的脚本标签src属性的哈希值集合,忽略查询参数
async function getSrcHash() {
    try {
        const html = await fetch(url).then((res) => res.text());
        const scriptSrcRegex = /<script\b[^>]*src="([^"]*)"/gi;
        const scriptSrcs = [...html.matchAll(scriptSrcRegex)].map(match => match[1]);
        // 开发环境中入口文件包含保存时的时间戳,判断时移除此参数以免开发时一直提示更新
        const cleanSrcs = scriptSrcs.map(src => src.split('?')[0]); 
        const encodedSrcs = new Set(cleanSrcs.map(src => encodeURIComponent(src)));
        return encodedSrcs;
    } catch (error) {
        console.error('Failed to fetch script hashes:', error);
        return new Set();
    }
}

// 比较当前脚本标签哈希值与新获取的哈希值
async function cmpHash() {
    try {
        const newHash = await getSrcHash();
        const storedHash = JSON.parse(localStorage.getItem('curHash')) || [];

        // 如果是新用户或首次访问,直接保存哈希值并退出函数
        if (storedHash.length === 0) {
            localStorage.setItem('curHash', JSON.stringify([...newHash]));
            return;
        }
        // 合并新旧哈希值集合
        let curHash = new Set(storedHash);
        if (!cmpSets(curHash, newHash)) {
            console.info("new:", newHash);
            console.info("old:", curHash);
            console.log("更新提示")
            clearInterval(timer);
            updateNotice();
        }

        // 保存最新的哈希值集合到localStorage
        localStorage.setItem('curHash', JSON.stringify([...newHash]));
    } catch (error) {
        console.error('Error comparing script hashes:', error);
    }
}


// 设置定时器,定期检查脚本更新
timer = setInterval(cmpHash, 30 * 1000);

// 页面加载时比较哈希值
async function init() {
    await cmpHash();
}
// 页面加载时执行初始化函数
init();

最后在main.js入口文件引入即可

javascript 复制代码
import { createApp } from 'vue'
import App from './App.vue'
import ElementPlus from 'element-plus'
import 'element-plus/dist/index.css'
import router from './router/index.ts'
import "./update.js" //引入自动更新脚本
import { createPinia } from 'pinia'

const pinia = createPinia()
const app = createApp(App)
app.use(router)
app.use(pinia)
app.use(ElementPlus)
app.mount('#app')

通过篡改localstorage中的curHash,可引导更新,效果如下

30s更新一次算频繁吗?

相关推荐
pianmian12 分钟前
arcgis几何与游标(1)
开发语言·python
-曾牛3 分钟前
【LangChain4j快速入门】5分钟用Java玩转GPT-4o-mini,Spring Boot整合实战!| 附源码
java·开发语言·人工智能·spring boot·ai·chatgpt
涵信12 分钟前
第九节:React HooksReact 18+新特性-React 19的use钩子如何简化异步操作?
前端·javascript·react.js
Aaaaaaaaaaayou20 分钟前
浅玩一下 Mobile Use
前端·llm
nanzhuhe21 分钟前
python中参数前**的含义
开发语言·python
这个昵称也不能用吗?21 分钟前
react-native搭建开发环境过程记录
前端·react native·cocoapods
hy_花花22 分钟前
Vue3.4之defineModel的用法
前端·vue.js
wt_cs25 分钟前
身份认证C#集成方案-数字时代身份证实名认证利器
开发语言·c#
DataFunTalk36 分钟前
Foundation Agent:深度赋能AI4DATA
前端·后端·算法
hboot38 分钟前
rust 全栈应用框架dioxus
前端·rust·全栈