Merge pull request #316 from szopu/django-credentials-field-py3-fix
Issue #142: CredentialField Python 3 fix
This commit is contained in:
@@ -21,17 +21,17 @@ the Django datastore.
|
|||||||
import oauth2client
|
import oauth2client
|
||||||
import base64
|
import base64
|
||||||
import pickle
|
import pickle
|
||||||
|
import six
|
||||||
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
|
from django.utils.encoding import smart_bytes, smart_text
|
||||||
from oauth2client.client import Storage as BaseStorage
|
from oauth2client.client import Storage as BaseStorage
|
||||||
|
|
||||||
|
|
||||||
__author__ = 'jcgregorio@google.com (Joe Gregorio)'
|
__author__ = 'jcgregorio@google.com (Joe Gregorio)'
|
||||||
|
|
||||||
|
|
||||||
class CredentialsField(models.Field):
|
class CredentialsField(six.with_metaclass(models.SubfieldBase, models.Field)):
|
||||||
|
|
||||||
__metaclass__ = models.SubfieldBase
|
|
||||||
|
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
if 'null' not in kwargs:
|
if 'null' not in kwargs:
|
||||||
@@ -46,12 +46,24 @@ class CredentialsField(models.Field):
|
|||||||
return None
|
return None
|
||||||
if isinstance(value, oauth2client.client.Credentials):
|
if isinstance(value, oauth2client.client.Credentials):
|
||||||
return value
|
return value
|
||||||
return pickle.loads(base64.b64decode(value))
|
return pickle.loads(base64.b64decode(smart_bytes(value)))
|
||||||
|
|
||||||
def get_db_prep_value(self, value, connection, prepared=False):
|
def get_prep_value(self, value):
|
||||||
if value is None:
|
if value is None:
|
||||||
return None
|
return None
|
||||||
return base64.b64encode(pickle.dumps(value))
|
return smart_text(base64.b64encode(pickle.dumps(value)))
|
||||||
|
|
||||||
|
def value_to_string(self, obj):
|
||||||
|
"""Convert the field value from the provided model to a string.
|
||||||
|
Used during model serialization.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
obj: db.Model, model object
|
||||||
|
Returns:
|
||||||
|
string, the serialized field value
|
||||||
|
"""
|
||||||
|
value = self._get_val_from_obj(obj)
|
||||||
|
return self.get_prep_value(value)
|
||||||
|
|
||||||
|
|
||||||
class FlowField(models.Field):
|
class FlowField(models.Field):
|
||||||
|
|||||||
@@ -36,10 +36,11 @@ os.environ['DJANGO_SETTINGS_MODULE'] = 'django_settings'
|
|||||||
sys.modules['django_settings'] = django_settings = imp.new_module(
|
sys.modules['django_settings'] = django_settings = imp.new_module(
|
||||||
'django_settings')
|
'django_settings')
|
||||||
django_settings.SECRET_KEY = 'xyzzy'
|
django_settings.SECRET_KEY = 'xyzzy'
|
||||||
|
from django.db import models
|
||||||
|
|
||||||
from oauth2client.django_orm import CredentialsField
|
from oauth2client.django_orm import CredentialsField
|
||||||
from oauth2client.django_orm import FlowField
|
from oauth2client.django_orm import FlowField
|
||||||
|
from oauth2client._helpers import _from_bytes, _to_bytes
|
||||||
|
|
||||||
__author__ = 'conleyo@google.com (Conley Owens)'
|
__author__ = 'conleyo@google.com (Conley Owens)'
|
||||||
|
|
||||||
@@ -58,10 +59,39 @@ class TestCredentialsField(unittest.TestCase):
|
|||||||
self.assertTrue(isinstance(self.field.to_python(self.pickle),
|
self.assertTrue(isinstance(self.field.to_python(self.pickle),
|
||||||
Credentials))
|
Credentials))
|
||||||
|
|
||||||
|
def test_field_unpickled_none(self):
|
||||||
|
self.assertEqual(self.field.to_python(None), None)
|
||||||
|
|
||||||
def test_field_pickled(self):
|
def test_field_pickled(self):
|
||||||
prep_value = self.field.get_db_prep_value(self.credentials,
|
prep_value = self.field.get_db_prep_value(self.credentials,
|
||||||
connection=None)
|
connection=None)
|
||||||
self.assertEqual(prep_value, self.pickle)
|
self.assertEqual(_to_bytes(prep_value), self.pickle)
|
||||||
|
|
||||||
|
|
||||||
|
class TestCredentialsFieldViaModel(unittest.TestCase):
|
||||||
|
|
||||||
|
class TestModel(models.Model):
|
||||||
|
credentials = CredentialsField()
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.model = self.TestModel()
|
||||||
|
# using the meta api:
|
||||||
|
# https://docs.djangoproject.com/en/1.8/ref/models/meta/#field-access-api
|
||||||
|
self.field = self.model._meta.get_field('credentials')
|
||||||
|
self.credentials = Credentials()
|
||||||
|
self.pickle_str = _from_bytes(base64.b64encode(pickle.dumps(
|
||||||
|
self.credentials
|
||||||
|
)))
|
||||||
|
|
||||||
|
def test_field_value_to_string(self):
|
||||||
|
self.model.credentials = self.credentials
|
||||||
|
value_str = self.field.value_to_string(self.model)
|
||||||
|
self.assertEqual(value_str, self.pickle_str)
|
||||||
|
|
||||||
|
def test_field_value_to_string_none(self):
|
||||||
|
self.model.credentials = None
|
||||||
|
value_str = self.field.value_to_string(self.model)
|
||||||
|
self.assertEqual(value_str, None)
|
||||||
|
|
||||||
|
|
||||||
class TestFlowField(unittest.TestCase):
|
class TestFlowField(unittest.TestCase):
|
||||||
|
|||||||
Reference in New Issue
Block a user