跳转至

cooldown

CoolDown 🔗

Bases: BaseDispatcher, Generic[T_SourceKey]

指示需要冷却时间才能执行操作

Source code in src/graia/ariadne/util/cooldown.py
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
class CoolDown(BaseDispatcher, Generic[T_SourceKey]):
    """指示需要冷却时间才能执行操作"""

    global_source: ClassVar[Dict[str, Dict[Any, datetime]]] = {}

    source: MutableMapping[T_SourceKey, datetime]

    def __init__(
        self,
        interval: Union[int, float, timedelta],
        source: Union[MutableMapping[T_SourceKey, datetime], str, None] = None,
        override_condition: Callable[..., Union[bool, Awaitable[bool]]] = lambda: False,
        stop_on_cooldown: bool = True,
    ) -> None:
        """初始化一个冷却时间

        Args:
            interval (Union[int, float, timedelta]): 冷却时间, 单位为秒
            source (Union[MutableMapping[int, datetime], str, None], optional): 冷却映射来源, \
                为字符串时从 ClassVar 查找.
            override_condition ((...) -> Union[bool, Awaitable[bool]], optional): 超越冷却限制的条件.
            stop_on_cooldown (bool, optional): 是否在未到冷却时间时直接停止执行. Defaults to True.
        """
        self.interval = interval if isinstance(interval, timedelta) else timedelta(seconds=interval)
        self.stop_on_cooldown: bool = stop_on_cooldown
        self.override_condition: Callable[..., Union[bool, Awaitable[bool]]] = override_condition
        self.override_signature = argument_signature(self.override_condition)
        if isinstance(source, str):
            self.source = self.global_source.setdefault(source, {})
        else:
            self.source = source or {}

    async def fetch_target_key(self, event: Dispatchable) -> T_SourceKey:
        """获取目标的键,以在 source 中获取对应的冷却信息

        Args:
            event (Dispatchable): 当前事件

        Returns:
            T_SourceKey: 目标的 “冷却哈希”
        """
        if not isinstance(event, MessageEvent):
            raise ExecutionStop
        return cast(T_SourceKey, event.sender.id)

    async def get(self, target: T_SourceKey, type: Type[T_Time]) -> Tuple[T_Time, bool]:
        """获取目标的冷却信息

        Args:
            target (T_SourceKey): 目标的 “冷却哈希”
            type (Type[T_Time]): 需要返回的类型

        Returns:
            Tuple[T_Time, bool]: 第一个值是剩余的冷却时间 (或下一次可执行时间),\
                第二个值是冷却是否完成 \
                如果 type 传入的是 Optional[XXX] 则第一个值可以是 None
        """
        current_time: datetime = datetime.now()
        next_exec_time: datetime = self.source.get(target, current_time)
        delta: timedelta = next_exec_time - current_time
        satisfied: bool = delta < timedelta(seconds=0)
        if NoneType in typing.get_args(type) and delta.total_seconds() <= 0:
            result = None, satisfied
        elif generic_issubclass(datetime, type):
            result = next_exec_time, satisfied
        elif generic_issubclass(timedelta, type):
            result = delta, satisfied
        elif generic_issubclass(float, type):
            result = delta.total_seconds(), satisfied
        elif generic_issubclass(int, type):
            result = int(delta.total_seconds()), satisfied
        else:
            result = None, satisfied
        return cast(Tuple[T_Time, bool], result)

    async def set(self, target: T_SourceKey) -> None:
        """更新目标的冷却信息

        Args:
            target (T_SourceKey): 目标的 “冷却哈希”
        """
        self.source[target] = datetime.now() + self.interval

    async def beforeExecution(self, interface: DispatcherInterface[Dispatchable]):
        event = interface.event
        target_key: T_SourceKey = await self.fetch_target_key(event)
        current_time: datetime = datetime.now()
        next_exec_time: datetime = self.source.get(target_key, current_time)
        delta: timedelta = next_exec_time - current_time
        satisfied: bool = delta <= timedelta(seconds=0)
        if not satisfied and self.stop_on_cooldown:
            param_dict: Dict[str, Any] = {}
            for name, anno, _ in self.override_signature:
                param_dict[name] = await interface.lookup_param(name, anno, None)
            res = self.override_condition(**param_dict)
            if not ((await res) if inspect.isawaitable(res) else res):
                raise ExecutionStop
        interface.local_storage[f"{__name__}:next_exec_time"] = next_exec_time
        interface.local_storage[f"{__name__}:delta"] = delta

    async def catch(self, interface: DispatcherInterface[Dispatchable]):
        annotation = interface.annotation
        next_exec_time: datetime = interface.local_storage[f"{__name__}:next_exec_time"]
        delta: timedelta = interface.local_storage[f"{__name__}:delta"]
        if NoneType in typing.get_args(annotation) and delta.total_seconds() <= 0:
            return Force(None)
        if generic_issubclass(datetime, annotation):
            return next_exec_time
        if generic_issubclass(timedelta, annotation):
            return delta
        if generic_issubclass(float, annotation):
            return delta.total_seconds()
        if generic_issubclass(int, annotation):
            return int(delta.total_seconds())

    async def afterDispatch(
        self,
        interface: DispatcherInterface[Dispatchable],
        exception: Optional[Exception],
        _: Optional[TracebackType],
    ):
        if not exception:
            event = interface.event
            await self.set(await self.fetch_target_key(event))

    if TYPE_CHECKING:

        @overload
        @contextlib.asynccontextmanager
        async def trigger(self, target: int) -> AsyncGenerator[Tuple[Optional[datetime], bool], None]:
            ...

        @overload
        @contextlib.asynccontextmanager
        async def trigger(self, target: int, type: Type[T_Time]) -> AsyncGenerator[Tuple[T_Time, bool], None]:
            ...

    @contextlib.asynccontextmanager
    async def trigger(
        self, target: T_SourceKey, type: Type[T_Time]
    ) -> AsyncGenerator[Tuple[T_Time, bool], None]:
        """触发冷却。

        Args:
            target (T_SourceKey): 目标的 “冷却哈希”
            type (Type[T_Time]): 需要返回的类型
        """
        try:
            yield await self.get(target, type)
        except Exception as e:
            raise e
        else:
            await self.set(target)

__init__(interval, source=None, override_condition=lambda: False, stop_on_cooldown=True) 🔗

初始化一个冷却时间

Parameters:

Name Type Description Default
interval Union[int, float, timedelta]

冷却时间, 单位为秒

required
source Union[MutableMapping[int, datetime], str, None]

冷却映射来源, 为字符串时从 ClassVar 查找.

None
override_condition ...) -> Union[bool, Awaitable[bool]]

超越冷却限制的条件.

lambda: False
stop_on_cooldown bool

是否在未到冷却时间时直接停止执行. Defaults to True.

True
Source code in src/graia/ariadne/util/cooldown.py
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
def __init__(
    self,
    interval: Union[int, float, timedelta],
    source: Union[MutableMapping[T_SourceKey, datetime], str, None] = None,
    override_condition: Callable[..., Union[bool, Awaitable[bool]]] = lambda: False,
    stop_on_cooldown: bool = True,
) -> None:
    """初始化一个冷却时间

    Args:
        interval (Union[int, float, timedelta]): 冷却时间, 单位为秒
        source (Union[MutableMapping[int, datetime], str, None], optional): 冷却映射来源, \
            为字符串时从 ClassVar 查找.
        override_condition ((...) -> Union[bool, Awaitable[bool]], optional): 超越冷却限制的条件.
        stop_on_cooldown (bool, optional): 是否在未到冷却时间时直接停止执行. Defaults to True.
    """
    self.interval = interval if isinstance(interval, timedelta) else timedelta(seconds=interval)
    self.stop_on_cooldown: bool = stop_on_cooldown
    self.override_condition: Callable[..., Union[bool, Awaitable[bool]]] = override_condition
    self.override_signature = argument_signature(self.override_condition)
    if isinstance(source, str):
        self.source = self.global_source.setdefault(source, {})
    else:
        self.source = source or {}

fetch_target_key(event) async 🔗

获取目标的键,以在 source 中获取对应的冷却信息

Parameters:

Name Type Description Default
event Dispatchable

当前事件

required

Returns:

Name Type Description
T_SourceKey T_SourceKey

目标的 “冷却哈希”

Source code in src/graia/ariadne/util/cooldown.py
74
75
76
77
78
79
80
81
82
83
84
85
async def fetch_target_key(self, event: Dispatchable) -> T_SourceKey:
    """获取目标的键,以在 source 中获取对应的冷却信息

    Args:
        event (Dispatchable): 当前事件

    Returns:
        T_SourceKey: 目标的 “冷却哈希”
    """
    if not isinstance(event, MessageEvent):
        raise ExecutionStop
    return cast(T_SourceKey, event.sender.id)

get(target, type) async 🔗

获取目标的冷却信息

Parameters:

Name Type Description Default
target T_SourceKey

目标的 “冷却哈希”

required
type Type[T_Time]

需要返回的类型

required

Returns:

Type Description
Tuple[T_Time, bool]

Tuple[T_Time, bool]: 第一个值是剩余的冷却时间 (或下一次可执行时间), 第二个值是冷却是否完成 如果 type 传入的是 Optional[XXX] 则第一个值可以是 None

Source code in src/graia/ariadne/util/cooldown.py
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
async def get(self, target: T_SourceKey, type: Type[T_Time]) -> Tuple[T_Time, bool]:
    """获取目标的冷却信息

    Args:
        target (T_SourceKey): 目标的 “冷却哈希”
        type (Type[T_Time]): 需要返回的类型

    Returns:
        Tuple[T_Time, bool]: 第一个值是剩余的冷却时间 (或下一次可执行时间),\
            第二个值是冷却是否完成 \
            如果 type 传入的是 Optional[XXX] 则第一个值可以是 None
    """
    current_time: datetime = datetime.now()
    next_exec_time: datetime = self.source.get(target, current_time)
    delta: timedelta = next_exec_time - current_time
    satisfied: bool = delta < timedelta(seconds=0)
    if NoneType in typing.get_args(type) and delta.total_seconds() <= 0:
        result = None, satisfied
    elif generic_issubclass(datetime, type):
        result = next_exec_time, satisfied
    elif generic_issubclass(timedelta, type):
        result = delta, satisfied
    elif generic_issubclass(float, type):
        result = delta.total_seconds(), satisfied
    elif generic_issubclass(int, type):
        result = int(delta.total_seconds()), satisfied
    else:
        result = None, satisfied
    return cast(Tuple[T_Time, bool], result)

set(target) async 🔗

更新目标的冷却信息

Parameters:

Name Type Description Default
target T_SourceKey

目标的 “冷却哈希”

required
Source code in src/graia/ariadne/util/cooldown.py
117
118
119
120
121
122
123
async def set(self, target: T_SourceKey) -> None:
    """更新目标的冷却信息

    Args:
        target (T_SourceKey): 目标的 “冷却哈希”
    """
    self.source[target] = datetime.now() + self.interval

trigger(target, type) async 🔗

trigger(
    target: int,
) -> AsyncGenerator[Tuple[Optional[datetime], bool], None]
trigger(
    target: int, type: Type[T_Time]
) -> AsyncGenerator[Tuple[T_Time, bool], None]

触发冷却。

Parameters:

Name Type Description Default
target T_SourceKey

目标的 “冷却哈希”

required
type Type[T_Time]

需要返回的类型

required
Source code in src/graia/ariadne/util/cooldown.py
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
@contextlib.asynccontextmanager
async def trigger(
    self, target: T_SourceKey, type: Type[T_Time]
) -> AsyncGenerator[Tuple[T_Time, bool], None]:
    """触发冷却。

    Args:
        target (T_SourceKey): 目标的 “冷却哈希”
        type (Type[T_Time]): 需要返回的类型
    """
    try:
        yield await self.get(target, type)
    except Exception as e:
        raise e
    else:
        await self.set(target)