|
12 | 12 | except ImportError:
|
13 | 13 | # python2
|
14 | 14 | pass
|
| 15 | + |
15 | 16 | import inspect
|
16 | 17 | import itertools
|
17 | 18 | import logging
|
@@ -175,8 +176,7 @@ def check(self, event_data):
|
175 | 176 | model attached to the current machine which is used to invoke
|
176 | 177 | the condition.
|
177 | 178 | """
|
178 |
| - predicate = getattr(event_data.model, self.func) if isinstance(self.func, string_types) else self.func |
179 |
| - |
| 179 | + predicate = event_data.machine.resolve_callable(self.func, event_data) |
180 | 180 | if event_data.machine.send_event:
|
181 | 181 | return predicate(event_data) == self.target
|
182 | 182 | return predicate(*event_data.args, **event_data.kwargs) == self.target
|
@@ -989,19 +989,45 @@ def callback(self, func, event_data):
|
989 | 989 | the callable will be resolved from the passed model in event_data. This function is not intended to
|
990 | 990 | be called directly but through state and transition callback definitions.
|
991 | 991 | Args:
|
992 |
| - func (callable or str): The callback function. |
| 992 | + func (string, callable): The callback function. |
| 993 | + 1. First, if the func is callable, just call it |
| 994 | + 2. Second, we try to import string assuming it is a path to a func |
| 995 | + 3. Fallback to a model attribute |
993 | 996 | event_data (EventData): An EventData instance to pass to the
|
994 | 997 | callback (if event sending is enabled) or to extract arguments
|
995 | 998 | from (if event sending is disabled).
|
996 | 999 | """
|
997 |
| - if isinstance(func, string_types): |
998 |
| - func = getattr(event_data.model, func) |
999 | 1000 |
|
| 1001 | + func = self.resolve_callable(func, event_data) |
1000 | 1002 | if self.send_event:
|
1001 | 1003 | func(event_data)
|
1002 | 1004 | else:
|
1003 | 1005 | func(*event_data.args, **event_data.kwargs)
|
1004 | 1006 |
|
| 1007 | + @staticmethod |
| 1008 | + def resolve_callable(func, event_data): |
| 1009 | + """ Converts path to a callable into callable |
| 1010 | + Args: |
| 1011 | + func (string, callable): Path to a callable |
| 1012 | + event_data (EventData): Currently processed event |
| 1013 | + Returns: |
| 1014 | + callable function resolved from string or func |
| 1015 | + """ |
| 1016 | + if isinstance(func, string_types): |
| 1017 | + try: |
| 1018 | + func = getattr(event_data.model, func) |
| 1019 | + except AttributeError: |
| 1020 | + try: |
| 1021 | + mod, name = func.rsplit('.', 1) |
| 1022 | + m = __import__(mod) |
| 1023 | + for n in mod.split('.')[1:]: |
| 1024 | + m = getattr(m, n) |
| 1025 | + func = getattr(m, name) |
| 1026 | + except (ImportError, AttributeError, ValueError): |
| 1027 | + raise AttributeError("Callable with name '%s' could neither be retrieved from the passed " |
| 1028 | + "model nor imported from a module.") |
| 1029 | + return func |
| 1030 | + |
1005 | 1031 | def _has_state(self, state):
|
1006 | 1032 | if isinstance(state, State):
|
1007 | 1033 | if state in self.states.values():
|
|
0 commit comments