################################################################################### # # 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 . # ################################################################################### 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'; """)