您当前的位置:首页 > 电脑百科 > 程序开发 > 编程百科

如何实现并部署自己的Npm解析服务

时间:2023-10-08 17:06:05  来源:微信公众号  作者:魔术师卡颂

如何实现并部署自己的Npm解析服务

大家好,我卡颂。

你是否好奇 —— codesandbox是如何在线运行代码的?

要回答这个问题,我们先看看前端项目是如何在本地跑起来的。简单来说分为3步:

  • 执行npm install安装依赖。
  • 使用打包工具(比如webpack)打包、编译代码(如果使用Vite会省去打包的步骤,但会执行「预构建」)。
  • 将步骤2的产物通过script标签注入页面。

codesandbox能在线运行代码,显然他也实现了上述步骤,具体来说,codesandbox内置了2个在线服务:

  • npm解析服务 —— 用于实现上述步骤1。
  • 在线打包服务 —— 用于实现上述步骤2、3。

本文我们来聊聊如何实现并部署自己的npm解析服务。

codesandbox简要工作原理

下面是一个常见的codesandbox界面,包含两部分:

  • 左边的文件系统、代码编辑器。
  • 右边的效果预览区域。

如何实现并部署自己的Npm解析服务

其中「效果预览区域」是一个iframe,对于上图中的例子,iframe的地址是https://pjdp86.csb.App/。如果你打开这个地址,会发现他就是代码的预览效果:

如何实现并部署自己的Npm解析服务

但这并不意味着codesandbox帮我们部署了项目。实际上,这个地址中前端代码是在页面打开后再编译、打包的。

打开codesandbox项目时经常看到的下述界面,就是前端编译代码的画面:

如何实现并部署自己的Npm解析服务

具体来说,当我们打开一个codesandbox项目,iframe对应地址初始化时,会执行如下操作:

  • 下载项目代码(即编辑器中显示的代码)。
  • 根据项目package.json中指明的依赖,从「npm解析服务」下载项目依赖的代码。
  • 下载在线打包器(一个mini webpack)、编译器(babel)相关代码。
  • 在线打包、编译。
  • 运行打包后的代码。

正是有了在线打包、编译的流程,codesandbox才能在线运行:

  • React项目(需要编译JSX)。
  • TS项目(需要编译TS语法)。
  • Vue项目(需要编译SFC文件)。

回到本文的主题 —— 「npm解析服务」。当我们从项目package.json中获取到依赖库的名称后,完全可以从CDN直接请求依赖库对应的代码,为什么还需要一个独立的「npm解析服务」呢?

npm解析服务的作用

之所以需要独立的「npm解析服务」,主要是因为 —— npm包本身可能还依赖别的npm包,如果每次初始化iframe时依次下载:

  • package.json中指定的依赖
  • 依赖的依赖
  • 依赖的依赖的依赖
  • ...

那会极大拖慢项目初始化的时间。同时,这样做也可能会下载大量实际不会使用的代码。

所以,需要一个「npm解析服务」,当第一个用户第一次请求某个库时,依次完成:

  • 从库的入口代码解析AST,分析其中的require语句,递归的解析这个库的依赖。
  • 下载依赖代码,将所有依赖的代码汇总到一个JSON文件。
  • 将步骤2的JSON文件保存在对象存储中。
  • 返回步骤2的JSON文件。

那么,后续所有用户在请求这个库时,都能直接从对象存储中直接获取解析好的JSON文件,这能极大提高在线安装依赖的速度。

比如,react@18.2.0经由「npm解析服务」解析后会返回如下JSON:

{
  "contents": {
    "/node_modules/react/index.js": {
      // 库的代码
      "content": "...省略",
      "isModule": false,
      // 依赖的其他模块
      "requires": [
        "./cjs/react.production.min.js",
        "./cjs/react.development.js"
      ]
    },
    "/node_modules/react/cjs/react.production.min.js": {/*省略*/},
    "/node_modules/react/cjs/react.development.js": {/*省略*/},
    "/node_modules/js-tokens/package.json": {/*省略*/},
    "/node_modules/loose-envify/package.json": {/*省略*/},
    "/node_modules/react/package.json": {/*省略*/}
  },
  // 库的版本信息
  "dependency": {
    "name": "react",
    "version": "18.2.0"
  },
  "peerDependencies": {},
  // 依赖的依赖
  "dependencyDependencies": {
    "loose-envify": {/*省略*/},
    "js-tokens": {/*省略*/}
  },
  "dependencyAliases": {}
}

上述JSON中,入口代码在/node_modules/react/index.js,通过递归分析他的AST,发现他依赖了:

  • "./cjs/react.production.min.js"
  • "./cjs/react.development.js"

于是,这2个文件对应代码也包含在JSON中。

当下一个用户加载的项目依赖react@18.2.0,就能直接从对象存储中获取上述JSON。

npm解析服务的实现

codesandbox在线打包相关的代码都是开源的,比如:

  • 编辑器的部分对应sandpack-react[1]
  • npm解析服务对应dependency-packager[2]
  • 在线打包服务对应codesandbox-client[3]

所以,我们可以基于dependency-packager部署自己的「npm解析服务」。

dependency-packager是一个serverless服务,通过AWS Lambda部署。由于采用的是开源的serverless框架,所以我们可以很方便的将项目中AWS Lambda的部分替换成其他serverless服务商(比如阿里云函数计算)。

整个dependency-packager包含两个serverless函数:

  • api:实际对外提供的服务
  • packager:根据包名和版本号生成JSON的服务

他们的关系如下:

如何实现并部署自己的Npm解析服务

其中,生成的JSON保存在AWS S3中。同样,这里也可以替换成其他云服务厂家的存储方案。

packager服务的工作流程如下:

如何实现并部署自己的Npm解析服务

其中,「验证依赖的入口文件」会尝试下面这些文件后缀:

const found = [
  path.join(basedir, pkg.module),
  path.join(basedir, pkg.module + ".js"),
  path.join(basedir, pkg.module + ".cjs"),
  path.join(basedir, pkg.module + ".mjs"),
  path.join(basedir, pkg.module, "index.js"),
  path.join(basedir, pkg.module, "index.mjs"),
].find((p) => {
  try {
    const l = fs.statSync(p);
    return l.isFile();
  } catch (e) {
    return false;
  }
});

验证完成后,会以package.json中的module或mAIn字段作为入口文件,将代码转换为AST,分析AST中的require语句(cjs语法中引入模块的语法),找到依赖的模块。最终将这些模块汇总在JSON中。

总结

codesandbox在线打包相关的代码都是开源的,包括:

  • 编辑器
  • npm解析服务
  • 在线打包服务

其中,npm解析服务作为一个serverless服务包括两部分:

  • api服务
  • packager服务

packager服务代码量不多,如果想尝试部署自己的serverless服务,是个不错的选择。

参考资料

[1]sandpack-react:https://Github.com/codesandbox/sandpack/tree/main/sandpack-react。

[2]dependency-packager:https://github.com/codesandbox/dependency-packager。

[3]codesandbox-client:https://github.com/codesandbox/codesandbox-client。



Tags:Npm   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
如何实现并部署自己的Npm解析服务
大家好,我卡颂。你是否好奇 —— codesandbox是如何在线运行代码的?要回答这个问题,我们先看看前端项目是如何在本地跑起来的。简单来说分为3步: 执行npm install安装...【详细内容】
2023-10-08  Search: Npm  点击:(245)  评论:(0)  加入收藏
关于包管理器Npm、Yarn和Pnpm的一些总结
写在前面在Node.js生态系统中,包管理器是至关重要的组件之一,它们负责维护各种应用程序和库之间的依赖关系。npm是Node.js的默认包管理器,它的初始版本是npm1,但是它很快就被npm...【详细内容】
2023-04-12  Search: Npm  点击:(277)  评论:(0)  加入收藏
前端日记之npm
初始化包描述文件(创建一个package.json文件)npm init 或 npm init -y修改npm镜像地址为国内的淘宝地址npm config set registry https://registry.npm.taobao.orgnpm confi...【详细内容】
2023-02-13  Search: Npm  点击:(176)  评论:(0)  加入收藏
goland中npm无法使用的问题及解决方法
这篇文章主要介绍了goland中npm无法使用的问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教goland中npm无法使用明明已经成功安...【详细内容】
2023-01-25  Search: Npm  点击:(441)  评论:(0)  加入收藏
npm安装cnpm后使用报错解决
错误信息如下:cnpm : 无法加载文件 D:\work\nodejs\node_global\cnpm.ps1,因为在此系统上禁止运行脚本。有关详细信息,请参阅 https:/go.microsoft.com/fwlink/?LinkID=135170...【详细内容】
2021-09-01  Search: Npm  点击:(488)  评论:(0)  加入收藏
npm简介
做前端开发,如果项目达到了一定的规模,就离不开npm了。下面简单介绍一下npm的知识。1、npm的含义npm,Node Package Manager的缩写,也就是“节点的包管理器”。顾名思义,npm是一...【详细内容】
2021-04-15  Search: Npm  点击:(19050)  评论:(0)  加入收藏
73个强无敌的NPM软件包
前端框架1.ReactReact 使用虚拟 DOM 将页面中的各个部分作为单独的组件进行管理,因此您可以只刷新该组件而非整个页面。React 经常配合 React-dom 与 React-router-dom 一同...【详细内容】
2020-10-15  Search: Npm  点击:(8405)  评论:(0)  加入收藏
使用npm安装TypeScript
命令行:npm install -g typescript 新建一个TypeScript文件,粘贴如下内容:function greeter(person) { return "Hello, " + person;}var user = "Jane User";document.bod...【详细内容】
2020-08-20  Search: Npm  点击:(838)  评论:(0)  加入收藏
通过cnpm搭建自己的私仓库
看到一个面试题,原生小程序中api的调用,都是回掉函数,怎么处理成优雅的链式调用呢?那肯定是通过promise 封装起来啊,怎么封装呢?很简单,代码如下/** * * promisify化api * * @param...【详细内容】
2020-03-05  Search: Npm  点击:(458)  评论:(0)  加入收藏
「译」 npm 包管理器
npm 包管理器npm is a package manager that comes bundled with Node. Contrary to popular belief npm is not an acronym. According to the npm FAQ It provides an eas...【详细内容】
2019-09-25  Search: Npm  点击:(794)  评论:(0)  加入收藏
▌简易百科推荐
即将过时的 5 种软件开发技能!
作者 | Eran Yahav编译 | 言征出品 | 51CTO技术栈(微信号:blog51cto) 时至今日,AI编码工具已经进化到足够强大了吗?这未必好回答,但从2023 年 Stack Overflow 上的调查数据来看,44%...【详细内容】
2024-04-03    51CTO  Tags:软件开发   点击:(5)  评论:(0)  加入收藏
跳转链接代码怎么写?
在网页开发中,跳转链接是一项常见的功能。然而,对于非技术人员来说,编写跳转链接代码可能会显得有些困难。不用担心!我们可以借助外链平台来简化操作,即使没有编程经验,也能轻松实...【详细内容】
2024-03-27  蓝色天纪    Tags:跳转链接   点击:(12)  评论:(0)  加入收藏
中台亡了,问题到底出在哪里?
曾几何时,中台一度被当做“变革灵药”,嫁接在“前台作战单元”和“后台资源部门”之间,实现企业各业务线的“打通”和全域业务能力集成,提高开发和服务效率。但在中台如火如荼之...【详细内容】
2024-03-27  dbaplus社群    Tags:中台   点击:(8)  评论:(0)  加入收藏
员工写了个比删库更可怕的Bug!
想必大家都听说过删库跑路吧,我之前一直把它当一个段子来看。可万万没想到,就在昨天,我们公司的某位员工,竟然写了一个比删库更可怕的 Bug!给大家分享一下(不是公开处刑),希望朋友们...【详细内容】
2024-03-26  dbaplus社群    Tags:Bug   点击:(5)  评论:(0)  加入收藏
我们一起聊聊什么是正向代理和反向代理
从字面意思上看,代理就是代替处理的意思,一个对象有能力代替另一个对象处理某一件事。代理,这个词在我们的日常生活中也不陌生,比如在购物、旅游等场景中,我们经常会委托别人代替...【详细内容】
2024-03-26  萤火架构  微信公众号  Tags:正向代理   点击:(10)  评论:(0)  加入收藏
看一遍就理解:IO模型详解
前言大家好,我是程序员田螺。今天我们一起来学习IO模型。在本文开始前呢,先问问大家几个问题哈~什么是IO呢?什么是阻塞非阻塞IO?什么是同步异步IO?什么是IO多路复用?select/epoll...【详细内容】
2024-03-26  捡田螺的小男孩  微信公众号  Tags:IO模型   点击:(8)  评论:(0)  加入收藏
为什么都说 HashMap 是线程不安全的?
做Java开发的人,应该都用过 HashMap 这种集合。今天就和大家来聊聊,为什么 HashMap 是线程不安全的。1.HashMap 数据结构简单来说,HashMap 基于哈希表实现。它使用键的哈希码来...【详细内容】
2024-03-22  Java技术指北  微信公众号  Tags:HashMap   点击:(11)  评论:(0)  加入收藏
如何从头开始编写LoRA代码,这有一份教程
选自 lightning.ai作者:Sebastian Raschka机器之心编译编辑:陈萍作者表示:在各种有效的 LLM 微调方法中,LoRA 仍然是他的首选。LoRA(Low-Rank Adaptation)作为一种用于微调 LLM(大...【详细内容】
2024-03-21  机器之心Pro    Tags:LoRA   点击:(12)  评论:(0)  加入收藏
这样搭建日志中心,传统的ELK就扔了吧!
最近客户有个新需求,就是想查看网站的访问情况。由于网站没有做google的统计和百度的统计,所以访问情况,只能通过日志查看,通过脚本的形式给客户导出也不太实际,给客户写个简单的...【详细内容】
2024-03-20  dbaplus社群    Tags:日志   点击:(4)  评论:(0)  加入收藏
Kubernetes 究竟有没有 LTS?
从一个有趣的问题引出很多人都在关注的 Kubernetes LTS 的问题。有趣的问题2019 年,一个名为 apiserver LoopbackClient Server cert expired after 1 year[1] 的 issue 中提...【详细内容】
2024-03-15  云原生散修  微信公众号  Tags:Kubernetes   点击:(6)  评论:(0)  加入收藏
站内最新
站内热门
站内头条