【Web IDE】WebContainer容器在浏览器中启动运行nodejs并使用vite启动项目

参考了文章WebContainer/api 基础(Web IDE 技术探索 一)

在浏览器中运行vite的vue3项目

示例站点

最终效果

主要流程

加载WebContainer=》加载代码压缩包=>解压代码压缩包=》生成文件树=》挂载文件树=》pnpm安装依赖=》启动项目

代码

复制代码
<script setup>
import { onMounted, ref } from 'vue'
import { WebContainer } from "@webcontainer/api";
import { mountZip } from '@/views/Containers/utls.js'
const webUrl = ref("");
const codeZip = '/code/vue-project.zip'
async function initContainer() {
  console.log("挂载")
  // Call only once
  const webcontainerInstance = await WebContainer.boot();
  const nodeV = await webcontainerInstance.spawn("node", ["-v"]);
  nodeV.output.pipeTo(
    new WritableStream({
      write(data) {
        console.log("node -v ==>", data);
      },
    })
  );
  const fileTree =  await mountZip(codeZip)
  console.log('挂载文件',fileTree)
  // 1. 挂载文件
  await webcontainerInstance.mount(fileTree);
  console.log("ls");
  const ls = await webcontainerInstance.spawn("ls", ["-al"]);
  ls.output.pipeTo(
    new WritableStream({
      write(data) {
        console.log(data);
      },
    })
  );
  // 2. 下载依赖
  console.log("pnpm install");
  const install = await webcontainerInstance.spawn("pnpm", ["install"]);
  install.output.pipeTo(
    new WritableStream({
      write(data) {
        console.log(data);
      },
    })
  );
  // 3. 判断exit 状态
  let code = await install.exit;
  if (code !== 0) {
    console.error("error to install.");
  }

  // 4. 启动服务
  console.log("npm run dev");
  const process = await webcontainerInstance.spawn("npm", ["run","dev"]);
  process.output.pipeTo(
    new WritableStream({
      write(data) {
        console.log(data);
      },
    })
  );
  // 5. 监听服务启动
  webcontainerInstance.on("server-ready", (port, url) => {
    console.log("server-ready", url);
    webUrl.value = url;
  });
}


onMounted(() => {
  mountZip(codeZip)
  initContainer()
})
</script>

<template>
<div>
  <iframe :src="webUrl" style="height: 100vh;width: 100%"/>
</div>
</template>

<style scoped>

</style>

工具函数

复制代码
import JSZip from 'jszip'
export async function  mountZip(zipUrl){
  console.log("读取zip文件",zipUrl)
  const fileTree = {}
  try {
    // 使用 fetch 获取 ZIP 数据
    const response = await fetch(zipUrl);
    const buffer = await response.arrayBuffer();

    // 使用 JSZip 处理获取到的数据
    const zip = new JSZip();
    const zipContents = await zip.loadAsync(buffer);

    // 处理解压后的内容
    for (const [relativePath, file] of Object.entries(zipContents.files)) {
      // console.log('relativePath',relativePath)
      if(file.dir){
        let dirList = relativePath.split('/')
        // console.log('dirList',dirList)
        if(dirList.length > 2){
          let tmp = fileTree
          for (let i = 0; i < dirList.length - 1; i++) {
            // console.log('tmp(dirList[i]',tmp[dirList[i]])
            if(tmp[dirList[i]]){
              tmp = tmp[dirList[i]].directory
            }else{
              tmp[dirList[i]] = {
                directory: {},
              }
            }
          }
        }else{
          fileTree[dirList[0]] = {
            directory: {},
          }
        }
      }else{
        let dirList = relativePath.split('/')
        // console.log('dirList',dirList)
        if(dirList.length > 1) {
          let tmp = fileTree
          for (let i = 0; i < dirList.length - 1; i++) {
            // console.log('tmp(dirList[i]', tmp[dirList[i]])
            if (tmp[dirList[i]]) {
              tmp = tmp[dirList[i]].directory
            } else {
              tmp[dirList[i]] = {
                directory: {},
              }
            }
          }
          // console.log('tmp',tmp)
          tmp[dirList[dirList.length - 1]] = {
            file: {
              contents: await file.async('string')
            },
          }
        }else{
          // console.log('根目录文件',dirList)
          fileTree[dirList[dirList.length - 1]] = {
            file: {
              contents: await file.async('string')
            },
          }
        }
      }
    }
  } catch (error) {
    console.error('获取 ZIP 数据时出错:', error);
  }
  console.log('fileTree',fileTree)
  return fileTree
}
相关推荐
Attacking-Coder1 分钟前
前端面试宝典---webpack面试题
前端·面试·webpack
极小狐26 分钟前
极狐GitLab 容器镜像仓库功能介绍
java·前端·数据库·npm·gitlab
程序猿阿伟38 分钟前
《Flutter社交应用暗黑奥秘:模式适配与色彩的艺术》
前端·flutter
rafael(一只小鱼)42 分钟前
黑马点评实战笔记
前端·firefox
weifont42 分钟前
React中的useSyncExternalStore使用
前端·javascript·react.js
初遇你时动了情1 小时前
js fetch流式请求 AI动态生成文本,实现逐字生成渲染效果
前端·javascript·react.js
影子信息1 小时前
css 点击后改变样式
前端·css
几何心凉1 小时前
如何使用 React Hooks 替代类组件的生命周期方法?
前端·javascript·react.js
小堃学编程1 小时前
前端学习(1)—— 使用HTML编写一个简单的个人简历展示页面
前端·javascript·html
CONTONUE2 小时前
如何调整yarn.nodemanager.vmem-pmem-ratio参数?
ide·eclipse