|
|
################################################################################### # # Copyright (C) 2017 MuK IT GmbH # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as # published by the Free Software Foundation, either version 3 of the # License, or (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Affero General Public License for more details. # # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. # ###################################################################################
import json import logging import psycopg2 import functools
from contextlib import closing from contextlib import contextmanager from datetime import datetime, date
from werkzeug.contrib.sessions import SessionStore
from odoo.sql_db import db_connect from odoo.tools import config
_logger = logging.getLogger(__name__)
def retry_database(func): @functools.wraps(func) def wrapper(self, *args, **kwargs): for attempts in range(1, 6): try: return func(self, *args, **kwargs) except psycopg2.InterfaceError as error: _logger.warn("SessionStore connection failed! (%s/5)" % attempts) if attempts >= 5: raise error return wrapper
class PostgresSessionStore(SessionStore): def __init__(self, *args, **kwargs): super(PostgresSessionStore, self).__init__(*args, **kwargs) self.dbname = config.get('session_store_dbname', 'session_store') self._setup_database(raise_exception=False) def _setup_database(self, raise_exception=True): try: with db_connect(self.dbname, allow_uri=True).cursor() as cursor: cursor.autocommit(True) self._create_table(cursor) except: self._create_database() self._setup_database()
def _create_database(self): with db_connect("postgres").cursor() as cursor: cursor.autocommit(True) cursor.execute("""
CREATE DATABASE {dbname} ENCODING 'unicode' TEMPLATE 'template0'; """.format(dbname=self.dbname))
def _create_table(self, cursor): self.cursor.execute("""
CREATE TABLE IF NOT EXISTS sessions ( sid varchar PRIMARY KEY, write_date timestamp without time zone NOT NULL, payload text NOT NULL ); """)
@contextmanager def open_cursor(self): connection = db_connect(self.dbname, allow_uri=True) cursor = connection.cursor() cursor.autocommit(True) yield cursor cursor.close() @retry_database def save(self, session): with open_cursor() as cursor: cursor.execute("""
INSERT INTO sessions (sid, write_date, payload) VALUES (%(sid)s, now() at time zone 'UTC', %(payload)s) ON CONFLICT (sid) DO UPDATE SET payload = %(payload)s, write_date = now() at time zone 'UTC'; """, dict(sid=session.sid, payload=json.dumps(dict(session))))
@retry_database def delete(self, session): with open_cursor() as cursor: cursor.execute("DELETE FROM sessions WHERE sid=%s;", [session.sid])
@retry_database def get(self, sid): if not self.is_valid_key(sid): return self.new() with open_cursor() as cursor: cursor.execute("""
SELECT payload, write_date FROM sessions WHERE sid=%s; """, [sid])
try: payload, write_date = cursor.fetchone() if write_date.date() != datetime.today().date(): cursor.execute("""
UPDATE sessions SET write_date = now() at time zone 'UTC' WHERE sid=%s; """, [sid])
return self.session_class(json.loads(payload), sid, False) except Exception: return self.session_class({}, sid, False) @retry_database def list(self): with open_cursor() as cursor: cursor.execute("SELECT sid FROM sessions;") return [record[0] for record in cursor.fetchall()] @retry_database def clean(self): with open_cursor() as cursor: cursor.execute("""
DELETE FROM sessions WHERE now() at time zone 'UTC' - write_date > '7 days'; """)
|