[tryton-debian-vcs] tryton-proteus branch upstream updated. upstream/3.2.1-1-gd80368c
Mathias Behrle
tryton-debian-vcs at alioth.debian.org
Thu Oct 23 12:19:35 UTC 2014
The following commit has been merged in the upstream branch:
https://alioth.debian.org/plugins/scmgit/cgi-bin/gitweb.cgi/?p=tryton/tryton-proteus.git;a=commitdiff;h=upstream/3.2.1-1-gd80368c
commit d80368c7e784756065f83f4732c88d88ea36e0b2
Author: Mathias Behrle <mathiasb at m9s.biz>
Date: Tue Oct 21 11:29:25 2014 +0200
Adding upstream version 3.4.0.
Signed-off-by: Mathias Behrle <mathiasb at m9s.biz>
diff --git a/CHANGELOG b/CHANGELOG
index 599fe1b..f46bf35 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,5 +1,9 @@
-Version 3.2.1 - 2014-07-02
+Version 3.4.0 - 2014-10-20
* Bug fixes (see mercurial logs for details)
+* New configuration syntax
+* Add Report
+* Explicitly set value of parent field
+* Add duplicate method
Version 3.2.0 - 2014-04-21
* Bug fixes (see mercurial logs for details)
diff --git a/PKG-INFO b/PKG-INFO
index d00840d..015002e 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
Metadata-Version: 1.1
Name: proteus
-Version: 3.2.1
+Version: 3.4.0
Summary: Library to access Tryton server as a client
Home-page: http://www.tryton.org/
Author: Tryton
Author-email: issue_tracker at tryton.org
License: LGPL-3
-Download-URL: http://downloads.tryton.org/3.2/
+Download-URL: http://downloads.tryton.org/3.4/
Description: proteus
=======
@@ -20,18 +20,14 @@ Description: proteus
Example of usage
----------------
- >>> from proteus import config, Model, Wizard
+ >>> from proteus import config, Model, Wizard, Report
- Creating a database
- ~~~~~~~~~~~~~~~~~~~
+ Configuration
+ ~~~~~~~~~~~~~
Configuration to connect to a sqlite memory database using trytond as module.
- >>> config = config.set_trytond(':memory:', database_type='sqlite')
-
- When connecting to a database that doesn't exist, Proteus will create it.
- If no database name was given, then Proteus will generate one. It will choose
- ':memory': for 'sqlite' type otherwise `'test_%' % int(time.time())`.
+ >>> config = config.set_trytond('sqlite:///:memory:')
Installing a module
~~~~~~~~~~~~~~~~~~~
@@ -108,6 +104,17 @@ Description: proteus
>>> party.categories #doctest: +ELLIPSIS
[proteus.Model.get('party.category')(...)]
+ Print party label
+ ~~~~~~~~~~~~~~~~~
+
+ There is a label report on `Party`.
+
+ >>> label = Report('party.label')
+
+ The report is executed with a list of records and some extra data.
+
+ >>> type_, data, print_, name = label.execute([party], {})
+
Support
-------
diff --git a/README b/README
index 0b23a11..53fd3ed 100644
--- a/README
+++ b/README
@@ -11,18 +11,14 @@ See INSTALL
Example of usage
----------------
- >>> from proteus import config, Model, Wizard
+ >>> from proteus import config, Model, Wizard, Report
-Creating a database
-~~~~~~~~~~~~~~~~~~~
+Configuration
+~~~~~~~~~~~~~
Configuration to connect to a sqlite memory database using trytond as module.
- >>> config = config.set_trytond(':memory:', database_type='sqlite')
-
-When connecting to a database that doesn't exist, Proteus will create it.
-If no database name was given, then Proteus will generate one. It will choose
-':memory': for 'sqlite' type otherwise `'test_%' % int(time.time())`.
+ >>> config = config.set_trytond('sqlite:///:memory:')
Installing a module
~~~~~~~~~~~~~~~~~~~
@@ -99,6 +95,17 @@ Append it to categories of the party
>>> party.categories #doctest: +ELLIPSIS
[proteus.Model.get('party.category')(...)]
+Print party label
+~~~~~~~~~~~~~~~~~
+
+There is a label report on `Party`.
+
+ >>> label = Report('party.label')
+
+The report is executed with a list of records and some extra data.
+
+ >>> type_, data, print_, name = label.execute([party], {})
+
Support
-------
diff --git a/proteus.egg-info/PKG-INFO b/proteus.egg-info/PKG-INFO
index d00840d..015002e 100644
--- a/proteus.egg-info/PKG-INFO
+++ b/proteus.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
Metadata-Version: 1.1
Name: proteus
-Version: 3.2.1
+Version: 3.4.0
Summary: Library to access Tryton server as a client
Home-page: http://www.tryton.org/
Author: Tryton
Author-email: issue_tracker at tryton.org
License: LGPL-3
-Download-URL: http://downloads.tryton.org/3.2/
+Download-URL: http://downloads.tryton.org/3.4/
Description: proteus
=======
@@ -20,18 +20,14 @@ Description: proteus
Example of usage
----------------
- >>> from proteus import config, Model, Wizard
+ >>> from proteus import config, Model, Wizard, Report
- Creating a database
- ~~~~~~~~~~~~~~~~~~~
+ Configuration
+ ~~~~~~~~~~~~~
Configuration to connect to a sqlite memory database using trytond as module.
- >>> config = config.set_trytond(':memory:', database_type='sqlite')
-
- When connecting to a database that doesn't exist, Proteus will create it.
- If no database name was given, then Proteus will generate one. It will choose
- ':memory': for 'sqlite' type otherwise `'test_%' % int(time.time())`.
+ >>> config = config.set_trytond('sqlite:///:memory:')
Installing a module
~~~~~~~~~~~~~~~~~~~
@@ -108,6 +104,17 @@ Description: proteus
>>> party.categories #doctest: +ELLIPSIS
[proteus.Model.get('party.category')(...)]
+ Print party label
+ ~~~~~~~~~~~~~~~~~
+
+ There is a label report on `Party`.
+
+ >>> label = Report('party.label')
+
+ The report is executed with a list of records and some extra data.
+
+ >>> type_, data, print_, name = label.execute([party], {})
+
Support
-------
diff --git a/proteus.egg-info/SOURCES.txt b/proteus.egg-info/SOURCES.txt
index 1eef609..b19dc88 100644
--- a/proteus.egg-info/SOURCES.txt
+++ b/proteus.egg-info/SOURCES.txt
@@ -15,7 +15,9 @@ proteus.egg-info/requires.txt
proteus.egg-info/top_level.txt
proteus.egg-info/zip-safe
proteus/tests/__init__.py
+proteus/tests/common.py
proteus/tests/test_config.py
proteus/tests/test_context.py
proteus/tests/test_model.py
+proteus/tests/test_report.py
proteus/tests/test_wizard.py
\ No newline at end of file
diff --git a/proteus.egg-info/requires.txt b/proteus.egg-info/requires.txt
index e8c47c8..232d2df 100644
--- a/proteus.egg-info/requires.txt
+++ b/proteus.egg-info/requires.txt
@@ -4,7 +4,7 @@ python-dateutil
cdecimal
[trytond]
-trytond >= 3.2, < 3.3
+trytond >= 3.4, < 3.5
[simplejson]
simplejson
\ No newline at end of file
diff --git a/proteus/__init__.py b/proteus/__init__.py
index 6c8d8ab..2f2b0f8 100644
--- a/proteus/__init__.py
+++ b/proteus/__init__.py
@@ -3,8 +3,8 @@
'''
A library to access Tryton's models like a client.
'''
-__version__ = "3.2.1"
-__all__ = ['Model', 'Wizard']
+__version__ = "3.4.0"
+__all__ = ['Model', 'Wizard', 'Report']
import sys
try:
import cdecimal
@@ -26,23 +26,32 @@ _MODELS = threading.local()
class _EvalEnvironment(dict):
'Dictionary for evaluation'
- def __init__(self, parent):
+ def __init__(self, parent, eval_type='eval'):
super(_EvalEnvironment, self).__init__()
self.parent = parent
+ assert eval_type in ('eval', 'on_change')
+ self.eval_type = eval_type
def __getitem__(self, item):
if item == '_parent_' + self.parent._parent_name \
- and self.parent.parent:
- return _EvalEnvironment(self.parent.parent)
- return self.parent._get_eval()[item]
+ and self.parent._parent:
+ return _EvalEnvironment(self.parent._parent,
+ eval_type=self.eval_type)
+ if self.eval_type == 'eval':
+ return self.parent._get_eval()[item]
+ else:
+ return self.parent._get_on_change_values(fields=[item])[item]
def __getattr__(self, item):
- return self.__getitem__(item)
+ try:
+ return self.__getitem__(item)
+ except KeyError:
+ raise AttributeError(item)
def get(self, item, default=None):
try:
- return self.__getattr__(item)
- except:
+ return self.__getitem__(item)
+ except KeyError:
pass
return super(_EvalEnvironment, self).get(item, default)
@@ -55,7 +64,13 @@ class _EvalEnvironment(dict):
__repr__ = __str__
def __contains__(self, item):
- return item in self.parent._fields
+ if item == '_parent_' + self.parent._parent_name \
+ and self.parent._parent:
+ return True
+ if self.eval_type == 'eval':
+ return item in self.parent._get_eval()
+ else:
+ return item in self.parent._fields
class FieldDescriptor(object):
@@ -97,23 +112,16 @@ class CharDescriptor(FieldDescriptor):
default = None
def __set__(self, instance, value):
- assert isinstance(value, basestring) or value in (None, False)
+ assert isinstance(value, basestring) or value is None
super(CharDescriptor, self).__set__(instance, value or '')
- def __get__(self, instance, owner):
- value = super(CharDescriptor, self).__get__(instance, owner)
- if value is False:
- value = None
- return value
-
class BinaryDescriptor(FieldDescriptor):
default = None
def __set__(self, instance, value):
- assert (isinstance(value, (basestring, buffer))
- or value in (None, False))
- super(BinaryDescriptor, self).__set__(instance, value or '')
+ assert isinstance(value, (basestring, buffer)) or value is None
+ super(BinaryDescriptor, self).__set__(instance, value)
class IntegerDescriptor(FieldDescriptor):
@@ -143,8 +151,6 @@ class NumericDescriptor(FieldDescriptor):
class ReferenceDescriptor(FieldDescriptor):
def __get__(self, instance, owner):
value = super(ReferenceDescriptor, self).__get__(instance, owner)
- if instance._parent_name == self.name:
- value = instance._parent
if isinstance(value, basestring):
model_name, id = value.split(',', 1)
if model_name:
@@ -158,7 +164,6 @@ class ReferenceDescriptor(FieldDescriptor):
if isinstance(value, basestring):
assert value.startswith(',')
elif isinstance(value, Model):
- assert value.id > 0 and not value._changed
assert value._config == instance._config
super(ReferenceDescriptor, self).__set__(instance, value)
@@ -172,25 +177,25 @@ class DateDescriptor(FieldDescriptor):
return value
def __set__(self, instance, value):
- assert isinstance(value, datetime.date) or value in (None, False)
+ assert isinstance(value, datetime.date) or value is None
super(DateDescriptor, self).__set__(instance, value)
class DateTimeDescriptor(FieldDescriptor):
def __set__(self, instance, value):
- assert isinstance(value, datetime.datetime) or value in (None, False)
+ assert isinstance(value, datetime.datetime) or value is None
super(DateTimeDescriptor, self).__set__(instance, value)
class TimeDescriptor(FieldDescriptor):
def __set__(self, instance, value):
- assert isinstance(value, datetime.time) or value in (None, False)
+ assert isinstance(value, datetime.time) or value is None
super(TimeDescriptor, self).__set__(instance, value)
class DictDescriptor(FieldDescriptor):
def __set__(self, instance, value):
- assert isinstance(value, dict) or value in (None, False)
+ assert isinstance(value, dict) or value is None
super(DictDescriptor, self).__set__(instance, value)
@@ -198,12 +203,8 @@ class Many2OneDescriptor(FieldDescriptor):
def __get__(self, instance, owner):
relation = Model.get(self.definition['relation'], instance._config)
value = super(Many2OneDescriptor, self).__get__(instance, owner)
- if instance._parent_name == self.name:
- value = instance._parent
- if isinstance(value, (int, long)) and value is not False:
+ if isinstance(value, (int, long)):
value = relation(value)
- elif not value:
- value = None
if self.name in instance._values:
instance._values[self.name] = value
return value
@@ -211,7 +212,6 @@ class Many2OneDescriptor(FieldDescriptor):
def __set__(self, instance, value):
assert isinstance(value, (Model, NoneType))
if value:
- assert value.id > 0 and not value._changed
assert value._config == instance._config
super(Many2OneDescriptor, self).__set__(instance, value)
@@ -342,7 +342,14 @@ class One2OneEvalDescriptor(Many2OneEvalDescriptor):
class One2ManyEvalDescriptor(EvalDescriptor):
def __get__(self, instance, owner):
- return [x.id for x in getattr(instance, self.name)]
+ # Directly use _values to prevent infinite recursion with
+ # One2ManyDescriptor which could evaluate this field to decode the
+ # context
+ value = instance._values.get(self.name, [])
+ if isinstance(value, ModelList):
+ return [x.id for x in value]
+ else:
+ return value
class Many2ManyEvalDescriptor(One2ManyEvalDescriptor):
@@ -411,7 +418,8 @@ class MetaModelFactory(object):
for field_name, definition in dict['_fields'].iteritems():
if field_name == 'id':
continue
- Descriptor = self.descriptors[definition['type']]
+ Descriptor = self.descriptors.get(definition['type'],
+ FieldDescriptor)
dict[field_name] = Descriptor(field_name, definition)
VDescriptor = self.value_descriptors.get(
definition['type'], ValueDescriptor)
@@ -469,27 +477,9 @@ class ModelList(list):
ctx.update(decoder.decode(self.context) if self.context else {})
return ctx
- def append(self, record):
- assert isinstance(record, Model)
- if self.parent:
- assert record._config == self.parent._config
- elif self:
- assert record._config == self[0]._config
- assert record._parent is None
- assert not record._parent_field_name
- assert not record._parent_name
- record._parent = self.parent
- record._parent_field_name = self.parent_field_name
- record._parent_name = self.parent_name
- res = super(ModelList, self).append(record)
- self._changed()
- return res
- append.__doc__ = list.append.__doc__
-
- def extend(self, iterable):
- iterable = list(iterable)
+ def __check(self, records):
config = None
- for record in iterable:
+ for record in records:
assert isinstance(record, Model)
if self.parent:
assert record._config == self.parent._config
@@ -499,13 +489,30 @@ class ModelList(list):
assert record._config == config
else:
config = record._config
- for record in iterable:
+ for record in records:
assert record._parent is None
assert not record._parent_field_name
assert not record._parent_name
record._parent = self.parent
record._parent_field_name = self.parent_field_name
record._parent_name = self.parent_name
+
+ # Set parent field to trigger on_change
+ if self.parent and self.parent_name in record._fields:
+ definition = record._fields[self.parent_name]
+ if definition['type'] in ('many2one', 'reference'):
+ setattr(record, self.parent_name, self.parent)
+
+ def append(self, record):
+ self.__check([record])
+ res = super(ModelList, self).append(record)
+ self._changed()
+ return res
+ append.__doc__ = list.append.__doc__
+
+ def extend(self, iterable):
+ iterable = list(iterable)
+ self.__check(iterable)
res = super(ModelList, self).extend(iterable)
self._changed()
return res
@@ -599,11 +606,9 @@ class Model(object):
getattr(self, field_name).extend(value)
else:
if definition['type'] == 'many2one':
- if (isinstance(value, (int, long)) and value is not False):
+ if isinstance(value, (int, long)):
relation = Model.get(definition['relation'])
value = relation(value)
- elif value is False:
- value = None
setattr(self, field_name, value)
__init__.__doc__ = object.__init__.__doc__
@@ -692,6 +697,13 @@ class Model(object):
self.reload()
return True
+ @classmethod
+ def duplicate(cls, records, default=None):
+ 'Duplicate the record'
+ ids = cls._proxy.copy([r.id for r in records], default,
+ cls._config.context)
+ return [cls(id) for id in ids]
+
def click(self, button):
'Click on button'
self.save()
@@ -749,6 +761,7 @@ class Model(object):
self._config.context))
def _default_set(self, values):
+ fieldnames = []
for field, value in values.iteritems():
if '.' in field:
continue
@@ -767,6 +780,9 @@ class Model(object):
field)
else:
self._values[field] = value
+ fieldnames.append(field)
+ for field in sorted(fieldnames):
+ self._on_change(field)
def _get_eval(self):
values = dict((x, getattr(self, '__%s_eval' % x))
@@ -774,23 +790,39 @@ class Model(object):
values['id'] = self.id
return values
- def _get_on_change_value(self):
+ def _get_on_change_values(self, skip=None, fields=None):
values = {'id': self.id}
- for field, definition in self._fields.iteritems():
- if field in self._values and field != 'id':
- if definition['type'] == 'one2many':
- values[field] = [x._get_on_change_value()
- for x in getattr(self, field)]
- else:
- values[field] = getattr(self, '__%s_eval' % field)
+ if fields:
+ definitions = ((f, self._fields[f]) for f in fields)
+ else:
+ definitions = self._fields.iteritems()
+ for field, definition in definitions:
+ if not fields:
+ if field == 'id' or (skip and field in skip):
+ continue
+ if (self.id >= 0
+ and (field not in self._values
+ or field not in self._changed)):
+ continue
+ if definition['type'] == 'one2many':
+ values[field] = [x._get_on_change_values(
+ skip={definition.get('relation_field', '')})
+ for x in getattr(self, field)]
+ elif (definition['type'] in ('many2one', 'reference')
+ and self._parent_name == definition['name']
+ and self._parent):
+ values[field] = self._parent._get_on_change_values(
+ skip={self._parent_field_name})
+ if definition['type'] == 'reference':
+ values[field] = (
+ self._parent.__class__.__name__, values[field])
+ else:
+ values[field] = getattr(self, '__%s_eval' % field)
return values
def _on_change_args(self, args):
res = {}
- values = self._get_on_change_value()
- if self._parent:
- values['_parent_%s' % self._parent_name] = \
- _EvalEnvironment(self._parent)
+ values = _EvalEnvironment(self, 'on_change')
for arg in args:
scope = values
for i in arg.split('.'):
@@ -942,7 +974,7 @@ class Wizard(object):
# Filter only modified values
data = {self.form_state:
dict((k, v) for k, v in
- self.form._get_on_change_value().iteritems()
+ self.form._get_on_change_values().iteritems()
if k in self.form._values)}
else:
data = {}
@@ -965,3 +997,17 @@ class Wizard(object):
if self.state == self.end_state:
self._proxy.delete(self.session_id, self._config.context)
+
+
+class Report(object):
+ 'Report class for Tryton reports'
+
+ def __init__(self, name, config=None, context=None):
+ super(Report, self).__init__()
+ self.name = name
+ self._config = config or proteus.config.get_config()
+ self._context = context or {}
+ self._proxy = self._config.get_proxy(name, type='report')
+
+ def execute(self, models, data):
+ return self._proxy.execute([m.id for m in models], data, self._context)
diff --git a/proteus/config.py b/proteus/config.py
index ebef61d..c01e451 100644
--- a/proteus/config.py
+++ b/proteus/config.py
@@ -9,7 +9,8 @@ import xmlrpclib
import threading
from decimal import Decimal
import datetime
-import time
+import os
+import urlparse
def dump_decimal(self, value, write):
@@ -50,6 +51,28 @@ xmlrpclib.Marshaller.dispatch[datetime.time] = dump_time
xmlrpclib.Marshaller.dispatch[buffer] = dump_buffer
+class XMLRPCDecoder(object):
+
+ decoders = {}
+
+ @classmethod
+ def register(cls, klass, decoder):
+ assert klass not in cls.decoders
+ cls.decoders[klass] = decoder
+
+ def __call__(self, dct):
+ if dct.get('__class__') in self.decoders:
+ return self.decoders[dct['__class__']](dct)
+ return dct
+
+XMLRPCDecoder.register('date',
+ lambda dct: datetime.date(dct['year'], dct['month'], dct['day']))
+XMLRPCDecoder.register('time',
+ lambda dct: datetime.time(dct['hour'], dct['minute'], dct['second'],
+ dct['microsecond']))
+XMLRPCDecoder.register('Decimal', lambda dct: Decimal(dct['decimal']))
+
+
def end_struct(self, data):
mark = self._marks.pop()
# map structs to Python dictionaries
@@ -57,14 +80,7 @@ def end_struct(self, data):
items = self._stack[mark:]
for i in range(0, len(items), 2):
dct[xmlrpclib._stringify(items[i])] = items[i + 1]
- if '__class__' in dct:
- if dct['__class__'] == 'date':
- dct = datetime.date(dct['year'], dct['month'], dct['day'])
- elif dct['__class__'] == 'time':
- dct = datetime.time(dct['hour'], dct['minute'], dct['second'],
- dct['microsecond'])
- elif dct['__class__'] == 'Decimal':
- dct = Decimal(dct['decimal'])
+ dct = XMLRPCDecoder()(dct)
self._stack[mark:] = [dct]
self._value = 0
@@ -154,7 +170,7 @@ class _TrytondMethod(object):
for i in inst]
if not rpc.readonly:
transaction.cursor.commit()
- Cache.resets(self._config.database_name)
+ Cache.resets(self._config.database_name)
return result
@@ -175,40 +191,32 @@ class TrytondProxy(object):
class TrytondConfig(Config):
'Configuration for trytond'
- def __init__(self, database_name=None, user='admin', database_type=None,
- language='en_US', password='', config_file=None):
+ def __init__(self, database=None, user='admin', language='en_US',
+ password='', config_file=os.environ.get('TRYTOND_CONFIG')):
super(TrytondConfig, self).__init__()
- from trytond.config import CONFIG
- CONFIG.update_etc(config_file)
- if database_type is not None:
- CONFIG['db_type'] = database_type
+ if not database:
+ database = os.environ.get('TRYTOND_DATABASE_URI')
+ else:
+ os.environ['TRYTOND_DATABASE_URI'] = database
+ from trytond.config import config
+ config.update_etc(config_file)
from trytond.pool import Pool
- from trytond import backend
- from trytond.protocols.dispatcher import create
from trytond.cache import Cache
from trytond.transaction import Transaction
- self.database_type = CONFIG['db_type']
- if database_name is None:
- if self.database_type == 'sqlite':
- database_name = ':memory:'
- else:
- database_name = 'test_%s' % int(time.time())
+ self.database = database
+ database_name = None
+ if database:
+ uri = urlparse.urlparse(database)
+ database_name = uri.path.strip('/')
+ if not database_name:
+ database_name = os.environ['DB_NAME']
self.database_name = database_name
self._user = user
self.config_file = config_file
Pool.start()
-
- with Transaction().start(None, 0) as transaction:
- cursor = transaction.cursor
- databases = backend.get('Database').list(cursor)
- if database_name not in databases:
- create(database_name, CONFIG['admin_passwd'], language, password)
-
- database_list = Pool.database_list()
self.pool = Pool(database_name)
- if database_name not in database_list:
- self.pool.init()
+ self.pool.init()
with Transaction().start(self.database_name, 0) as transaction:
Cache.clean(database_name)
@@ -219,14 +227,13 @@ class TrytondConfig(Config):
], limit=1)[0].id
with transaction.set_user(self.user):
self._context = User.get_preferences(context_only=True)
- Cache.resets(database_name)
+ Cache.resets(database_name)
__init__.__doc__ = object.__init__.__doc__
def __repr__(self):
return "proteus.config.TrytondConfig"\
"('%s', '%s', '%s', config_file=%s)"\
- % (self.database_name, self._user, self.database_type,
- self.config_file)
+ % (self.database, self._user, self.config_file)
__repr__.__doc__ = object.__repr__.__doc__
def __eq__(self, other):
@@ -234,12 +241,12 @@ class TrytondConfig(Config):
raise NotImplementedError
return (self.database_name == other.database_name
and self._user == other._user
- and self.database_type == other.database_type
+ and self.database == other.database
and self.config_file == other.config_file)
def __hash__(self):
return hash((self.database_name, self._user,
- self.database_type, self.config_file))
+ self.database, self.config_file))
def get_proxy(self, name, type='model'):
'Return Proxy class'
@@ -254,11 +261,11 @@ class TrytondConfig(Config):
return methods
-def set_trytond(database_name=None, user='admin', database_type=None,
- language='en_US', password='', config_file=None):
+def set_trytond(database=None, user='admin', language='en_US', password='',
+ config_file=None):
'Set trytond package as backend'
- _CONFIG.current = TrytondConfig(database_name, user, database_type,
- language=language, password=password, config_file=config_file)
+ _CONFIG.current = TrytondConfig(database, user, language=language,
+ password=password, config_file=config_file)
return _CONFIG.current
diff --git a/proteus/tests/common.py b/proteus/tests/common.py
new file mode 100644
index 0000000..15e1622
--- /dev/null
+++ b/proteus/tests/common.py
@@ -0,0 +1,18 @@
+#This file is part of Tryton. The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+import os
+from unittest import TestCase
+
+os.environ.setdefault('TRYTOND_DATABASE_URI', 'sqlite://')
+os.environ.setdefault('DB_NAME', ':memory:')
+from trytond.tests.test_tryton import db_exist, create_db
+
+from proteus import config
+
+
+class ProteusTestCase(TestCase):
+
+ def setUp(self):
+ if not db_exist():
+ create_db()
+ self.config = config.set_trytond()
diff --git a/proteus/tests/test_config.py b/proteus/tests/test_config.py
index 35433a3..2ad83f2 100644
--- a/proteus/tests/test_config.py
+++ b/proteus/tests/test_config.py
@@ -1,13 +1,10 @@
#This file is part of Tryton. The COPYRIGHT file at the top level of
#this repository contains the full copyright notices and license terms.
-from unittest import TestCase
import proteus.config
+from .common import ProteusTestCase
-class TestConfig(TestCase):
-
- def setUp(self):
- proteus.config.set_trytond(database_type='sqlite')
+class TestConfig(ProteusTestCase):
def test_proxy(self):
config = proteus.config.get_config()
@@ -22,7 +19,7 @@ class TestConfig(TestCase):
def test_trytond_config_eq(self):
config1 = proteus.config.get_config()
- proteus.config.set_trytond(database_type='sqlite')
+ proteus.config.set_trytond()
config2 = proteus.config.get_config()
self.assertEqual(config1, config2)
diff --git a/proteus/tests/test_context.py b/proteus/tests/test_context.py
index dc7faea..7cbacec 100644
--- a/proteus/tests/test_context.py
+++ b/proteus/tests/test_context.py
@@ -1,14 +1,9 @@
# This file is part of Tryton. The COPYRIGHT file at the top level of this
# repository contains the full copyright notices and license terms.
+from .common import ProteusTestCase
-from unittest import TestCase
-from proteus import config
-
-class TestContext(TestCase):
-
- def setUp(self):
- self.config = config.set_trytond(database_type='sqlite')
+class TestContext(ProteusTestCase):
def test_config(self):
prev_ctx = self.config._context
diff --git a/proteus/tests/test_model.py b/proteus/tests/test_model.py
index 0a87cfc..f6aa197 100644
--- a/proteus/tests/test_model.py
+++ b/proteus/tests/test_model.py
@@ -1,13 +1,10 @@
#This file is part of Tryton. The COPYRIGHT file at the top level of
#this repository contains the full copyright notices and license terms.
-from unittest import TestCase
-from proteus import config, Model
+from proteus import Model
+from .common import ProteusTestCase
-class TestModel(TestCase):
-
- def setUp(self):
- config.set_trytond(database_type='sqlite')
+class TestModel(ProteusTestCase):
def test_class_cache(self):
User1 = Model.get('res.user')
@@ -39,7 +36,7 @@ class TestModel(TestCase):
admin.create_uid = admin
admin.create_uid = None
- User(write_uid=False)
+ User(write_uid=None)
def test_one2many(self):
Group = Model.get('res.group')
@@ -278,6 +275,17 @@ class TestModel(TestCase):
test.save()
test.delete()
+ def test_duplicate(self):
+ User = Model.get('res.user')
+ test = User()
+ test.name = 'Test duplicate'
+ test.login = 'test duplicate'
+ test.save()
+ copy, = User.duplicate([test], {'name': 'Test copy'})
+ self.assertEqual(copy.name, 'Test copy')
+ self.assertEqual(copy.login, 'test duplicate (copy)')
+ self.assertNotEqual(copy.id, test.id)
+
def test_on_change(self):
Trigger = Model.get('ir.trigger')
diff --git a/proteus/tests/test_report.py b/proteus/tests/test_report.py
new file mode 100644
index 0000000..d11c7bc
--- /dev/null
+++ b/proteus/tests/test_report.py
@@ -0,0 +1,21 @@
+#This file is part of Tryton. The COPYRIGHT file at the top level of
+#this repository contains the full copyright notices and license terms.
+from unittest import TestCase
+from proteus import config, Report, Model
+from .common import ProteusTestCase
+
+
+class TestReport(ProteusTestCase):
+
+ def test_model_graph(self):
+ IrModel = Model.get('ir.model')
+ models = IrModel.find([])
+ data = {
+ 'level': 1,
+ 'filter': '',
+ }
+ report = Report('ir.model.graph')
+ type_, data, print_, name = report.execute(models, data)
+ self.assertEqual(type_, 'png')
+ self.assertEqual(print_, False)
+ self.assertEqual(name, 'Graph')
diff --git a/proteus/tests/test_wizard.py b/proteus/tests/test_wizard.py
index 86fb2a8..68b5226 100644
--- a/proteus/tests/test_wizard.py
+++ b/proteus/tests/test_wizard.py
@@ -1,13 +1,10 @@
#This file is part of Tryton. The COPYRIGHT file at the top level of
#this repository contains the full copyright notices and license terms.
-from unittest import TestCase
-from proteus import config, Wizard, Model
+from proteus import Wizard, Model
+from .common import ProteusTestCase
-class TestWizard(TestCase):
-
- def setUp(self):
- config.set_trytond(database_type='sqlite')
+class TestWizard(ProteusTestCase):
def test_translation_clean(self):
translation_clean = Wizard('ir.translation.clean')
--
tryton-proteus
More information about the tryton-debian-vcs
mailing list