質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

87.78%

mysql.connector.errors.OperationalError: Mysql Connection not available. が解決できない

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 4,871

score 104

前提・実現したいこと

DockerのMySQLイメージからコンテナを作成してDBを運用しています。
基本的には問題なく動いているのですが作成から翌日までそのままにして
翌朝アクセスしようとするとエラーが出る問題を発見しました。
MySQLとの接続が切れているといったことを言われるのですが解決策がわからなかったため質問させていただきます。

あれから毎日この問題に悩まされています。
MySQL自体はい動いているように思えます。

gunicorn message:app --config gunicornconf.py

で接続を行ったままで放置し、翌朝アクセスするとこのエラーが出るようになっています。

Ctrl+Z
pkill -SIGKILL -f gunicorn

で一度切断した後、再度接続すると元通り復活します。

何をどうしたら起動しっぱなしで外部からのアクセスを常に受け取れる状態にして置けるんですか?

発生している問題・エラーメッセージ

[2016-09-20 01:15:24 +0000] [14653] [ERROR] Error handling request /message_get
Traceback (most recent call last):
 File "/root/.pyenv/versions/3.5.2/lib/python3.5/site-packages/gunicorn/workers/sync.py", line 135, in handle
   self.handle_request(listener, req, client, addr)
 File "/root/.pyenv/versions/3.5.2/lib/python3.5/site-packages/gunicorn/workers/sync.py", line 176, in handle_request
   respiter = self.wsgi(environ, resp.start_response)
 File "/root/.pyenv/versions/3.5.2/lib/python3.5/site-packages/falcon/api.py", line 189, in __call__
   responder(req, resp, **params)
 File "/var/www/apps/message.py", line 31, in on_get
   msg = self.showMessage()
 File "/var/www/apps/message.py", line 21, in getMessage
   cursor = self.connector.cursor()
 File "/root/.pyenv/versions/3.5.2/lib/python3.5/site-packages/mysql/connector/connection.py", line 809, in cursor
   raise errors.OperationalError("MySQL Connection not available.")
mysql.connector.errors.OperationalError: MySQL Connection not available.

補足情報(言語/FW/ツール等のバージョンなど)

Python3 
gunicorn
falcon
MySQL5

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • mahata

    2016/09/27 14:35

    接続が切れたときは MySQL コンテナはまだ動いているのでしょうか? `docker ps` で何か動いていることが確認できますか?

    キャンセル

  • suvera

    2016/09/27 17:24

    動いています。
    docker ps でコンテナが生きていることは確認しました。

    キャンセル

  • popobot

    2016/09/27 18:46

    全然わかってないですが、これと同じ問題ですか?
    http://aroundthedistance.hatenadiary.jp/entry/2015/04/27/121230

    キャンセル

  • suvera

    2016/09/28 09:05

    自分も問題が理解できているわけではないですが
    同じ問題に思えます。

    キャンセル

回答 2

+1

時間がたつとエラーになるということは、サーバ側の無通信タイムアウトで強制切断されているのではないでしょうか?無通信タイムアウトにかぎらず、サーバ側の都合で接続が切れることはあるので、切れていたら再接続するというのがベストプラクティスになります。
File "/var/www/apps/message.py", line 21, in getMessage
のところが修正できるのであれば、以下のようなコードで「切れていたら再接続」が実装できるように思われます。

self.connector.ping(reconnect=True)
   cursor = self.connector.cursor()

もう一つの方法としては、コネクションプールを使います。
おそらく、今はコネクションをプログラムの中の変数に保持しているのだと思いますが、これをやめて、コネクションプールを使うように変更します。

dbconfig = {
  "database": "test",
  "user":     "joe"
}
# プールから取得(このとき、接続が切れていてもプールのロジックで復旧してくれる)
cnx = mysql.connector.connect(pool_name = "mypool",
                              pool_size = 3,
                              **dbconfig)
# cnx を使ってデータベースにアクセス
...
# プールに返却
cnx.close()

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

check解決した方法

0

http://qiita.com/satococoa/items/e3396d9d75b9cf7e6214

こちらを参考に時間を伸ばしたところ一応の解決は見えました。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/09/30 17:08

    そうですか。ただ、根本的な解決になっていないと思います。本来、コネクションをプログラムで長時間保持すること自身良くありません。性能が気にならないなら毎回再接続しても良いくらいです。コネクションを長時間保持すると、途中の通信機器とかで遮断されたり、サーバを再起動するだけで、コネクションが切れたりします。
    私の案にそってプログラムを修正すべきと思いますが、message.py は修正できないのでしょうか?

    キャンセル

  • 2016/10/03 16:32

    コメントありがとうございます。
    即急に接続できる時間を伸ばす必要があったため今回はこちらの処理で対応しました。

    ただ根本的な解決にい至っていないことは理解しているので
    解決すべきものとしてIssueにしました。
    色々な問題で今すぐ手を付けられないのですが近いうちにmit0223さんの方法を参考に解決させます。
    解答ありがとうございました。

    キャンセル

  • 2016/10/03 17:31

    > 根本的な解決にい至っていないことは理解しているので解決すべきものとしてIssueにしました。

    安心しました。老婆心ですみません。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 87.78%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る