5c5491f9de
Changes: * using oslo_config instead of oslo.config * using oslo_concurrency instead of oslo.concurrency * using oslo_db instead of oslo.db * using oslo_i18n instead of oslo.i18n * using oslo_messaging instead of oslo.messaging * using oslo_middleware instead of oslo.middleware * using oslo_serialization instead of oslo.serialization * using oslo_utils instead of oslo.utils Change-Id: Ib0f18603ca5b0885256a39a96a3620d05260a272 Closes-bug: #1414587
110 lines
3.4 KiB
Python
110 lines
3.4 KiB
Python
# Copyright (c) 2013 Mirantis Inc.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
# implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
from oslo_serialization import jsonutils
|
|
from sqlalchemy.ext import mutable
|
|
from sqlalchemy import types as st
|
|
|
|
|
|
class JSONEncoded(st.TypeDecorator):
|
|
"""Represents an immutable structure as a json-encoded string."""
|
|
|
|
impl = st.TEXT
|
|
|
|
def process_bind_param(self, value, dialect):
|
|
if value is not None:
|
|
value = jsonutils.dumps(value)
|
|
return value
|
|
|
|
def process_result_value(self, value, dialect):
|
|
if value is not None:
|
|
value = jsonutils.loads(value)
|
|
return value
|
|
|
|
|
|
# TODO(slukjanov): verify this implementation
|
|
class MutableDict(mutable.Mutable, dict):
|
|
@classmethod
|
|
def coerce(cls, key, value):
|
|
"""Convert plain dictionaries to MutableDict."""
|
|
if not isinstance(value, MutableDict):
|
|
if isinstance(value, dict):
|
|
return MutableDict(value)
|
|
|
|
# this call will raise ValueError
|
|
return mutable.Mutable.coerce(key, value)
|
|
else:
|
|
return value
|
|
|
|
def update(self, e=None, **f):
|
|
"""Detect dictionary update events and emit change events."""
|
|
dict.update(self, e, **f)
|
|
self.changed()
|
|
|
|
def __setitem__(self, key, value):
|
|
"""Detect dictionary set events and emit change events."""
|
|
dict.__setitem__(self, key, value)
|
|
self.changed()
|
|
|
|
def __delitem__(self, key):
|
|
"""Detect dictionary del events and emit change events."""
|
|
dict.__delitem__(self, key)
|
|
self.changed()
|
|
|
|
|
|
# TODO(slukjanov): verify this implementation
|
|
class MutableList(mutable.Mutable, list):
|
|
@classmethod
|
|
def coerce(cls, key, value):
|
|
"""Convert plain lists to MutableList."""
|
|
if not isinstance(value, MutableList):
|
|
if isinstance(value, list):
|
|
return MutableList(value)
|
|
|
|
# this call will raise ValueError
|
|
return mutable.Mutable.coerce(key, value)
|
|
else:
|
|
return value
|
|
|
|
def __add__(self, value):
|
|
"""Detect list add events and emit change events."""
|
|
list.__add__(self, value)
|
|
self.changed()
|
|
|
|
def append(self, value):
|
|
"""Detect list add events and emit change events."""
|
|
list.append(self, value)
|
|
self.changed()
|
|
|
|
def __setitem__(self, key, value):
|
|
"""Detect list set events and emit change events."""
|
|
list.__setitem__(self, key, value)
|
|
self.changed()
|
|
|
|
def __delitem__(self, i):
|
|
"""Detect list del events and emit change events."""
|
|
list.__delitem__(self, i)
|
|
self.changed()
|
|
|
|
|
|
def JsonDictType():
|
|
"""Returns an SQLAlchemy Column Type suitable to store a Json dict."""
|
|
return MutableDict.as_mutable(JSONEncoded)
|
|
|
|
|
|
def JsonListType():
|
|
"""Returns an SQLAlchemy Column Type suitable to store a Json array."""
|
|
return MutableList.as_mutable(JSONEncoded)
|