Python mysql (използвайки pymysql) автоматично повторно свързване

Не съм сигурен дали това е възможно, но търся начин да се свържа отново с базата данни mysql, когато връзката се загуби. Всички връзки се държат в gevent опашка, но това не би трябвало да има значение според мен. Сигурен съм, че ако отделя малко време, мога да измисля начин да се свържа отново с базата данни. Въпреки това погледнах кода на pymysql и видях, че има метод "ping" в класа на връзката, който не съм сигурен как точно да използвам.

Методът изглежда, че ще се свърже отново за първи път, но след това отново превключи флага за повторно свързване на False? Мога ли да използвам този метод или има друг начин за установяване на връзка, ако тя се загуби? Дори и да не е pymysql, как хората се справят със спирането на сървърите на бази данни и трябва да възстановят връзката с mysql сървъра?

def ping(self, reconnect=True):
    ''' Check if the server is alive '''
    if self.socket is None:
        if reconnect:
            self._connect()
            reconnect = False
        else:
            raise Error("Already closed")
    try:
        self._execute_command(COM_PING, "")
        return self._read_ok_packet()
    except Exception:
        if reconnect:
            self._connect()
            return self.ping(False)
        else:
            raise

person opensourcegeek    schedule 27.03.2014    source източник
comment
Не съм сигурен дали това ще бъде полезно, но погледнете тази рецепта за ReconnectingConnectionPool за Twisted gist.github.com /powdahound/174056   -  person Peter Gibson    schedule 28.03.2014
comment
Приложих го - gist.github.com/opensourcegeek/9822127   -  person opensourcegeek    schedule 28.03.2014
comment
Pinging преди изпълнение на заявка се счита за анти-модел, който губи ресурси и е ненадежден: percona.com/blog/2010/05/05/   -  person Jeff Widman    schedule 19.08.2016
comment
Намерих решение, използвайки метода „ping“ на „връзка“ в документацията на PyMySQL и поставих пример тук   -  person rmmariano    schedule 26.04.2019


Отговори (4)


Най-накрая получих работещо решение, може да помогне на някого.

from gevent import monkey
monkey.patch_socket()
import logging

import gevent
from gevent.queue import Queue
import pymysql as db

logging.basicConfig(level=logging.DEBUG)
LOGGER = logging.getLogger("connection_pool")


class ConnectionPool:
    def __init__(self, db_config, time_to_sleep=30, test_run=False):
        self.username = db_config.get('user')
        self.password = db_config.get('password')
        self.host = db_config.get('host')
        self.port = int(db_config.get('port'))
        self.max_pool_size = 20
        self.test_run = test_run
        self.pool = None
        self.time_to_sleep = time_to_sleep
        self._initialize_pool()

    def get_initialized_connection_pool(self):
        return self.pool

    def _initialize_pool(self):
        self.pool = Queue(maxsize=self.max_pool_size)
        current_pool_size = self.pool.qsize()
        if current_pool_size < self.max_pool_size:  # this is a redundant check, can be removed
            for _ in xrange(0, self.max_pool_size - current_pool_size):
                try:
                    conn = db.connect(host=self.host,
                                      user=self.username,
                                      passwd=self.password,
                                      port=self.port)
                    self.pool.put_nowait(conn)

                except db.OperationalError, e:
                    LOGGER.error("Cannot initialize connection pool - retrying in {} seconds".format(self.time_to_sleep))
                    LOGGER.exception(e)
                    break
        self._check_for_connection_loss()

    def _re_initialize_pool(self):
        gevent.sleep(self.time_to_sleep)
        self._initialize_pool()

    def _check_for_connection_loss(self):
        while True:
            conn = None
            if self.pool.qsize() > 0:
                conn = self.pool.get()

            if not self._ping(conn):
                if self.test_run:
                    self.port = 3306

                self._re_initialize_pool()

            else:
                self.pool.put_nowait(conn)

            if self.test_run:
                break
            gevent.sleep(self.time_to_sleep)

    def _ping(self, conn):
        try:
            if conn is None:
                conn = db.connect(host=self.host,
                                  user=self.username,
                                  passwd=self.password,
                                  port=self.port)
            cursor = conn.cursor()
            cursor.execute('select 1;')
            LOGGER.debug(cursor.fetchall())
            return True

        except db.OperationalError, e:
            LOGGER.warn('Cannot connect to mysql - retrying in {} seconds'.format(self.time_to_sleep))
            LOGGER.exception(e)
            return False

# test (pytest compatible) -------------------------------------------------------------------------------------------
import logging

from src.py.ConnectionPool import ConnectionPool

logging.basicConfig(level=logging.DEBUG)
LOGGER = logging.getLogger("test_connection_pool")


def test_get_initialized_connection_pool():
    config = {
        'user': 'root',
        'password': '',
        'host': '127.0.0.1',
        'port': 3305
    }
    conn_pool = ConnectionPool(config, time_to_sleep=5, test_run=True)
    pool = conn_pool.get_initialized_connection_pool()
    # when in test run the port will be switched back to 3306
    # so the queue size should be 20 - will be nice to work 
    # around this rather than test_run hack
    assert pool.qsize() == 20
person opensourcegeek    schedule 28.03.2014

Е, имам същия проблем в моето приложение и намерих метод в документация на PyMySQL, която пингва към сървъра и проверява дали връзката е затворена или не, ако е затворена, след това се свързва отново.

from pymysql import connect
from pymysql.cursors import DictCursor

# create the connection
connection = connect(host='host', port='port', user='user', 
                     password='password', db='db', 
                     cursorclass=DictCursor)

# get the cursor
cursor = connection.cursor()

# if the connection was lost, then it reconnects
connection.ping(reconnect=True)      

# execute the query
cursor.execute(query)

Надявам се да помогне.

person rmmariano    schedule 26.04.2019

Най-лесният начин е да проверите връзката точно преди да изпратите запитване.

Можете да направите това, като създадете малък клас, който съдържа два метода: connect и query:

import pymysql
import pymysql.cursors

class DB:
    def connect(self):
        self.conn = pymysql.connect(
                             host=hostname,
                             user=username,
                             password=password,
                             db=dbname,
                             charset='utf8mb4',
                             cursorclass=pymysql.cursors.DictCursor,
                             port=3306)

    def query(self, sql):
        try:
            cursor = self.conn.cursor()
            cursor.execute(sql)
        except pymysql.OperationalError:
            self.connect()
            cursor = self.conn.cursor()
            cursor.execute(sql)
        return cursor

db = DB()

Сега, когато изпратите заявка, използвайки db.query("example SQL"), заявката автоматично се подготвя да срещне грешка при свързване и се свързва отново, използвайки self.connect(), ако е необходимо.

Запомнете: Това е опростен пример. Обикновено бихте искали да позволите на PyMySQL да ви помогне да избегнете специални знаци във вашите заявки. За да направите това, ще трябва да добавите 2-ри параметър в метода query и да продължите оттам.

person Seth Connell    schedule 16.08.2018
comment
Никога не трябва да използвате освен: без дори да се ограничавате до Изключение. В този случай трябва да е поне pymssql.StandardError. Обикновено не искате да улавяте SystemExit изключение sys.exit(), например, за да предотвратите затварянето на процеса, защото продължавайки от него, вашият процес ще остане в много нестабилно състояние. - person Marko Kohtala; 18.12.2018

логиката е доста проста, ако връзката се затвори, опитайте да се свържете отново няколко пъти, в този случай използвам максимални опити за 15 пъти за повторно свързване или пинг.

import pymysql, pymysql.cursors
conn = pymysql.connect(
                         host=hostname,
                         user=username,
                         password=password,
                         db=dbname,
                         charset='utf8mb4',
                         cursorclass=pymysql.cursors.DictCursor,
                         )
cursor = conn.cursor()
# you can do transactions to database and when you need conn later, just make sure the server is still connected
if conn.open is False:
   max_try = 15
   try = 0
   while conn.open is False:
       if try < max_try:
           conn.ping() # autoreconnect is true by default
       try +=1

# check the conn again to make sure it connected
if conn.open:
    # statements when conn is successfully reconnect to the server
else:
    # it must be something wrong : server, network etc
person broid    schedule 07.05.2019
comment
че conn.open не е в PEP 249 и изглежда, че проверява дали има обект на сокет (в ssl контекст). Тази проверка може да не гарантира, че когато отдалеченият сървър изчезне, conn.open ще върне false, ако връзката няма да работи, когато основният сокет избира/пропитва. - person Rich Andrews; 02.02.2021