MySQL 8.0-客户端不支持服务器请求的身份验证协议;考虑升级MySQL客户端


272

由于某种原因,我无法与服务器建立简单连接。我使用默认设置安装了最新的MySQL Community 8.0数据库以及Node.JS。

这是我的node.js代码

    var mysql = require('mysql');

    var con = mysql.createConnection({
      host: "localhost",
      user: "root",
      password: "password",
      insecureAuth : true
    });

    con.connect(function(err) {
      if (err) throw err;
      console.log("Connected!");
    });

以下是在命令提示符中发现的错误:

C:\Users\mysql-test>node app.js
    C:\Users\mysql-test\node_modules\mysql\lib\protocol\Parse
    r.js:80
            throw err; // Rethrow non-MySQL errors
            ^

Error: ER_NOT_SUPPORTED_AUTH_MODE: Client does not support authentication protocol requested by server; consider upgrading MySQL client
    at Handshake.Sequence._packetToError (C:\Users\mysql-
test\node_modules\mysql\lib\protocol\sequences\Sequence.js:52:14)
    at Handshake.ErrorPacket (C:\Users\mysql-test\node_mo
dules\mysql\lib\protocol\sequences\Handshake.js:130:18)
    at Protocol._parsePacket (C:\Users\mysql-test\node_mo
dules\mysql\lib\protocol\Protocol.js:279:23)
    at Parser.write (C:\Users\mysql-test\node_modules\mys
ql\lib\protocol\Parser.js:76:12)
    at Protocol.write (C:\Users\mysql-test\node_modules\m
ysql\lib\protocol\Protocol.js:39:16)
    at Socket.<anonymous> (C:\Users\mysql-test\node_modul
es\mysql\lib\Connection.js:103:28)
    at Socket.emit (events.js:159:13)
    at addChunk (_stream_readable.js:265:12)
    at readableAddChunk (_stream_readable.js:252:11)
    at Socket.Readable.push (_stream_readable.js:209:10)
    --------------------
    at Protocol._enqueue (C:\Users\mysql-test\node_module
s\mysql\lib\protocol\Protocol.js:145:48)
    at Protocol.handshake (C:\Users\mysql-test\node_modul
es\mysql\lib\protocol\Protocol.js:52:23)
    at Connection.connect (C:\Users\mysql-test\node_modul
es\mysql\lib\Connection.js:130:18)
    at Object.<anonymous> (C:\Users\mysql-test\server.js:
11:5)
at Module._compile (module.js:660:30)
at Object.Module._extensions..js (module.js:671:10)
at Module.load (module.js:573:32)
at tryModuleLoad (module.js:513:12)
at Function.Module._load (module.js:505:3)
at Function.Module.runMain (module.js:701:10)

我已经阅读了一些内容,例如:https : //dev.mysql.com/doc/refman/5.5/en/old-client.html https://github.com/mysqljs/mysql/issues/1507

但是我仍然不确定如何解决我的问题。任何帮助,将不胜感激:D


2
所有使用SQLTools扩展名的VS Code用户都应参阅此文章,以防他们遇到问题,尤其是如果您刚刚在开发计算机上升级了mysql实例
OzzyTheGiant

Answers:


676

在MYSQL Workbench中执行以下查询

ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY 'password'

root作为用户 localhost的您的网址,并password为您的密码

然后运行此查询以刷新特权:

flush privileges;

完成后尝试使用节点连接。

如果那行不通,请尝试一下@'localhost'


47
这对我有用。不要忘记flush privileges;以后。
阿里·赫萨里

12
它在没有@localhost的情况下对我有用(我想可能是因为它不是针对根用户的)
NicolasZ

10
如果您知道的话,用您的根密码替换“密码”
Vedha Peri 18-10-12

11
它可以工作……但是为什么它在此查询之前不起作用以及为什么它在此查询之后仍然起作用仍然是一个问题。
saksham

9
@GuilhermeMatuella,您好:这是因为caching_sha2_passwordMySQL 8.0中引入了该功能,但尚未实现Node.js版本。
Daksh Gargas

105

首先让我们弄清楚发生了什么。

MySQL 8支持可插入身份验证方法。默认情况下,将使用其中一个命名的名称caching_sha2_password,而不是我们的旧版本mysql_native_password源代码)。很明显,使用加密算法并多次握手比使用普通密码传递24年来更加安全!

现在,问题出mysqljs在Node(用于安装npm i mysql和在Node代码中使用的软件包)尚不支持MySQL 8的这种新的默认身份验证方法。问题在这里:https//github.com/mysqljs/mysql/issues/1507,并且在3年后的2019年7月仍然开放。

(2019年6月更新:mysqljs中现在有一个新的PR来解决这个问题! (2020年2月更新:显然它计划在mysqljs版本3中推出

您的选择

选项1)使用旧的“ native_password”降级“ MySQL”以进行身份​​验证

这就是每个人在这里提出的建议(例如,上方的最佳答案)。您只需进入mysql并运行查询,然后root使用旧native_password方法进行身份验证就可以了:

ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password ...

好消息是,生活会变得很简单,您仍然可以使用Sequel Pro这样的旧工具而不会出现任何问题。但是问题是,您没有利用可用的更安全的东西(并且很酷,请阅读下文)。

选项2)用MySQL Connecter X DevAPI替换“ Node”包

MySQL X DevAPI for Node是由http://dev.mysql.com官方提供的Node的Mysqljs软件包的替代品。

它就像支持caching_sha2_password身份验证的超级按钮一样工作。(只需确保将端口33060用于X协议通信即可。)

不好的是,您已经离开了我们mysql每个人都习惯并依赖的旧程序包。

好消息是,您的应用现在更安全了,您可以利用我们的好朋友没有提供的大量新功能!只需查看X DevAPI教程,您就会发现它具有大量可以方便使用的新性感功能。您只需要付出学习曲线的代价即可,这预计会随任何技术升级一起出现。:)

PS。不幸的是,这个XDevAPI包还没有类型定义(TypeScript可以理解),因此,如果您使用的是Typescript,则会遇到问题。我尝试使用dts-gen和生成.d.ts dtsmake,但未成功。所以记住这一点。

干杯!


3
感谢您发布有关一岁问题的详细答案。:]
Daksh Gargas

3
不错,一个答案实际上是在解释mysql_native_password而不是仅仅告诉您这样做
wizloc

1
是的,一次正确的解释。很好,你爱丁。我希望如何连接应用程序不会自动将端口3306配置为MySQL。我们有很多麻烦将MySQL和MariaDB的版本从3306撤离。–
Trunk

:小心甲骨文的插件,你可能会得到一个很大的惩罚theregister.co.uk/2019/10/04/oracle_virtualbox_merula
尤哈Untinen

这是一个很好的答案,值得更多的支持!
6月温斯顿

82

使用旧mysql_native_password作品:

ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY 'YourRootPassword';
-- or
CREATE USER 'foo'@'%' IDENTIFIED WITH mysql_native_password BY 'bar';
-- then
FLUSH PRIVILEGES;

这是因为caching_sha2_passwordMySQL 8.0中引入了该功能,但尚未实现Node.js版本。您可以查看此拉取请求此问题以获取更多信息。可能很快就会有修复程序!


像魅力一样工作。终于解脱了!
帽子

添加@localhost和仅'root'之间有什么区别
filemonczyk

查询有效。但是仍然节点无法连接到MySQL服务器。任何想法?这是错误,{“代码”:“ ER_ACCESS_DENIED_ERROR”,“ errno”:1045,“ sqlMessage”:“对用户'root'@'localhost'的访问被拒绝(使用密码:是)”,“ sqlState”:“ 28000” ,“ fatal”:true}
Sanjay Pradeep

26

MySQL 8的完整步骤

连接到MySQL

$ mysql -u root -p
Enter password: (enter your root password)

重置你的密码

(替换your_new_password为您要使用的密码)

mysql> ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY 'your_new_password';
mysql> FLUSH PRIVILEGES;
mysql> quit

然后尝试使用节点连接


@sky,请参阅上面的详细信息。
艾丁(Aidin)

1
@Aidin谢谢,这非常有帮助。
天空

17

尽管接受的答案是正确的,但我还是希望创建一个新用户,然后使用该用户访问数据库。

create user nodeuser@localhost identified by 'nodeuser@1234';
grant all privileges on node.* to nodeuser@localhost;
ALTER USER 'nodeuser'@localhost IDENTIFIED WITH mysql_native_password BY 'nodeuser@1234';

2
如果有人想走这条路,这是一篇带有一些解释的文章:digitalocean.com/community/tutorials/…–
Devin

12

如果遇到此错误,但仍想使用MySQL版本8,则可以通过在创建数据库时告诉MySQL服务器使用旧版身份验证插件来实现此目的Docker

因此,您的compose文件将如下所示:

# Use root/example as user/password credentials

version: '3.1'

services:

  db:
    image: mysql:8.0.15
    command: --default-authentication-plugin=mysql_native_password
    restart: always
    environment:
       MYSQL_ROOT_PASSWORD: 'pass'
       MYSQL_DATABASE: 'db'
       MYSQL_USER: 'user'
       MYSQL_PASSWORD: 'pass'
    ports:
      - 3318:3306
    # Change this to your local path
    volumes:
      - ~/Database/ORM_Test:/var/lib/mysql

我正在使用它command,但是mysql在Node.js中使用时仍然收到不受支持的身份验证错误消息
Juha Untinen

完整的堆栈跟踪在这里:pastebin.com/SzayQzdhdocker-compose.ymlpastebin.com/7pUrWYDs错误本身是:Error: ER_NOT_SUPPORTED_AUTH_MODE: Client does not support authentication protocol requested by server; consider upgrading MySQL client
Juha Untinen

1
@JuhaUntinen确保已删除Docker容器(由docker-compose创建的容器)和卷(〜/ Database / ORM_Test),然后再次运行“ docker-compose up”。否则,将不会应用“命令”中的更改。对我来说,这很有帮助。
Vitalii Ivanov,

11

在Mysql最新的docker容器中

ALTER USER root IDENTIFIED WITH mysql_native_password BY 'password';

9

如果ALTER USER ...命令行对您不起作用,并且您使用的是Windows 10,请尝试执行以下步骤:

1)在Windows搜索栏中输入MySQL

2)打开MySQL Windows Installer-社区

3)查找“ MySQL服务器”,然后单击“重新配置” 第三步

4)单击“下一步”,直到到达“身份验证方法”阶段

5)在“身份验证方法”阶段,选中第二个选项“使用旧版身份验证方法” 第5步

6)然后按照Windows安装程序给出的步骤进行操作,直到结束

7)完成后,从Windows搜索栏中进入“服务”,单击“开始”“ MySql81”。

现在,再试一次,MySQL和Node.js之间的连接应该正常工作!


4

在MySQL 8.0中,caching_sha2_password是默认的身份验证插件,而不是mysql_native_password. ...

该问题的大多数答案都将身份验证机制从降级caching_sha2_passwordmysql_native_password。从安全角度来看,这是非常令人失望的。

文档广泛讨论了caching_sha2_password,当然也为什么不应该将其作为降级身份验证方法的首选。

这样,我相信爱丁的答案应该是公认的答案。不要使用与服务器版本匹配的连接器来降级身份验证方法。


3

您可以在这里找到原始文档:https : //dev.mysql.com/doc/dev/connector-nodejs/8.0/

'use strict';

const mysqlx = require('@mysql/xdevapi');

const options = {
  host: 'localhost',
  port: 33060,
  password: '******',
  user: 'root',
  schema: 'yourconference'
};

mysqlx.getSession(options)
  .then(session => {
          console.log(session.inspect());
           session.close();
  }).catch(err => {
    console.error(err.stack);
    process.exit(1);
  });

它看起来更复杂,但我认为我们应该遵循官方指南!b
Juneyoung,

3

我在服务器上有MYSQL,在另一台服务器上有nodejs应用程序

在MYSQL Workbench中执行以下查询

ALTER USER'root'@'%'由mysql_native_password标识为'password'


2

对于在Windows 10 mysql上安装的现有mysql 8.0,

(1)启动安装程序,

(2)单击“快速操作”(位于MySQL服务器左侧)下的“重新配置”,然后

(3)单击“下一步”前进到接下来的两个屏幕,直到到达

(4)在“身份验证方法”中,选择“使用旧式身份验证方法(保持MySQL 5.x兼容性)”

(5)持续单击直到安装完成


2

在MySQL 8+中,新的默认身份验证caching_sha2_password代替mysql_native_password。本机mysql软件包尚不支持新的,更安全的身份验证方法,但是您应该考虑使用该软件包@mysql/xdevapi,Oracle正式支持和维护该软件包。

要安装新软件包,请运行:

npm install @mysql/xdevapi --save --save-exact

要连接到数据库并插入一些值:

const mysqlx = require('@mysql/xdevapi');
var myTable;

mysqlx
    .getSession({
        user: 'root',
        password: '*****',
        host: 'localhost',
        port: 33060
    })
    .then(function (session) {

    // Accessing an existing table
    myTable = session.getSchema('Database_Name').getTable('Table_Name');

    // Insert SQL Table data
    return myTable
        .insert(['first_name', 'last_name'])
        .values(['John', 'Doe'])
        .execute()
    });

官方的软件包文档可以在这里找到:https : //dev.mysql.com/doc/dev/connector-nodejs/8.0/


2

如果您使用的是docker,那么它对我有用!

docker-compose.yml添加以下行:

mysql:
   ...    
   command: --default-authentication-plugin=mysql_native_password
   restart: always

之后,再来一次down容器up


1

除上述答案外; 执行以下命令后

ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY 'password'

如果出现以下错误:

[ERROR] Column count of mysql.user is wrong. Expected 42, found 44. The table is probably corrupted

然后以管理员身份尝试cmd;在cmd中设置MySQL服务器bin文件夹的路径

set path=%PATH%;D:\xampp\mysql\bin;

然后运行命令:

mysql_upgrade --force -uroot -p

这将更新服务器和系统表。

然后,您应该能够在工作台的查询中成功运行以下命令:

 ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY 'password'    

然后记住执行以下命令:

flush privileges;

完成所有这些步骤后,便应该能够成功连接到您的MySQL数据库。希望这可以帮助...



0

检查您的MySQL用户的特权和用户名/密码。

对于捕获错误,使用覆盖_delegateError方法总是有用的。在您的情况下,它必须看起来像:

var mysql = require('mysql');

var con = mysql.createConnection({
  host: "localhost",
  user: "root",
  password: "password",
  insecureAuth : true
});

var _delegateError = con._protocol._delegateError;

con._protocol._delegateError = function(err, sequence) {
    if (err.fatal)
        console.trace('MySQL fatal error: ' + err.message);

    return _delegateError.call(this, err, sequence);
};

con.connect(function(err) {
  if (err) throw err;

  console.log("Connected!");
});

这种构造将帮助您跟踪致命错误。


0

只是在尝试了无数次尝试之后才弄清楚了。最终对我来说是添加require('dotenv').config().sequelizerc文件中。显然sequelize-cli不读取env变量。


0

降级可能不是一个好的选择,因为:

  1. 其升级是有原因的(以提供更好的身份验证)。
  2. 您可能没有足够的权限进行此类更改。

您可以使用mysql2package代替mysql。它的大多数API与mysqljs兼容。此外,它也有希望的支持。

像这样使用它: const mysql = require('mysql2/promise')

您可以mysql2在此处了解更多信息:https : //www.npmjs.com/package/mysql2

希望能帮助到你。:)


0

您可以跳过ORM,构建器等,并使用sqler和简化DB / SQL管理sqler-mdb

-- create this file at: db/mdb/read.table.rows.sql
SELECT TST.ID AS "id", TST.NAME AS "name", NULL AS "report",
TST.CREATED_AT AS "created", TST.UPDATED_AT AS "updated"
FROM TEST TST
WHERE UPPER(TST.NAME) LIKE CONCAT(CONCAT('%', UPPER(:name)), '%') 
const conf = {
  "univ": {
    "db": {
      "mdb": {
        "host": "localhost",
        "username":"admin",
        "password": "mysqlpassword"
      }
    }
  },
  "db": {
    "dialects": {
      "mdb": "sqler-mdb"
    },
    "connections": [
      {
        "id": "mdb",
        "name": "mdb",
        "dir": "db/mdb",
        "service": "MySQL",
        "dialect": "mdb",
        "pool": {},
        "driverOptions": {
          "connection": {
            "multipleStatements": true
          }
        }
      }
    ]
  }
};

// create/initialize manager
const manager = new Manager(conf);
await manager.init();

// .sql file path is path to db function
const result = await manager.db.mdb.read.table.rows({
  binds: {
    name: 'Some Name'
  }
});

console.log('Result:', result);

// after we're done using the manager we should close it
process.on('SIGINT', async function sigintDB() {
  await manager.close();
  console.log('Manager has been closed');
});

-1

我在MySQL上也遇到了同样的问题,我通过使用XAMPP与MySQL连接并在Windows上的MySQL(控制面板-管理工具-服务)窗口以及db.js文件夹(负责数据库)中停止服务来解决该问题。将密码设为空(在这里您可以看到:)

const mysql = require('mysql');
const connection = mysql.createConnection({
  host: 'localhost',
  user: 'root',
  password: ''
});

请不要添加您面临的问题作为所面临问题的答案
Aloy A Sen
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.