Skip to content

Commit 7b3de7d

Browse files
authored
Added the ability to disable hooks (#326)
* Added the ability to disable hooks Generic fix for issue #236 * Moved HooksDisabled-check to the top * Updated documentation for hooks_disabled context manager
1 parent b6fc461 commit 7b3de7d

File tree

4 files changed

+80
-0
lines changed

4 files changed

+80
-0
lines changed

addons/source-python/packages/source-python/memory/hooks.py

+49
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,16 @@
55
# =============================================================================
66
# >> IMPORTS
77
# =============================================================================
8+
# Python
9+
from contextlib import contextmanager
10+
811
# Source.Python Imports
912
# Core
1013
from core import AutoUnload
1114
# Memory
1215
from _memory import HookType
16+
from _memory import set_hooks_disabled
17+
from _memory import get_hooks_disabled
1318
from memory import Function
1419

1520

@@ -19,6 +24,9 @@
1924
__all__ = ('HookType',
2025
'PostHook',
2126
'PreHook',
27+
'set_hooks_disabled',
28+
'get_hooks_disabled',
29+
'hooks_disabled'
2230
)
2331

2432

@@ -73,3 +81,44 @@ class PostHook(_Hook):
7381
"""Decorator class used to create post hooks that auto unload."""
7482

7583
hook_type = HookType.POST
84+
85+
86+
# =============================================================================
87+
# >> FUNCTIONS
88+
# =============================================================================
89+
@contextmanager
90+
def hooks_disabled(disabled=True):
91+
"""Temporarily disable or enable all hook callbacks. By default hooks are
92+
enabled. Thus, this context manager is mainly used to temporarily disable
93+
hook callbacks. If the context ends, the original value is restored. This
94+
can be used e. g. to avoid recursive calls when damaging a player in a
95+
``on_take_damage`` hook or ``player_hurt`` event.
96+
97+
.. note::
98+
99+
This would only disable hooks created with Source.Python. Hooks that
100+
have been created by other server plugins will still be called.
101+
102+
Example:
103+
104+
.. code:: python
105+
106+
from players.entity import Player
107+
from memory.hooks import hooks_disabled
108+
109+
# Get a Player instance of the player with index 1
110+
player = Player(1)
111+
112+
# Damage the player. This would call e. g. on_take_damage hooks
113+
player.take_damage(5)
114+
115+
# To avoid calling the on_take_damage hooks, use the following:
116+
with hooks_disabled()
117+
player.take_damage(5)
118+
"""
119+
old = get_hooks_disabled()
120+
set_hooks_disabled(disabled)
121+
try:
122+
yield
123+
finally:
124+
set_hooks_disabled(old)

src/core/modules/memory/memory_hooks.cpp

+5
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,8 @@ using namespace boost::python;
4444
// g_mapCallbacks[<CHook *>][<HookType_t>] -> [<object>, <object>, ...]
4545
std::map<CHook *, std::map<HookType_t, std::list<object> > > g_mapCallbacks;
4646

47+
bool g_HooksDisabled;
48+
4749

4850
// ============================================================================
4951
// >> HELPER FUNCTIONS
@@ -80,6 +82,9 @@ object GetArgument(CHook* pHook, int iIndex)
8082
// ============================================================================
8183
bool SP_HookHandler(HookType_t eHookType, CHook* pHook)
8284
{
85+
if (g_HooksDisabled)
86+
return false;
87+
8388
std::list<object> callbacks = g_mapCallbacks[pHook][eHookType];
8489

8590
// No need to do all this stuff, if there is no callback registered

src/core/modules/memory/memory_hooks.h

+12
Original file line numberDiff line numberDiff line change
@@ -76,4 +76,16 @@ class CStackData
7676
//---------------------------------------------------------------------------------
7777
bool SP_HookHandler(HookType_t eHookType, CHook* pHook);
7878

79+
extern bool g_HooksDisabled;
80+
81+
inline void SetHooksDisabled(bool value)
82+
{
83+
g_HooksDisabled = value;
84+
}
85+
86+
inline bool GetHooksDisabled()
87+
{
88+
return g_HooksDisabled;
89+
}
90+
7991
#endif // MEMORY_HOOKS_H

src/core/modules/memory/memory_wrap.cpp

+14
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,8 @@ void export_protection(scope);
6868
// ============================================================================
6969
DECLARE_SP_MODULE(_memory)
7070
{
71+
SetHooksDisabled(false);
72+
7173
export_function_info(_memory);
7274
export_binary_file(_memory);
7375
export_pointer(_memory);
@@ -956,6 +958,18 @@ void export_functions(scope _memory)
956958
":param DataType data_type: The data type you would like to get the size of.\n"
957959
":param int alignment: The alignment that should be used."
958960
);
961+
962+
def("get_hooks_disabled",
963+
&GetHooksDisabled,
964+
"Return whether or not hook callbacks are disabled.\n"
965+
"\n"
966+
":rtype: bool");
967+
968+
def("set_hooks_disabled",
969+
&SetHooksDisabled,
970+
"Set whether or not hook callbacks are disabled.\n"
971+
"\n"
972+
":param bool disabled: If ``True``, hook callbacks are disabled.");
959973
}
960974

961975

0 commit comments

Comments
 (0)