[Python-modules-commits] [django-fsm] 01/01: Import django-fsm_2.5.0.orig.tar.gz

Michael Fladischer fladi at moszumanska.debian.org
Mon Mar 6 11:27:45 UTC 2017


This is an automated email from the git hooks/post-receive script.

fladi pushed a commit to branch upstream-experimental
in repository django-fsm.

commit 5925f6f84acb77a9ad46fd8b220c5e181a2418db
Author: Michael Fladischer <FladischerMichael at fladi.at>
Date:   Mon Mar 6 11:37:49 2017 +0100

    Import django-fsm_2.5.0.orig.tar.gz
---
 CHANGELOG.rst                                      |  9 +++
 README.rst                                         | 30 ++++-----
 django_fsm/__init__.py                             | 20 +++---
 .../management/commands/graph_transitions.py       | 71 +++++++++++++++-------
 setup.cfg                                          |  2 +
 setup.py                                           |  3 +-
 tests/testapp/tests/test_multi_resultstate.py      | 29 +++++++++
 tox.ini                                            | 38 ++++++------
 8 files changed, 139 insertions(+), 63 deletions(-)

diff --git a/CHANGELOG.rst b/CHANGELOG.rst
index a804f97..9d38bd5 100644
--- a/CHANGELOG.rst
+++ b/CHANGELOG.rst
@@ -1,6 +1,15 @@
 Changelog
 =========
 
+django-fsm 2.5.0 2017-03-04
+~~~~~~~~~~~~~~~~~~~~~~~~~~~
+
+- graph_transition command fix for django 1.10
+- graph_transition command supports GET_STATE targets
+- signal data extended with method args/kwargs and field
+- sets allowed to be passed to the transition decorator
+
+
 django-fsm 2.4.0 2016-05-14
 ~~~~~~~~~~~~~~~~~~~~~~~~~~~
 
diff --git a/README.rst b/README.rst
index 1d52913..31cca8b 100644
--- a/README.rst
+++ b/README.rst
@@ -1,7 +1,7 @@
 Django friendly finite state machine support
 ============================================
 
-|Build Status| |Downloads| |Gitter|
+|Build Status| |Gitter|
 
 django-fsm adds simple declarative states management for django models.
 
@@ -149,13 +149,13 @@ direct state field modification.
     model.state = 'invalid' # Raises AttributeError
 
 Note that calling
-```refresh_from_db`` <https://docs.djangoproject.com/en/1.8/ref/models/instances/#django.db.models.Model.refresh_from_db>`__
+`refresh_from_db <https://docs.djangoproject.com/en/1.8/ref/models/instances/#django.db.models.Model.refresh_from_db>`_
 on a model instance with a protected FSMField will cause an exception.
 
 `target`
 ~~~~~~~~
 
-`target` state parameter could points to the specific state or `django_fsm.State` implementation
+`target` state parameter could point to a specific state or `django_fsm.State` implementation
 
 .. code:: python
           
@@ -164,7 +164,7 @@ on a model instance with a protected FSMField will cause an exception.
                 source='*',
                 target=RETURN_VALUE('for_moderators', 'published'))
     def publish(self, is_public=False):
-        return 'need_moderation' if is_public else 'published'
+        return 'for_moderators' if is_public else 'published'
 
     @transition(
         field=state,
@@ -229,8 +229,8 @@ Permissions
 It is common to have permissions attached to each model transition.
 ``django-fsm`` handles this with ``permission`` keyword on the
 ``transition`` decorator. ``permission`` accepts a permission string, or
-callable that expects ``user`` argument and returns True if user can
-perform the transition
+callable that expects ``instance`` and ``user`` arguments and returns
+True if user can perform the transition.
 
 .. code:: python
 
@@ -374,10 +374,10 @@ model.save()
         state = FSMField(default='new')
 
 For guaranteed protection against race conditions caused by concurrently
-executed transitions, make sure: \* Your transitions do not have any
-side effects except for changes in the database, \* You always run the
-save() method on the object within ``django.db.transaction.atomic()``
-block.
+executed transitions, make sure:
+
+- Your transitions do not have any side effects except for changes in the database,
+- You always run the save() method on the object within ``django.db.transaction.atomic()`` block.
 
 Following these recommendations, you can rely on
 ConcurrentTransitionMixin to cause a rollback of all the changes that
@@ -411,17 +411,17 @@ your ``INSTALLED_APPS``:
 Changelog
 ---------
 
-django-fsm 2.4.0 2016-05-14
+django-fsm 2.5.0 2017-03-04
 ~~~~~~~~~~~~~~~~~~~~~~~~~~~
 
-- graph_transition commnad now works with multiple  FSM's per model
-- Add ability to set target state from transition return value or callable
+- graph_transition command fix for django 1.10
+- graph_transition command supports GET_STATE targets
+- signal data extended with method args/kwargs and field
+- sets allowed to be passed to the transition decorator
 
 
 
 .. |Build Status| image:: https://travis-ci.org/kmmbvnr/django-fsm.svg?branch=master
    :target: https://travis-ci.org/kmmbvnr/django-fsm
-.. |Downloads| image:: https://img.shields.io/pypi/dm/django-fsm.svg
-   :target: https://pypi.python.org/pypi/django-fsm
 .. |Gitter| image:: https://badges.gitter.im/Join%20Chat.svg
    :target: https://gitter.im/kmmbvnr/django-fsm?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge
diff --git a/django_fsm/__init__.py b/django_fsm/__init__.py
index 74e0274..d93cff9 100644
--- a/django_fsm/__init__.py
+++ b/django_fsm/__init__.py
@@ -245,7 +245,7 @@ class FSMFieldMixin(object):
         state_choices = kwargs.pop('state_choices', None)
         choices = kwargs.get('choices', None)
         if state_choices is not None and choices is not None:
-            raise ValueError('Use one of choices or state_choces value')
+            raise ValueError('Use one of choices or state_choices value')
 
         if state_choices is not None:
             choices = []
@@ -308,8 +308,11 @@ class FSMFieldMixin(object):
             'sender': instance.__class__,
             'instance': instance,
             'name': method_name,
+            'field': meta.field,
             'source': current_state,
-            'target': next_state
+            'target': next_state,
+            'method_args' : args,
+            'method_kwargs' : kwargs
         }
 
         pre_transition.send(**signal_kwargs)
@@ -321,6 +324,7 @@ class FSMFieldMixin(object):
                     next_state = next_state.get_state(
                         instance, transition, result,
                         args=args, kwargs=kwargs)
+                    signal_kwargs['target'] = next_state
                 self.set_proxy(instance, next_state)
                 self.set_state(instance, next_state)
         except Exception as exc:
@@ -349,10 +353,10 @@ class FSMFieldMixin(object):
             for transition in meta.transitions.values():
                 yield transition
 
-    def contribute_to_class(self, cls, name, virtual_only=False):
+    def contribute_to_class(self, cls, name, **kwargs):
         self.base_cls = cls
 
-        super(FSMFieldMixin, self).contribute_to_class(cls, name, virtual_only=virtual_only)
+        super(FSMFieldMixin, self).contribute_to_class(cls, name, **kwargs)
         setattr(cls, self.name, self.descriptor_class(self))
         setattr(cls, 'get_all_{0}_transitions'.format(self.name),
                 curry(get_all_FIELD_transitions, field=self))
@@ -500,7 +504,7 @@ def transition(field, source='*', target=None, on_error=None, conditions=[], per
             fsm_meta = FSMMeta(field=field, method=func)
             setattr(func, '_django_fsm', fsm_meta)
 
-        if isinstance(source, (list, tuple)):
+        if isinstance(source, (list, tuple, set)):
             for state in source:
                 func._django_fsm.add_transition(func, state, target, on_error, conditions, permission, custom)
         else:
@@ -526,7 +530,8 @@ def can_proceed(bound_method, check_conditions=True):
     conditions.
     """
     if not hasattr(bound_method, '_django_fsm'):
-        raise TypeError('%s method is not transition' % bound_method.im_func.__name__)
+        im_func = getattr(bound_method, 'im_func', getattr(bound_method, '__func__'))
+        raise TypeError('%s method is not transition' % im_func.__name__)
 
     meta = bound_method._django_fsm
     im_self = getattr(bound_method, 'im_self', getattr(bound_method, '__self__'))
@@ -541,7 +546,8 @@ def has_transition_perm(bound_method, user):
     Returns True if model in state allows to call bound_method and user have rights on it
     """
     if not hasattr(bound_method, '_django_fsm'):
-        raise TypeError('%s method is not transition' % bound_method.im_func.__name__)
+        im_func = getattr(bound_method, 'im_func', getattr(bound_method, '__func__'))
+        raise TypeError('%s method is not transition' % im_func.__name__)
 
     meta = bound_method._django_fsm
     im_self = getattr(bound_method, 'im_self', getattr(bound_method, '__self__'))
diff --git a/django_fsm/management/commands/graph_transitions.py b/django_fsm/management/commands/graph_transitions.py
index bf4a636..8eea6bf 100644
--- a/django_fsm/management/commands/graph_transitions.py
+++ b/django_fsm/management/commands/graph_transitions.py
@@ -5,7 +5,7 @@ from optparse import make_option
 from django.core.management.base import BaseCommand
 from django.utils.encoding import smart_text
 
-from django_fsm import FSMFieldMixin
+from django_fsm import FSMFieldMixin, GET_STATE, RETURN_VALUE
 
 try:
     from django.db.models import get_apps, get_app, get_models, get_model
@@ -14,6 +14,10 @@ except ImportError:
     from django.apps import apps
     NEW_META_API = True
 
+from django import VERSION
+
+HAS_ARGPARSE = VERSION >= (1, 10)
+
 
 def all_fsm_fields_data(model):
     if NEW_META_API:
@@ -44,18 +48,13 @@ def generate_dot(fields_data):
             else:
                 source_name = node_name(field, transition.source)
                 if transition.target is not None:
-                    target_name = node_name(field, transition.target)
-                    if isinstance(transition.source, int):
-                        source_label = [smart_text(name[1]) for name in field.choices if name[0] == transition.source][0]
+                    if isinstance(transition.target, GET_STATE) or isinstance(transition.target, RETURN_VALUE):
+                        for transition_target_index, transition_target in enumerate(transition.target.allowed_states):
+                            add_transition(transition.source, transition_target, transition.name,
+                                           source_name, field, sources, targets, edges)
                     else:
-                        source_label = transition.source
-                    sources.add((source_name, source_label))
-                    if isinstance(transition.target, int):
-                        target_label = [smart_text(name[1]) for name in field.choices if name[0] == transition.target][0]
-                    else:
-                        target_label = transition.target
-                    targets.add((target_name, target_label))
-                    edges.add((source_name, target_name, (('label', transition.name),)))
+                        add_transition(transition.source, transition.target, transition.name,
+                                       source_name, field, sources, targets, edges)
             if transition.on_error:
                 on_error_name = node_name(field, transition.on_error)
                 targets.add((on_error_name, transition.on_error))
@@ -99,21 +98,49 @@ def generate_dot(fields_data):
     return result
 
 
+def add_transition(transition_source, transition_target, transition_name, source_name, field, sources, targets, edges):
+    target_name = node_name(field, transition_target)
+    if isinstance(transition_source, int):
+        source_label = [smart_text(name[1]) for name in field.choices if name[0] == transition_source][0]
+    else:
+        source_label = transition_source
+    sources.add((source_name, source_label))
+    if isinstance(transition_target, int):
+        target_label = [smart_text(name[1]) for name in field.choices if name[0] == transition_target][0]
+    else:
+        target_label = transition_target
+    targets.add((target_name, target_label))
+    edges.add((source_name, target_name, (('label', transition_name),)))
+
+
 class Command(BaseCommand):
     requires_system_checks = True
 
-    option_list = BaseCommand.option_list + (
-        make_option('--output', '-o', action='store', dest='outputfile',
-                    help=('Render output file. Type of output dependent on file extensions. '
-                          'Use png or jpg to render graph to image.')),
-        # NOQA
-        make_option('--layout', '-l', action='store', dest='layout', default='dot',
-                    help=('Layout to be used by GraphViz for visualization. '
-                          'Layouts: circo dot fdp neato nop nop1 nop2 twopi')),
-    )
+    if not HAS_ARGPARSE:
+        option_list = BaseCommand.option_list + (
+            make_option('--output', '-o', action='store', dest='outputfile',
+                        help=('Render output file. Type of output dependent on file extensions. '
+                              'Use png or jpg to render graph to image.')),
+            # NOQA
+            make_option('--layout', '-l', action='store', dest='layout', default='dot',
+                        help=('Layout to be used by GraphViz for visualization. '
+                              'Layouts: circo dot fdp neato nop nop1 nop2 twopi')),
+        )
+        args = "[appname[.model[.field]]]"
+    else:
+        def add_arguments(self, parser):
+            parser.add_argument(
+                '--output', '-o', action='store', dest='outputfile',
+                help=('Render output file. Type of output dependent on file extensions. '
+                      'Use png or jpg to render graph to image.'))
+            parser.add_argument(
+                '--layout', '-l', action='store', dest='layout', default='dot',
+                help=('Layout to be used by GraphViz for visualization. '
+                      'Layouts: circo dot fdp neato nop nop1 nop2 twopi'))
+            parser.add_argument('args', nargs='*',
+                                help=('[appname[.model[.field]]]'))
 
     help = ("Creates a GraphViz dot file with transitions for selected fields")
-    args = "[appname[.model[.field]]]"
 
     def render_output(self, graph, **options):
         filename, format = options['outputfile'].rsplit('.', 1)
diff --git a/setup.cfg b/setup.cfg
new file mode 100644
index 0000000..2a9acf1
--- /dev/null
+++ b/setup.cfg
@@ -0,0 +1,2 @@
+[bdist_wheel]
+universal = 1
diff --git a/setup.py b/setup.py
index 68eb170..61eb245 100644
--- a/setup.py
+++ b/setup.py
@@ -7,7 +7,7 @@ except IOError:
 
 setup(
     name='django-fsm',
-    version='2.4.0',
+    version='2.5.0',
     description='Django friendly finite state machine support.',
     author='Mikhail Podgurskiy',
     author_email='kmmbvnr at gmail.com',
@@ -27,6 +27,7 @@ setup(
         "Framework :: Django",
         "Framework :: Django :: 1.8",
         "Framework :: Django :: 1.9",
+        "Framework :: Django :: 1.10",
         'Programming Language :: Python',
         'Programming Language :: Python :: 2.6',
         'Programming Language :: Python :: 2.7',
diff --git a/tests/testapp/tests/test_multi_resultstate.py b/tests/testapp/tests/test_multi_resultstate.py
index e7b385d..ebdb9c4 100644
--- a/tests/testapp/tests/test_multi_resultstate.py
+++ b/tests/testapp/tests/test_multi_resultstate.py
@@ -1,6 +1,7 @@
 from django.db import models
 from django.test import TestCase
 from django_fsm import FSMField, transition, RETURN_VALUE, GET_STATE
+from django_fsm.signals import pre_transition, post_transition
 
 
 class MultiResultTest(models.Model):
@@ -38,3 +39,31 @@ class Test(TestCase):
         instance = MultiResultTest(state='for_moderators')
         instance.moderate(allowed=False)
         self.assertEqual(instance.state, 'rejected')
+
+
+class TestSignals(TestCase):
+    def setUp(self):
+        self.pre_transition_called = False
+        self.post_transition_called = False
+        pre_transition.connect(self.on_pre_transition, sender=MultiResultTest)
+        post_transition.connect(self.on_post_transition, sender=MultiResultTest)
+
+    def on_pre_transition(self, sender, instance, name, source, target, **kwargs):
+        self.assertEqual(instance.state, source)
+        self.pre_transition_called = True
+
+    def on_post_transition(self, sender, instance, name, source, target, **kwargs):
+        self.assertEqual(instance.state, target)
+        self.post_transition_called = True
+
+    def test_signals_called_with_get_state(self):
+        instance = MultiResultTest(state='for_moderators')
+        instance.moderate(allowed=False)
+        self.assertTrue(self.pre_transition_called)
+        self.assertTrue(self.post_transition_called)
+
+    def test_signals_called_with_return_value(self):
+        instance = MultiResultTest()
+        instance.publish(is_public=True)
+        self.assertTrue(self.pre_transition_called)
+        self.assertTrue(self.post_transition_called)
diff --git a/tox.ini b/tox.ini
index 53e4e4b..6856c4b 100644
--- a/tox.ini
+++ b/tox.ini
@@ -1,33 +1,35 @@
 [tox]
 envlist =
-    py26-{dj16}
-    py27-{dj16,dj18,dj19}
-    py33-{dj16,dj18}
-    py{34,35}-{dj18,dj19}
+    py26-dj{16}
+    py27-dj{16,18,19,110,111}
+    py33-dj{16,18}
+    py{34,35}-dj{18,19,110,111}
 skipsdist = True
 
 [testenv]
-basepython =
-    py26: python2.6
-    py27: python2.7
-    py33: python3.3
-    py34: python3.4
-    py35: python3.5
 deps =
     py26: ipython==2.1.0
-    {py27,py32,py33,py34,py35}: ipython==4.1.1    
+    {py27,py32,py33,py34,py35}: ipython==4.1.1
     dj16: Django==1.6.11
     dj16: django-jenkins==0.17.0
     dj16: coverage<=3.999
     dj16: django-guardian==1.3.2
-    dj18: Django==1.8.9
+    dj18: Django==1.8.13
     dj18: django-jenkins==0.18.1
-    dj18: coverage==4.0.3
-    dj18: django-guardian==1.4.1
-    dj19: Django==1.9.2
-    dj19: django-jenkins==0.18.1
-    dj19: coverage==4.0.3
-    dj19: django-guardian==1.4.1
+    dj18: coverage==4.1
+    dj18: django-guardian==1.4.4
+    dj19: Django==1.9.7
+    dj19: django-jenkins==0.19.0
+    dj19: coverage==4.1
+    dj19: django-guardian==1.4.4
+    dj110: Django==1.10.5
+    dj110: django-jenkins==0.19.0
+    dj110: coverage==4.1
+    dj110: django-guardian==1.4.4
+    dj111: Django==1.11b1
+    dj111: django-jenkins==0.110.0
+    dj111: coverage==4.3.4
+    dj111: django-guardian==1.4.6
     graphviz==0.4.10
     pep8==1.7.0
     pyflakes==1.0.0

-- 
Alioth's /usr/local/bin/git-commit-notice on /srv/git.debian.org/git/python-modules/packages/django-fsm.git



More information about the Python-modules-commits mailing list