session.py 1.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061
  1. # -*- coding: utf-8 -*-
  2. """SQLAlchemy session."""
  3. from __future__ import absolute_import, unicode_literals
  4. from sqlalchemy import create_engine
  5. from sqlalchemy.ext.declarative import declarative_base
  6. from sqlalchemy.orm import sessionmaker
  7. from sqlalchemy.pool import NullPool
  8. from kombu.utils.compat import register_after_fork
  9. ResultModelBase = declarative_base()
  10. __all__ = ['SessionManager']
  11. def _after_fork_cleanup_session(session):
  12. session._after_fork()
  13. class SessionManager(object):
  14. def __init__(self):
  15. self._engines = {}
  16. self._sessions = {}
  17. self.forked = False
  18. self.prepared = False
  19. if register_after_fork is not None:
  20. register_after_fork(self, _after_fork_cleanup_session)
  21. def _after_fork(self):
  22. self.forked = True
  23. def get_engine(self, dburi, **kwargs):
  24. if self.forked:
  25. try:
  26. return self._engines[dburi]
  27. except KeyError:
  28. engine = self._engines[dburi] = create_engine(dburi, **kwargs)
  29. return engine
  30. else:
  31. return create_engine(dburi, poolclass=NullPool)
  32. def create_session(self, dburi, short_lived_sessions=False, **kwargs):
  33. engine = self.get_engine(dburi, **kwargs)
  34. if self.forked:
  35. if short_lived_sessions or dburi not in self._sessions:
  36. self._sessions[dburi] = sessionmaker(bind=engine)
  37. return engine, self._sessions[dburi]
  38. else:
  39. return engine, sessionmaker(bind=engine)
  40. def prepare_models(self, engine):
  41. if not self.prepared:
  42. ResultModelBase.metadata.create_all(engine)
  43. self.prepared = True
  44. def session_factory(self, dburi, **kwargs):
  45. engine, session = self.create_session(dburi, **kwargs)
  46. self.prepare_models(engine)
  47. return session()