9

当依赖项未声明为模块时,如何从依赖项中解决“无法在模块外使用 import 语句”?


我想使用Svelte/kit 中的验证器来验证电子邮件但是,在导入 ESM 版本时,我收到“无法在模块外使用 import 语句”错误。我正在使用 pnpm 而不是 npm 或纱线。

import isEmail from 'validator/es/lib/isEmail'
/node_modules/.pnpm/validator@13.6.0/node_modules/validator/es/lib/isEmail.js:1
import assertString from './util/assertString';
^^^^^^

SyntaxError: Cannot use import statement outside a module
    at Object.compileFunction (node:vm:355:18)
    at wrapSafe (node:internal/modules/cjs/loader:1039:15)
    at Module._compile (node:internal/modules/cjs/loader:1073:27)
    at Object.Module._extensions..js (node:internal/modules/cjs/loader:1138:10)
    at Module.load (node:internal/modules/cjs/loader:989:32)
    at Function.Module._load (node:internal/modules/cjs/loader:829:14)
    at Module.require (node:internal/modules/cjs/loader:1013:19)
    at require (node:internal/modules/cjs/helpers:93:18)
    at nodeRequire 

看来验证器正在尝试使用 import 语句,但它的 package.json 没有指定"type": "module". 我的猜测是这是错误的根本原因。

调试步骤

  • package.json 有"type": "module"
  • 升级到最新版本的节点
  • 尝试使用非 esm 版本的验证器'validator/lib/isEmail',但这会导致与此线程无关的其他错误。

有关的

元数据

  • 节点:v16.2.0
  • Sveltekit:v1.0.0-next.115
  • 验证者:13.6.0
4

5 回答 5

3

你试过这样导入吗?

import validator from 'validator'

我尝试使用最新的 SvelteKit 重现您的问题。这工作正常:

// index.svelte
<script>
    import validator from 'validator';
    let result = validator.isEmail('foo@bar.com');
    console.log(result);
</script>

当我将导入语句更改为:

import validator from 'validator/es/lib/isEmail'

我从你的问题中得到了错误(不能在模块外使用 import 语句)。

导入validator/es/lib/isEmail据说只导入库的一个子集。我不确定它会有多大的不同。它可能没有任何区别。稍大的构建击败了不起作用的构建。我建议先让它工作,然后如果你真的需要优化构建大小。

于 2021-06-23T15:26:48.180 回答
0

尝试像这样导入

import { isEmail } from "validator"
于 2021-06-18T19:09:08.730 回答
0

在没有更多信息的情况下,如果您的问题仅在于您所依赖的代码有问题,请考虑在短期内使用补丁包之类的东西进行必要的调整,从长远来看,请在validator.js 存储库中打开 PR !

对于补丁包,只需"type": "module"在 package.json 中添加node_modules/validator,然后运行npx patch-package validator​​. 然后,您可以对输出的 diff 文件进行版本控制,并且假设 patch-package 是(开发)依赖项,则使用 npm 挂钩自动进行更改。

这里库的行为或缺陷可能来自开发人员,更要记住 webpack 之类的东西,它们按照自己的条件进行 esm 导入,而不是看起来 svelte 正在使用的节点模块解析模式。或者至少问题是如何走到这一步的。(这可能是错误的!)。

于 2021-06-22T02:10:10.027 回答
0

在阅读了他们的文档后:https://github.com/validatorjs/validator.js#es6,看起来es导入中的部分是,所以它可以是 treeshakable 的。您可以通过正常导入来修复此错误:

import isURL from 'validator/lib/isURL';
于 2021-09-14T17:27:16.590 回答
-1

尝试像下面这样导入

const {isEmail} = require('validator');
于 2021-06-23T10:22:03.217 回答