11
7

MySQL 8.3環境でmysql2 gemをインストールする方法

Last updated at Posted at 2024-04-17

はじめに

MySQL 8.3の環境でmysql2 gemをインストールするのにちょっと苦労したので、対処法を書いておきます。

実行環境

本記事は以下の環境で検証しました。

  • MySQL 8.3.0 (Homebrewでインストール)
  • macOS 14.4.1
  • MacBook Air M3
  • Ruby 3.3.0

注意事項

この記事を書いている時点ではmysql2の最新バージョンは0.5.6です。
よって、mysql2 0.5.6を使う前提で記事を書きますが、さらに新しいバージョンがリリースされたときは対処法が異なる可能性があります。

念のため用語の整理

本記事ではMySQLとmysql2というよく似た用語が出てきます。
プログラミング初心者の方はこれらをしっかり区別しながら読み進めてください。

  • MySQL = RDBMS(DBサーバー)。執筆時点の最新バージョンは8.3.0(2024/1/16リリース)
  • mysql2 = MySQLをRubyで操作するためのgem。執筆時点の最新バージョンは0.5.6(2024/2/16リリース)

TL;DR(長いので最初に結論)

対処法をざっくりまとめると以下の通りです。

  • mysql2 0.5.5を使っている場合はbundle update mysql2で0.5.6以上にする
  • それでもエラーになる(mysql2 0.5.6にアップデートできない)場合は以下のコマンドでmysql2 0.5.6(もしくはそれよりも新しい最新バージョン)を単体でインストールする
gem install mysql2 -v 0.5.6 -- --with-ldflags=-L$(brew --prefix zstd)/lib
  • インストールに成功したら再度bundle update mysql2を実行する

ここから下では詳細な対処法を説明します。

mysql2 0.5.5を使っていると発生するエラー

mysql2 gemを使っているRailsプロジェクトでbundle installを実行すると以下のようなエラーが出ました。

Fetching mysql2-0.5.5.gem
Building native extensions. This could take a while...
ERROR:  Error installing mysql2:
	ERROR: Failed to build gem native extension.

    current directory: /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.5/ext/mysql2
/Users/jnito/.rbenv/versions/3.3.0/bin/ruby extconf.rb
checking for rb_absint_size()... yes
checking for rb_absint_singlebit_p()... yes
checking for rb_gc_mark_movable()... yes
checking for rb_wait_for_single_fd()... yes
checking for rb_enc_interned_str() in ruby.h... yes
-----
Using --with-openssl-dir=/opt/homebrew/opt/openssl@3
-----
-----
Using mysql_config at /opt/homebrew/bin/mysql_config
-----
checking for mysql.h... yes
checking for errmsg.h... yes
checking for SSL_MODE_DISABLED in mysql.h... yes
checking for SSL_MODE_PREFERRED in mysql.h... yes
checking for SSL_MODE_REQUIRED in mysql.h... yes
checking for SSL_MODE_VERIFY_CA in mysql.h... yes
checking for SSL_MODE_VERIFY_IDENTITY in mysql.h... yes
checking for MYSQL.net.vio in mysql.h... yes
checking for MYSQL.net.pvio in mysql.h... no
checking for MYSQL_DEFAULT_AUTH in mysql.h... yes
checking for MYSQL_ENABLE_CLEARTEXT_PLUGIN in mysql.h... yes
checking for SERVER_QUERY_NO_GOOD_INDEX_USED in mysql.h... yes
checking for SERVER_QUERY_NO_INDEX_USED in mysql.h... yes
checking for SERVER_QUERY_WAS_SLOW in mysql.h... yes
checking for MYSQL_OPTION_MULTI_STATEMENTS_ON in mysql.h... yes
checking for MYSQL_OPTION_MULTI_STATEMENTS_OFF in mysql.h... yes
checking for my_bool in mysql.h... no
-----
Don't know how to set rpath on your system, if MySQL libraries are not in path mysql2 may not load
-----
-----
Setting libpath to /opt/homebrew/Cellar/mysql/8.3.0_1/lib
-----
creating Makefile

current directory: /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.5/ext/mysql2
make DESTDIR\= sitearchdir\=./.gem.20240418-36184-4r35cx sitelibdir\=./.gem.20240418-36184-4r35cx clean

current directory: /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.5/ext/mysql2
make DESTDIR\= sitearchdir\=./.gem.20240418-36184-4r35cx sitelibdir\=./.gem.20240418-36184-4r35cx
compiling client.c
In file included from client.c:15:
./mysql_enc_name_to_ruby.h:43:1: warning: a function definition without a prototype is deprecated in all versions of C and is not supported in C2x [-Wdeprecated-non-prototype]
mysql2_mysql_enc_name_to_rb_hash (str, len)
^
./mysql_enc_name_to_ruby.h:86:1: warning: a function definition without a prototype is deprecated in all versions of C and is not supported in C2x [-Wdeprecated-non-prototype]
mysql2_mysql_enc_name_to_rb (str, len)
^
client.c:1438:3: error: call to undeclared function 'mysql_ssl_set'; ISO C99 and later do not support implicit function declarations [-Wimplicit-function-declaration]
  mysql_ssl_set(wrapper->client,
  ^
client.c:1438:3: note: did you mean 'mysql_close'?
/opt/homebrew/Cellar/mysql/8.3.0_1/include/mysql/mysql.h:797:14: note: 'mysql_close' declared here
void STDCALL mysql_close(MYSQL *sock);
             ^
2 warnings and 1 error generated.
make: *** [client.o] Error 1

make failed, exit code 2

Gem files will remain installed in /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.5 for inspection.
Results logged to /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/extensions/arm64-darwin-23/3.3.0/mysql2-0.5.5/gem_make.out

エラーの原因は以下の部分のようです。

client.c:1438:3: error: call to undeclared function 'mysql_ssl_set'; ISO C99 and later do not support implicit function declarations [-Wimplicit-function-declaration]
  mysql_ssl_set(wrapper->client,
  ^

上のエラーメッセージに出ているmysql_ssl_setという関数はMySQL 8.3で削除されたようです。

mysql_ssl_set was deprecated on 8.2 and removed on 8.3.
https://github.com/brianmario/mysql2/issues/1346

ただし、この問題はmysql2 0.5.6で解決されています。

Support for libmysqlclient 8.3
https://github.com/brianmario/mysql2/releases/tag/0.5.6

mysql2 0.5.6で発生するエラー

そこで、mysql2 gemのバージョンを上げてみます。

bundle update mysql2

すると今後は別のエラーが出ました。

Fetching mysql2-0.5.6.gem
Building native extensions. This could take a while...
ERROR:  Error installing mysql2:
	ERROR: Failed to build gem native extension.

    current directory: /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.6/ext/mysql2
/Users/jnito/.rbenv/versions/3.3.0/bin/ruby extconf.rb
checking for rb_absint_size()... yes
checking for rb_absint_singlebit_p()... yes
checking for rb_gc_mark_movable()... yes
checking for rb_wait_for_single_fd()... yes
checking for rb_enc_interned_str() in ruby.h... yes
-----
Using --with-openssl-dir=/opt/homebrew/opt/openssl@3
-----
-----
Using mysql_config at /opt/homebrew/bin/mysql_config
-----
checking for mysql.h... yes
checking for errmsg.h... yes
checking for SSL_MODE_DISABLED in mysql.h... yes
checking for SSL_MODE_PREFERRED in mysql.h... yes
checking for SSL_MODE_REQUIRED in mysql.h... yes
checking for SSL_MODE_VERIFY_CA in mysql.h... yes
checking for SSL_MODE_VERIFY_IDENTITY in mysql.h... yes
checking for MYSQL.net.vio in mysql.h... yes
checking for MYSQL.net.pvio in mysql.h... no
checking for MYSQL_DEFAULT_AUTH in mysql.h... yes
checking for MYSQL_ENABLE_CLEARTEXT_PLUGIN in mysql.h... yes
checking for SERVER_QUERY_NO_GOOD_INDEX_USED in mysql.h... yes
checking for SERVER_QUERY_NO_INDEX_USED in mysql.h... yes
checking for SERVER_QUERY_WAS_SLOW in mysql.h... yes
checking for MYSQL_OPTION_MULTI_STATEMENTS_ON in mysql.h... yes
checking for MYSQL_OPTION_MULTI_STATEMENTS_OFF in mysql.h... yes
checking for my_bool in mysql.h... no
checking for mysql_ssl_set() in mysql.h... no
-----
Don't know how to set rpath on your system, if MySQL libraries are not in path mysql2 may not load
-----
-----
Setting libpath to /opt/homebrew/Cellar/mysql/8.3.0_1/lib
-----
creating Makefile

current directory: /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.6/ext/mysql2
make DESTDIR\= sitearchdir\=./.gem.20240418-36441-xndr2u sitelibdir\=./.gem.20240418-36441-xndr2u clean

current directory: /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.6/ext/mysql2
make DESTDIR\= sitearchdir\=./.gem.20240418-36441-xndr2u sitelibdir\=./.gem.20240418-36441-xndr2u
compiling client.c
In file included from client.c:15:
./mysql_enc_name_to_ruby.h:43:1: warning: a function definition without a prototype is deprecated in all versions of C and is not supported in C2x [-Wdeprecated-non-prototype]
mysql2_mysql_enc_name_to_rb_hash (str, len)
^
./mysql_enc_name_to_ruby.h:86:1: warning: a function definition without a prototype is deprecated in all versions of C and is not supported in C2x [-Wdeprecated-non-prototype]
mysql2_mysql_enc_name_to_rb (str, len)
^
2 warnings generated.
compiling infile.c
compiling mysql2_ext.c
compiling result.c
result.c:304:35: warning: implicit conversion loses integer precision: 'unsigned long' to 'int' [-Wshorten-64-to-32]
        precision = field->length - (field->decimals > 0 ? 2 : 1);
                  ~ ~~~~~~~~~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
1 warning generated.
compiling statement.c
linking shared-object mysql2/mysql2.bundle
ld: library 'zstd' not found
clang: error: linker command failed with exit code 1 (use -v to see invocation)
make: *** [mysql2.bundle] Error 1

make failed, exit code 2

Gem files will remain installed in /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/gems/mysql2-0.5.6 for inspection.
Results logged to /Users/jnito/.rbenv/versions/3.3.0/lib/ruby/gems/3.3.0/extensions/arm64-darwin-23/3.3.0/mysql2-0.5.6/gem_make.out

どうやらzstdというライブラリが見つからないようです。

ld: library 'zstd' not found
clang: error: linker command failed with exit code 1 (use -v to see invocation)

まず、zstdがインストールされているかどうか確認しましょう。

$ which zstd
/opt/homebrew/bin/zstd

もしzstdが見つからなければインストールしてください。

brew install zstd

そして、以下のコマンドでmysql2 0.5.6をインストールします(参考)。

gem install mysql2 -v 0.5.6 -- --with-ldflags=-L$(brew --prefix zstd)/lib

インストールが成功したことを確認します。

Building native extensions with: '--with-ldflags=-L/opt/homebrew/opt/zstd/lib'
This could take a while...
Successfully installed mysql2-0.5.6
1 gem installed

それから、Railsプロジェクトのmysql2をアップデートしてください。

bundle update mysql2

これで問題なくmysql2 gemが使えるようになるはずです。

参考文献

おまけ:8.0にダウングレードしたMySQLを8.3に戻す

mysql2 0.5.6がリリースされる前はMySQL 8.3から8.0にダウングレードしろ、という解決策が提示されていました。

わざわざ古いバージョンを使用するのはあまり好ましくない対処法なので、もしダウングレードしていた場合はMySQL 8.3(もしくはそれよりも新しい最新バージョン)に上げ直しましょう。

手順は以下の通りです。

  • brew uninstall mysql@8.0 でMySQL 8.0をアンインストール
  • ~/.zshrcを開いて以下の行があれば削除
export PATH="/opt/homebrew/opt/mysql@8.0/bin:$PATH"
  • ターミナルを再起動
  • brew install mysqlで最新のMySQLをインストール
  • brew services start mysqlでMySQLを起動(もしくはbrew services restart mysqlで再起動)
  • MySQL Serverのバージョンが8.3であることを確認
$ mysqld --version
/opt/homebrew/Cellar/mysql/8.3.0_1/bin/mysqld  Ver 8.3.0 for macos14.2 on arm64 (Homebrew)
  • MySQL Clientのバージョンも8.3であることを確認
$ mysql --version
mysql  Ver 8.3.0 for macos14.2 on arm64 (Homebrew)
  • mysql -u rootでサーバに接続できることを確認

ここまで確認したら、本記事の「mysql2 0.5.6で発生するエラー」で説明している対処法を実施してください。

11
7
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
11
7