如何修复必须使用导入来加载 ES 模块 discord.js

Posted

技术标签:

【中文标题】如何修复必须使用导入来加载 ES 模块 discord.js【英文标题】:How to fix Must use import to load ES Module discord.js 【发布时间】:2021-11-02 10:04:48 【问题描述】:

我正在开发一个机器人,我做了一个烤命令我收到了这个错误

internal/modules/cjs/loader.js:1089
      throw new ERR_REQUIRE_ESM(filename, parentPath, packageJsonPath);
      ^

Error [ERR_REQUIRE_ESM]: Must use import to load ES Module: C:\Users\acer\Documents\test\node_modules\node-fetch\src\index.js
require() of ES modules is not supported.
require() of C:\Users\acer\Documents\test\node_modules\node-fetch\src\index.js from C:\Users\acer\Documents\test\commands\roast.js is an ES module file as it is a .js file whose nearest parent package.json contains "type": "module" which defines all .js files in that package scope as ES modules.
Instead rename index.js to end in .cjs, change the requiring code to use import(), or remove "type": "module" from C:\Users\acer\Documents\test\node_modules\node-fetch\package.json.

←[90m    at Object.Module._extensions..js (internal/modules/cjs/loader.js:1089:13)←[39m
←[90m    at Module.load (internal/modules/cjs/loader.js:937:32)←[39m
←[90m    at Function.Module._load (internal/modules/cjs/loader.js:778:12)←[39m
←[90m    at Module.require (internal/modules/cjs/loader.js:961:19)←[39m
←[90m    at require (internal/modules/cjs/helpers.js:92:18)←[39m
    at Object.<anonymous> (C:\Users\acer\Documents\test\commands\roast.js:3:15)
←[90m    at Module._compile (internal/modules/cjs/loader.js:1072:14)←[39m
←[90m    at Object.Module._extensions..js (internal/modules/cjs/loader.js:1101:10)←[39m
←[90m    at Module.load (internal/modules/cjs/loader.js:937:32)←[39m
←[90m    at Function.Module._load (internal/modules/cjs/loader.js:778:12)←[39m 
  code: ←[32m'ERR_REQUIRE_ESM'←[39m

这是我的roast.js

const random = require("something-random-on-discord").Random
const oneLinerJoke = require('one-liner-joke');
const fetch = require('node-fetch')
const Discord = require('discord.js')
module.exports = 
    name : 'roast',
    description : 'roasts a user',
    
  async execute(message, args)
       if (!args[0]) return message.channel.send('Invalid Format')
       const mentionedMember = message.guild.mentions.member.first();
       if (!mentionedMember) return message.channel.send('User not found')
        let msg = await message.channel.send('Setting a roast...')

        fetch('http://evilinsult.com/generate_insult.php?lang=en&type=json')
            .then(res => res.json())
            .then(json => 
                message.channel.send(json.insult)
            );
    


这是我的 main.js

// Import the discord.js module
const Discord = require('discord.js');
const fs = require('fs');
// Create an instance of a Discord client
const client = new Discord.Client();
client.commands = new Discord.Collection();
const commandFiles = fs.readdirSync('./commands').filter(file => file.endsWith('.js'));
const prefix = "$"

/**
 * The ready event is vital, it means that only _after_ this will your bot start reacting to information
 * received from Discord
 */
client.on('ready', () => 
  console.log('I am ready!');
);
 
client.on('message', message => 
  if (!message.content.startsWith(prefix) || message.author.bot) return;

  const args = message.content.slice(prefix.length).trim().split(/ +/);
  const command = args.shift().toLowerCase();


  if(command === 'pingg')
    client.commands.get('pingg').execute(message, args);
  
  if(command === 'roast')
    client.commands.get('roast').execute(message, args);
  




  if (!client.commands.has(command)) return;

  try 
      client.commands.get(command).execute(message, args);
   catch (error) 
      console.error(error);
      message.reply('there was an error trying to execute that command!');
  
);
 

commandFiles.forEach(file => 
  const command = file.split(/.js$/)[0];
  client.commands.set(command, require(`./commands/$file`));
);



client.login('censored');

【问题讨论】:

解决方法在错误信息中:Instead rename index.js to end in .cjs, change the requiring code to use import(), or remove "type": "module" node-fetch 版本 3(当前最新)不允许需要蜜蜂 希望对您有所帮助:gist.github.com/sindresorhus/a39789f98801d908bbc7ff3ecc99d99c 【参考方案1】:

如错误所示:

package.json contains "type": "module" which defines all .js files in that package scope as ES modules

你通常有 3 个选项:

    尝试从 package.json 中删除这一行,以便在您的代码中仅使用 requires。 或保留此行并在您的应用 (ES6) 中的所有位置使用 import 语句而不是 require 语句。 将您的文件重命名为 index.cjs,然后在此特定文件中允许使用 require,而您仍然必须在其他文件中使用 import。

在您的特定情况下,您使用的是 node-fetch,它现在仅是 esmodule(从版本 3 开始)。 那么唯一可用的解决方案就是在任何地方使用 import。

requires 应该变成这样:

import Random from 'something-random-on-discord';
import oneLinerJoke from 'one-liner-joke';
import fetch from 'node-fetch';
import Discord from 'discord.js';

编辑: 您还可以删除节点获取或恢复到分支 v2 以继续使用 require。

【讨论】:

"你通常有 2 个选项:" 你有 3 个选项:Instead rename index.js to end in .cjs, change the requiring code to use import(), or remove "type": "module" 当我使用导入语句时,我得到import Random from 'something-random-on-discord'; ^^^^^^ SyntaxError: Cannot use import statement outside a module 我在 package.json 中没有 "type": "module" 我的错。这是节点获取的错误。但解决方案仍然是添加它并转换为导入。 我更新了我的答案以反映这个新的解决方案:) 你能把问题标记为已解决吗【参考方案2】:

正如README.md 中提到的那样... 另一种实现 fetch@3 的方法是使用节点 v12.20 中支持的异步 import() 导入 node-fetch

const fetch = (...args) => import('node-fetch').then((default: fetch) => fetch(...args));

这也适用于 commonjs(下一个请求不会重新导入 node-fetch 因为模块被缓存) 这可以使 Node 进程启动更快,并且只在需要时延迟加载 node-fetch

这是另一种预加载方式:

const fetchP = import('node-fetch').then(mod => mod.default)
const fetch = (...args) => fetchP.then(fn => fn(...args))

您不必将您的孔项目转换为 ESM,只需 b/c 我们做到了。您也可以继续使用 v2 分支 - 我们将不断更新 v2 的错误/安全问题。但没有那么多新功能......

(还是推荐别人转ESM doe)

对于只关注 Types 的 TypeScript 用户,你可以这样做: import type Request from 'node-fetch'(这不会导入或转译任何东西 - 这个输入注释将被丢弃)

#1279有一个关于如何在 cjs 项目中导入/要求节点获取的漏洞部分

【讨论】:

名称 fetchP 背后的目的是什么?比如 P 代表 Promise 是什么意思? 是的,p 代表承诺:P

以上是关于如何修复必须使用导入来加载 ES 模块 discord.js的主要内容,如果未能解决你的问题,请参考以下文章

ElectronJS:必须使用导入来加载 ES 模块

将 CommonJS 默认导出导入为命名导出/无法加载 ES 模块

将 D3.js 7.0.0 与 Next.js 11.0.1 一起使用时,如何解决“[ERR_REQUIRE_ESM]:必须使用导入来加载 ES 模块”?

在 Node.js 中导入:错误“必须使用导入来加载 ES 模块”[重复]

npx create-react-app 不工作“必须使用导入来加载 ES 模块:” [关闭]

如何使用 ES6 模块导入来导入路径