WebContainerapi 基础(Web IDE 技术探索 一)

news2024/12/23 23:35:44

前言

随着web技术的发展,在网页端直接运行node.js、实现微型操作系统已经不再是难事。今天介绍的 WebContainers就是一个基于浏览器的运行时用于执行 Node.js 应用程序和操作系统命令,它完全运行在您的浏览器页面中,提供了文件系统运行进程的能力,同时**内置了 nodejs、npm/yarn/pnpm 等包管理器。**也就是说,基于WebContainers,可以直接在网页端运行 node -v、npm install、npm run dev等命令,甚至能对文件系统进行操作,例如fs.writeFile、 fs.readFile(但是这个仅是在虚拟环境中,并不会在系统上真正生成文件)。

如果想在Web 端实现代码编辑、项目运行、执行文件等操作,还是有必要学习下的。

技术应用

stackblitz

CodeSandBox

Web Containers

官网 :Dev environments In your web app.

网上学习资源较少,如若有误,望海涵哈!WebContainer API非常适合交互式编码体验,它最常见的用例包括生产级IDE、编程教程、下一代文档、人工智能应用程序或员工入职平台。WebContainers已经由StackBlitz经典编辑器、Codeflow、官方SveltKit教程和Cloudflare Wrangler工作人员等数百万用户进行了测试,因此稳定性及可靠性无容置疑。

安装

npm i @webcontainer/api

使用

<template>
  <div>App</div>
</template>

<script setup>
import { onMounted } from "vue";
import { WebContainer } from "@webcontainer/api";

async function initContainer() {
  // Call only once
  const webcontainerInstance = await WebContainer.boot();
}

onMounted(initContainer);
</script>

正常是要报错的,WebContainers需要SharedArray Buffer,而这反过来又要求它们运行的网站是跨源隔离的。

配置跨源隔离

// 配置 WebContainer/api 跨源隔离
headers: {
    "Cross-Origin-Embedder-Policy": "require-corp",
    "Cross-Origin-Opener-Policy": "same-origin",
},

并且!官网还给了我们警告:请注意,boot方法只能调用一次,并且只能创建一个WebContainer实例。

spawn

spawn 是执行命令的关键函数,必须要学会哈,返回值是WebContainerProcess,例如:

// 执行 npm install
const install = await webcontainerInstance.spawn('npm', ['i']);

如上例,我们想要执行并输出 node -v、npm -v 怎么操作呢?

  const nodeV = await webcontainerInstance.spawn("node", ["-v"]);
  console.log(nodeV);

这显然不是我们想要的结果,因为返回值是一个WebContainerProcess,如下:

WebContainerProcess

**exit: Promise<number>**:进程的退出代码的状态,其实可以理解为 await Promise 的状态,它的返回值是Promise,因此需要等待哦!

**input:[WritableStream](https://developer.mozilla.org/en-US/docs/Web/API/WritableStream "WritableStream")<string>**:可以理解为传入终端的附加参数,具体的还没研究透。

**output:[ReadableStream](https://developer.mozilla.org/en-US/docs/Web/API/ReadableStream "ReadableStream")<string>:?**接收所有终端输出的流,包括派生进程及其子进程发出的stdout和stderr。这个就可以接收终端的输出了,具体用法如下:

  const nodeV = await webcontainerInstance.spawn("node", ["-v"]);
  nodeV.output.pipeTo(
    new WritableStream({
      write(data) {
        console.log("node -v ==>", data);
      },
    })
  );

Methods:**kill()**杀死/结束一个process。

Methods:**resize()**调整附着终端的大小。

如上,便是WebContainerProcess 的所有属性及方法,我们常用的是exit、**output、kill**。

teardown

销毁WebContainer实例,使其不可用,并释放其资源。之后,可以通过调用boot来获得一个新的WebContainer实例。

FileSystemTree

FileSystemTree和FileSystemAPI是整个WebContainer的核心,因此必须先介绍这两个东西,有了这基础,后面去操作API才不会那么吃力,很多博主就开始照着官网卡卡操作,什么含义也没讲清楚。

FileSystemTree,如名,是一种树状结构,用于描述要装入的文件夹的内容,在webcontainer中,如何来创建或描述文件间的关系?

file

const tree = {}

如上,这就是个空目录,根路径为 ‘/’,给它添加 一个文件,先手动添加哈,后面介绍API会给大家讲解:

const tree = {
     'index.js': {
        file: { // 标识是文件 [file | directory]
          contents: 'const x = 1;',  // 文件内容
        },
      },
}

mount

将上诉文件使用 mount 挂载到container上:

  await webcontainerInstance.mount(tree);

readdir

使用API读取目录,查看目录结构:

  // Call only once
  const webcontainerInstance = await WebContainer.boot();

  await webcontainerInstance.mount(fileTree);
  // readdir 读取目录结构
  const files = await webcontainerInstance.fs.readdir("/");
  console.log(files);

可以看到,index.js 已经挂载上去了。

directory

进行文件夹创建及挂载,使用directory 标识:

export const fileTree = {
  "index.js": {
    file: {
      contents: `const x=1;`,
    },
  },
  // 创建 src 目录
  src: {
    directory: {
      // directory 标识是目录
      // 里面有文件夹的话,继续嵌套 directory 标识
      // 创建 src/main.js
      "main.js": {
        file: {
          contents: `console.log('main.js')`,
        },
      },
    },
  },
};

当我们在读取 / 的时候,发现并没有将main.js 文件一并输出,我们可以使用递归查询,后面到API在介绍。

FileSystemAPI

API无非就是node fs API的思想:

mkdir

创建文件夹,如果目录已经存在,则会抛出异常!

readdir

读取给定的目录并返回其文件和目录的数组,这个是读取不了下级目录结构的哈,需要使用递归实现。可通过传递配置项,以获取更多信息,例如判断文件是文件夹还是文件:

// 执行 readdir 的时候,可以进行参数传递,以获取不同的返回值

interface Options {
  encoding?: BufferEncoding;
  withFileTypes?: boolean;
}

// 当传递 withFileTypes = true 的时候,会返回Dirent objects 的数组

/**
 * @description 读取目录结构
 * @param { string } } root
 */
async function readDir(root) {
  let result = {}; // 存储所有的目录结构

  // 读取
  const files = await webcontainerInstance.fs.readdir(root, {
    withFileTypes: true,
  });
  // 解析
  files.forEach(async (item) => {
    // 如果是文件夹,则继续调用自身
    if (item.isDirectory()) {
      result[item.name] = await readDir(`${root}/${item.name}`);
    }
    // 是文件,则存储到 result 中
    else result[item.name] = item.name;
  });

  return result;
}

这样才能读取到完整的目录结构,看项目中的实际应用哈!

readFile

读取给定路径的文件。如果该文件不存在,它将引发一个错误。

const bytes = await webcontainerInstance.fs.readFile('/package.json');

const content = await webcontainerInstance.fs.readFile('/index.js', 'utf-8');

rename

文件重命名,路径必须存在哈,不能这个文件命名为另外文件夹下的文件!

await webcontainerInstance.fs.rename('/src/index.js', '/src/main.js');

非法!!不同目录下不能rename。

await webcontainerInstance.fs.rename('/src/index.js', '/demo/main.js');

rm

删除文件或目录。如果路径是一个文件,它将删除该文件。如果路径是目录,则需要第二个参数,并将选项递归设置为true,以删除目录及其内部的所有内容,包括嵌套文件夹。可传递参数

interface Options {
  force?: boolean;
  recursive?: boolean;
}

force:当为true时,如果路径不存在,则会忽略异常。
recursive:如果为true,它将递归地删除目录,包括嵌套目录。

1. 删除文件

await webcontainerInstance.fs.rm("/src/main.js", { recursive: true });

2. 删除文件夹(必须删除子项 ==>recursive: true)

  await webcontainerInstance.fs.rm("/src/utils", { recursive: true });

writeFile

又来一个关键方法!写入文件。将文件写入给定的路径。如果该文件不存在,它将创建一个新文件,如果该文件存在,它将覆盖该文件。支持写入string | Uint8Array,还能指定字符编码格式。

await webcontainerInstance.fs.writeFile('/src/main.js', 'console.log("Hello from WebContainers!")');

// writeFile 写入文件
  await webcontainerInstance.fs.writeFile(
    "/package.json",
    JSON.stringify({
      name: "my-app",
      version: "0.0.1",
      dependencies: {
        vite: "^5.0.0",
      },
      scripts: {
        dev: "vite",
      },
    })
  );

读取package.json 的内容:

// 读取文件夹结构
  const json = await webcontainerInstance.fs.readFile("/package.json", {
    encoding: "utf-8",
  });
  console.log("json", json);

watch

监听文件/文件夹的修改,自身返回监听对象,身上有 close 方法,用于停止监听:

let watchFile = webcontainerInstance.fs.watch('/src/main.js', (event) => {
    console.log(`action: ${event}`);
});

// ... your code

watchFile.close() // 停止监听

webcontainerInstance.fs.watch('/src', { recursive: true }, (event, filename) => {
    console.log(`file: ${filename} action: ${event}`);
});

搭建应用

上诉讲述了WebContainers API、FileTree、 FileAPI,望大家好好理解,下面正式开始实践:

node -v

let version = await webcontainerInstance.spawn("node", ["-v"]);
  version.output.pipeTo(
    new WritableStream({
      write(data) {
        console.log("node -v :", data);
      },
    })
  );

在不需要文件的情况下,你甚至可以不挂载文件。

ls -l

  // Call only once
  webcontainerInstance = await WebContainer.boot();
  await webcontainerInstance.mount(fileTree);
  let ls = await webcontainerInstance.spawn("ls", ["-l"]);
  ls.output.pipeTo(
    new WritableStream({
      write(data) {
        console.log("ls -l:", data);
      },
    })
  );

node index.js

// 注意路径的写法,不带根路径哈
await webcontainerInstance.spawn("node", ["src/main.js"]);

npm run dev

run dev 是前端项目常用的命令,需要依赖的文件有 package.json、index.html:

export const fileTree = {
  "index.html": {
    file: {
      contents: `<!DOCTYPE html>
      <html lang="en">
        <head>
          <meta charset="UTF-8" />
          <link rel="icon" type="image/svg+xml" href="/vite.svg" />
          <meta name="viewport" content="width=device-width, initial-scale=1.0" />
          <title>Vite + Vue</title>
        </head>
        <body>
          <div id="app">这是 WebContainer 测试页面</div>
        </body>
      </html>
      `,
    },
  },
  "package.json": {
    file: {
      contents: `{
        "name": "my-app",
        "version": "0.0.1",
        "dependencies": {
          "vite": "^5.0.0"
        },
        "script": {
          "dev": "vite"
        }
      }
      `,
    },
  },
};

json文件必须是JSON格式哈,不然会报错:

完整代码如下:

  // Call only once
  webcontainerInstance = await WebContainer.boot();
  // 1. 挂载文件
  await webcontainerInstance.mount(fileTree);
  // 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) return 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);
  });

当我们打开url时,报错,这个是限制预览哈,大家感兴趣可以关注issue

那我们启动后,如何预览页面呢?使用Iframe!

  // 5. 监听服务启动
  webcontainerInstance.on("server-ready", (port, url) => {
    console.log("server-ready", url);
    const iframe = document.querySelector("iframe");
    iframe.src = url;
  });

如上,使用iframe即可正常预览,至于生产环境能不能直接 open new tab,还得验证。

总结

从Webcontainer基础、FileSystemTree、FileSystemAPI,再到实践,一步步进行技术验证,望大家对webcontainers有一个基础的认识与了解,相信大家也能看出,webcontainer是操作的核心,其他的什么内容编辑、terminal都是其他技术型,所以在本篇中,没有涉及其他多余的技术,就是希望大家明白,webcontainer 才是核心。

至于如何结合Terminal、monaco实现Web IDE,我们下一节继续讲解!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2264457.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

解决pytorch安装中的三个错误

查明已安装python版本为3.12.7后&#xff0c;创建虚拟环境。 报错内容&#xff1a;ArgumentError: one of the arguments -n/–name -p/–prefix is required 解决方式&#xff1a; 输入 conda create -n pytorch python3.8即可安装成功。 参考文章&#xff1a;https://blo…

LeetCode:104.二叉树的最大深度

跟着carl学算法&#xff0c;本系列博客仅做个人记录&#xff0c;建议大家都去看carl本人的博客&#xff0c;写的真的很好的&#xff01; 代码随想录 LeetCode&#xff1a;104.二叉树的最大深度 给定一个二叉树 root &#xff0c;返回其最大深度。 二叉树的 最大深度 是指从根节…

上传文件(vue3)

使用el-upload 先上传到文件服务器&#xff0c;生成url 然后点击确定按钮&#xff1a; 保存数据 <template><el-dialog top"48px" width"500" title"新增协议" :modelValue"visible" close"handleClose()">…

PostgreSql+Pgpool-II配置高可用集群(超详细)

一.配置示例 1.1. 基本配置示例 1.1.1. 开始 首先&#xff0c;我们必须学习如何在使用复制之前安装和配置 Pgpool-II 和数据库节点。 1.1.1.1. 安装 Pgpool-II 安装 Pgpool-II 非常简单。 在已解压源 tar ball 的目录中&#xff0c; 执行以下命令。 $ ./configure$ make$ …

2012年西部数学奥林匹克试题(几何)

2012/G1 △ A B C \triangle ABC △ABC 内有一点 P P P, P P P 在 A B AB AB, A C AC AC 上的投影分别为 E E E, F F F, 射线 B P BP BP, C P CP CP 分别交 △ A B C \triangle ABC △ABC 的外接圆于点 M M M, N N N. r r r 为 △ A B C \triangle ABC △ABC 的内…

【自动驾驶】单目摄像头实现自动驾驶3D目标检测

&#x1f351;个人主页&#xff1a;Jupiter. &#x1f680; 所属专栏&#xff1a;传知代码 欢迎大家点赞收藏评论&#x1f60a; 目录 概述算法介绍演示效果图像推理视频推理 核心代码算法处理过程使用方式环境搭建下载权重文件pytorch 推理&#xff08;自动选择CPU或GPU&#x…

帝国CMS自动生成标题图片并写进数据库

帝国CMS背景可自定义&#xff0c;可单独背景也可以随机背景,此插件根帝国cms官方论坛帖子改的&#xff0c;增加了生成图片后写入数据库,笔者的古诗词网 www.gushichi.com 也是这样设置的。 效果图 将下面的代码插入到/e/class/userfun.php中增加如下函数 单独背景代码 //自动…

5G -- 5G网络架构

5G组网场景 从4G到5G的网络演进&#xff1a; 1、UE -> 4G基站 -> 4G核心网 * 部署初中期&#xff0c;利用存量网络&#xff0c;引入5G基站&#xff0c;4G与5G基站并存 2、UE -> (4G基站、5G基站) -> 4G核心网 * 部署中后期&#xff0c;引入5G核心网&am…

达梦官方工具 SQLark数据迁移(oracle->达梦数据库)

应国产化需求需要,需将系统中涉及的各中间件替换成国产中间件,此文介绍了从Oracle迁移数据至达梦dm8的步骤,该文在windos环境下已验证测试过 1 SQLark介绍 SQLark是一款专为信创应用开发者设计的数据库开发和管理工具。它支持快速查询、创建和管理多种类型的数据库系统&#xf…

【Mac】安装 PaddleOCR

环境&#xff1a;Mac M1 芯片 1、安装 1.1 安装 Anaconda Anaconda 安装较为简单&#xff0c;直接在 Anaconda 官网 下载pkg文件&#xff0c;根据向导提示完成安装。 Anaconda 用于搭建 Python 虚拟环境&#xff0c;目的是为了避免与之前环境安装库的版本冲突&#xff0c;另…

AI的进阶之路:从机器学习到深度学习的演变(二)

AI的进阶之路&#xff1a;从机器学习到深度学习的演变&#xff08;一&#xff09; 三、机器学习&#xff08;ML&#xff09;&#xff1a;AI的核心驱动力 3.1 机器学习的核心原理 机器学习&#xff08;Machine Learning, ML&#xff09;突破了传统编程的局限&#xff0c;它不再…

渗透测试实战—某医院安全评估测试

免责声明&#xff1a;文章来源于真实渗透测试&#xff0c;已获得授权&#xff0c;且关键信息已经打码处理&#xff0c;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本…

垂起固定翼无人机大面积森林草原巡检技术详解

垂起固定翼无人机大面积森林草原巡检技术是一种高效、精准的监测手段&#xff0c;以下是对该技术的详细解析&#xff1a; 一、垂起固定翼无人机技术特点 垂起固定翼无人机结合了多旋翼和固定翼无人机的优点&#xff0c;具备垂直起降、飞行距离长、速度快、高度高等特点。这种无…

2024年《网络安全事件应急指南》

在这个信息技术日新月异的时代&#xff0c;网络攻击手段的复杂性与日俱增&#xff0c;安全威胁层出不穷&#xff0c;给企事业单位的安全防护能力带 来了前所未有的挑战。深信服安全应急响应中心&#xff08;以下简称“应急响应中心”&#xff09;编写了《网络安全事件应急指南》…

PHP阶段一

PHP 一门编程语言 运行在服务器端 专门用户开发网站的 脚本后缀名.php 与HTML语言进行混编&#xff0c;脚本后缀依然是.php 解释型语言&#xff0c;不要编译直接运行 PHP运行需要环境&#xff1a; Windows phpstudy Linux 单独安装 Web 原理简述 1、打开浏览器 2、输入u…

【C#】实现Json转Lua (Json2Lua)

关键词: C#、JsonToLua、Json2Lua、对象序列化Lua 前提需引入NewtonsofJson&#xff0c;引入方法可先在Visual Studio 2019 将Newtonsoft.Json.dll文件导入Unity的Plugins下。 Json格式字符串转Lua格式字符串&#xff0c;效果如下&#xff1a; json字符串 {"1": &q…

GM_T 0039《密码模块安全检测要求》题目

单项选择题 根据GM/T 0039《密码模块安全检测要求》,送检单位的密码模块应包括()密码主管角色。 A.一个 B.两个 C.至少一个 D.至少两个 正确答案:C 多项选择题 根据GM/T 0039《密码模块安全检测要求》,关于非入侵式安全,以下属于安全三级密码模块要求的是()。 …

在 Ubuntu 上安装 Muduo 网络库的详细指南

在 Ubuntu 上安装 Muduo 网络库的详细指南 首先一份好的安装教程是非常重要的 C muduo网络库知识分享01 - Linux平台下muduo网络库源码编译安装-CSDN博客 像这篇文章就和shit一样&#xff0c;安装到2%一定会卡住&#xff0c;如果你不幸用了这个那真是遭老罪了 环境&#xf…

MySQL存储引擎-数据文件

InnoDB数据文件存储结构 分为一个ibd数据文件–>Segment&#xff08;段&#xff09;–>Extent&#xff08;区&#xff09;–>Page&#xff08;页&#xff09;–>Row&#xff08;行&#xff09; Tablesapce 表空间&#xff0c;用于存储多个ibd数据文件&#xff…

医药垃圾分类管理系统|Java|SSM|JSP|

【技术栈】 1⃣️&#xff1a;架构: B/S、MVC 2⃣️&#xff1a;系统环境&#xff1a;Windowsh/Mac 3⃣️&#xff1a;开发环境&#xff1a;IDEA、JDK1.8、Maven、Mysql5.7 4⃣️&#xff1a;技术栈&#xff1a;Java、Mysql、SSM、Mybatis-Plus、JSP、jquery,html 5⃣️数据库可…