跳转至

base

Ariadne 基础的 parser, 包括 DetectPrefix 与 DetectSuffix

ContainKeyword 🔗

Bases: ChainDecorator

消息中含有指定关键字

Source code in src/graia/ariadne/message/parser/base.py
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
class ContainKeyword(ChainDecorator):
    """消息中含有指定关键字"""

    def __init__(self, keyword: str) -> None:
        """初始化

        Args:
            keyword (str): 关键字
        """
        self.keyword: str = keyword

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        if self.keyword not in chain:
            raise ExecutionStop
        return chain

__init__(keyword) 🔗

初始化

Parameters:

Name Type Description Default
keyword str

关键字

required
Source code in src/graia/ariadne/message/parser/base.py
132
133
134
135
136
137
138
def __init__(self, keyword: str) -> None:
    """初始化

    Args:
        keyword (str): 关键字
    """
    self.keyword: str = keyword

DetectPrefix 🔗

Bases: ChainDecorator

前缀检测器

Source code in src/graia/ariadne/message/parser/base.py
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
class DetectPrefix(ChainDecorator):
    """前缀检测器"""

    def __init__(self, prefix: Union[str, Iterable[str]]) -> None:
        """初始化前缀检测器.

        Args:
            prefix (Union[str, Iterable[str]]): 要匹配的前缀
        """
        self.prefix: List[str] = [prefix] if isinstance(prefix, str) else list(prefix)

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        for prefix in self.prefix:
            if chain.startswith(prefix):
                return chain.removeprefix(prefix).removeprefix(" ")

        raise ExecutionStop

__init__(prefix) 🔗

初始化前缀检测器.

Parameters:

Name Type Description Default
prefix Union[str, Iterable[str]]

要匹配的前缀

required
Source code in src/graia/ariadne/message/parser/base.py
42
43
44
45
46
47
48
def __init__(self, prefix: Union[str, Iterable[str]]) -> None:
    """初始化前缀检测器.

    Args:
        prefix (Union[str, Iterable[str]]): 要匹配的前缀
    """
    self.prefix: List[str] = [prefix] if isinstance(prefix, str) else list(prefix)

DetectSuffix 🔗

Bases: ChainDecorator

后缀检测器

Source code in src/graia/ariadne/message/parser/base.py
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
class DetectSuffix(ChainDecorator):
    """后缀检测器"""

    def __init__(self, suffix: Union[str, Iterable[str]]) -> None:
        """初始化后缀检测器.

        Args:
            suffix (Union[str, Iterable[str]]): 要匹配的后缀
        """
        self.suffix: List[str] = [suffix] if isinstance(suffix, str) else list(suffix)

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        for suffix in self.suffix:
            if chain.endswith(suffix):
                return chain.removesuffix(suffix).removesuffix(" ")
        raise ExecutionStop

__init__(suffix) 🔗

初始化后缀检测器.

Parameters:

Name Type Description Default
suffix Union[str, Iterable[str]]

要匹配的后缀

required
Source code in src/graia/ariadne/message/parser/base.py
61
62
63
64
65
66
67
def __init__(self, suffix: Union[str, Iterable[str]]) -> None:
    """初始化后缀检测器.

    Args:
        suffix (Union[str, Iterable[str]]): 要匹配的后缀
    """
    self.suffix: List[str] = [suffix] if isinstance(suffix, str) else list(suffix)

FuzzyDispatcher 🔗

Bases: BaseDispatcher

Source code in src/graia/ariadne/message/parser/base.py
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
class FuzzyDispatcher(BaseDispatcher):
    scope_map: ClassVar[DefaultDict[str, List[str]]] = defaultdict(list)
    event_ref: ClassVar["Dict[int, Dict[str, Tuple[str, float]]]"] = {}

    def __init__(self, template: str, min_rate: float = 0.6, scope: str = "") -> None:
        """初始化

        Args:
            template (str): 模板字符串
            min_rate (float): 最小匹配阈值
            scope (str): 作用域
        """
        self.template: str = template
        self.min_rate: float = min_rate
        self.scope: str = scope
        self.scope_map[scope].append(template)

    async def beforeExecution(self, interface: DispatcherInterface):
        event = interface.event
        if id(event) not in self.event_ref:
            chain: MessageChain = await interface.lookup_param("message_chain", MessageChain, None)
            text_frags: List[str] = []
            for element in chain:
                if isinstance(element, Plain):
                    text_frags.append(element.text)
                else:
                    text_frags.append(str(element))
            text = "".join(text_frags)
            matcher = difflib.SequenceMatcher()
            matcher.set_seq2(text)
            rate_calc = self.event_ref[id(event)] = {}
            weakref.finalize(event, lambda d: self.event_ref.pop(d), id(event))
            for scope, templates in self.scope_map.items():
                max_match: float = 0.0
                for template in templates:
                    matcher.set_seq1(template)
                    if matcher.real_quick_ratio() < max_match:
                        continue
                    if matcher.quick_ratio() < max_match:
                        continue
                    if matcher.ratio() < max_match:
                        continue
                    rate_calc[scope] = (template, matcher.ratio())
                    max_match = matcher.ratio()
        win_template, win_rate = self.event_ref[id(event)].get(self.scope, (self.template, 0.0))
        if win_template != self.template or win_rate < self.min_rate:
            raise ExecutionStop

    async def catch(self, i: DispatcherInterface) -> Optional[float]:
        event = i.event
        _, rate = self.event_ref[id(event)].get(self.scope, (self.template, 0.0))
        if generic_issubclass(float, i.annotation) and "rate" in i.name:
            return rate

__init__(template, min_rate=0.6, scope='') 🔗

初始化

Parameters:

Name Type Description Default
template str

模板字符串

required
min_rate float

最小匹配阈值

0.6
scope str

作用域

''
Source code in src/graia/ariadne/message/parser/base.py
330
331
332
333
334
335
336
337
338
339
340
341
def __init__(self, template: str, min_rate: float = 0.6, scope: str = "") -> None:
    """初始化

    Args:
        template (str): 模板字符串
        min_rate (float): 最小匹配阈值
        scope (str): 作用域
    """
    self.template: str = template
    self.min_rate: float = min_rate
    self.scope: str = scope
    self.scope_map[scope].append(template)

FuzzyMatch 🔗

Bases: ChainDecorator

模糊匹配

Warning

我们更推荐使用 FuzzyDispatcher 来进行模糊匹配操作, 因为其具有上下文匹配数量限制.

Source code in src/graia/ariadne/message/parser/base.py
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
class FuzzyMatch(ChainDecorator):
    """模糊匹配

    Warning:
        我们更推荐使用 FuzzyDispatcher 来进行模糊匹配操作, 因为其具有上下文匹配数量限制.
    """

    def __init__(self, template: str, min_rate: float = 0.6) -> None:
        """初始化

        Args:
            template (str): 模板字符串
            min_rate (float): 最小匹配阈值
        """
        self.template: str = template
        self.min_rate: float = min_rate

    def match(self, chain: MessageChain):
        """匹配消息链"""
        text_frags: List[str] = []
        for element in chain:
            if isinstance(element, Plain):
                text_frags.append(element.text)
            else:
                text_frags.append(str(element))
        text = "".join(text_frags)
        matcher = difflib.SequenceMatcher(a=text, b=self.template)
        # return false when **any** ratio calc falls undef the rate
        if matcher.real_quick_ratio() < self.min_rate:
            return False
        if matcher.quick_ratio() < self.min_rate:
            return False
        return matcher.ratio() >= self.min_rate

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        if not self.match(chain):
            raise ExecutionStop
        return chain

__init__(template, min_rate=0.6) 🔗

初始化

Parameters:

Name Type Description Default
template str

模板字符串

required
min_rate float

最小匹配阈值

0.6
Source code in src/graia/ariadne/message/parser/base.py
293
294
295
296
297
298
299
300
301
def __init__(self, template: str, min_rate: float = 0.6) -> None:
    """初始化

    Args:
        template (str): 模板字符串
        min_rate (float): 最小匹配阈值
    """
    self.template: str = template
    self.min_rate: float = min_rate

match(chain) 🔗

匹配消息链

Source code in src/graia/ariadne/message/parser/base.py
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
def match(self, chain: MessageChain):
    """匹配消息链"""
    text_frags: List[str] = []
    for element in chain:
        if isinstance(element, Plain):
            text_frags.append(element.text)
        else:
            text_frags.append(str(element))
    text = "".join(text_frags)
    matcher = difflib.SequenceMatcher(a=text, b=self.template)
    # return false when **any** ratio calc falls undef the rate
    if matcher.real_quick_ratio() < self.min_rate:
        return False
    if matcher.quick_ratio() < self.min_rate:
        return False
    return matcher.ratio() >= self.min_rate

MatchContent 🔗

Bases: ChainDecorator

匹配字符串 / 消息链

Source code in src/graia/ariadne/message/parser/base.py
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
class MatchContent(ChainDecorator):
    """匹配字符串 / 消息链"""

    def __init__(self, content: Union[str, MessageChain]) -> None:
        """初始化

        Args:
            content (Union[str, MessageChain]): 匹配内容
        """
        self.content: Union[str, MessageChain] = content

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        if isinstance(self.content, str) and str(chain) != self.content:
            raise ExecutionStop
        if isinstance(self.content, MessageChain) and chain != self.content:
            raise ExecutionStop
        return chain

__init__(content) 🔗

初始化

Parameters:

Name Type Description Default
content Union[str, MessageChain]

匹配内容

required
Source code in src/graia/ariadne/message/parser/base.py
149
150
151
152
153
154
155
def __init__(self, content: Union[str, MessageChain]) -> None:
    """初始化

    Args:
        content (Union[str, MessageChain]): 匹配内容
    """
    self.content: Union[str, MessageChain] = content

MatchRegex 🔗

Bases: ChainDecorator, BaseDispatcher

匹配正则表达式

Source code in src/graia/ariadne/message/parser/base.py
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
195
196
class MatchRegex(ChainDecorator, BaseDispatcher):
    """匹配正则表达式"""

    def __init__(self, regex: str, flags: re.RegexFlag = re.RegexFlag(0), full: bool = True) -> None:
        """初始化匹配正则表达式.

        Args:
            regex (str): 正则表达式
            flags (re.RegexFlag): 正则表达式标志
            full (bool): 是否要求完全匹配, 默认为 True.
        """
        self.regex: str = regex
        self.flags: re.RegexFlag = flags
        self.pattern = re.compile(self.regex, self.flags)
        self.match_func = self.pattern.fullmatch if full else self.pattern.match

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        if not self.match_func(str(chain)):
            raise ExecutionStop
        return chain

    async def beforeExecution(self, interface: DispatcherInterface[MessageEvent]):
        _mapping_str, _map = interface.event.message_chain._to_mapping_str()
        if res := self.match_func(_mapping_str):
            interface.local_storage["__parser_regex_match_obj__"] = res
            interface.local_storage["__parser_regex_match_map__"] = _map
        else:
            raise ExecutionStop

    async def catch(self, interface: DispatcherInterface[MessageEvent]):
        if interface.annotation is re.Match:
            return interface.local_storage["__parser_regex_match_obj__"]

__init__(regex, flags=re.RegexFlag(0), full=True) 🔗

初始化匹配正则表达式.

Parameters:

Name Type Description Default
regex str

正则表达式

required
flags RegexFlag

正则表达式标志

RegexFlag(0)
full bool

是否要求完全匹配, 默认为 True.

True
Source code in src/graia/ariadne/message/parser/base.py
168
169
170
171
172
173
174
175
176
177
178
179
def __init__(self, regex: str, flags: re.RegexFlag = re.RegexFlag(0), full: bool = True) -> None:
    """初始化匹配正则表达式.

    Args:
        regex (str): 正则表达式
        flags (re.RegexFlag): 正则表达式标志
        full (bool): 是否要求完全匹配, 默认为 True.
    """
    self.regex: str = regex
    self.flags: re.RegexFlag = flags
    self.pattern = re.compile(self.regex, self.flags)
    self.match_func = self.pattern.fullmatch if full else self.pattern.match

MatchTemplate 🔗

Bases: ChainDecorator

模板匹配

Source code in src/graia/ariadne/message/parser/base.py
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
class MatchTemplate(ChainDecorator):
    """模板匹配"""

    def __init__(self, template: List[Union[Type[Element], Element, str]]) -> None:
        """初始化

        Args:
            template (List[Union[Type[Element], Element]]): \
                匹配模板,可以为 `Element` 类或其 `Union`, `str`, `Plain` 实例
        """
        self.template: List[Union[Tuple[Type[Element], ...], Element, str]] = []
        for element in template:
            if element is Plain:
                element = "*"
            if isinstance(element, type):
                self.template.append((element,))
            elif get_origin(element) in Unions:  # Union
                assert Plain not in get_args(element), "Leaving Plain here leads to ambiguity"  # TODO
                self.template.append(get_args(element))
            elif isinstance(element, Element) and not isinstance(element, Plain):
                self.template.append(element)
            else:
                element = (
                    re.escape(element.text)
                    if isinstance(element, Plain)
                    else fnmatch.translate(element)[:-2]  # truncating the ending \Z
                )
                if self.template and isinstance(self.template[-1], str):
                    self.template[-1] += element
                else:
                    self.template.append(element)

    def match(self, chain: MessageChain):
        """匹配消息链"""
        chain = chain.as_sendable()
        if len(self.template) != len(chain):
            return False
        for element, template in zip(chain, self.template):
            if isinstance(template, tuple) and not isinstance(element, template):
                return False
            elif isinstance(template, Element) and element != template:
                return False
            elif isinstance(template, str):
                if not isinstance(element, Plain) or not re.match(template, element.text):
                    return False
        return True

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        if not self.match(chain):
            raise ExecutionStop
        return chain

__init__(template) 🔗

初始化

Parameters:

Name Type Description Default
template List[Union[Type[Element], Element]]

匹配模板,可以为 Element 类或其 Union, str, Plain 实例

required
Source code in src/graia/ariadne/message/parser/base.py
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
def __init__(self, template: List[Union[Type[Element], Element, str]]) -> None:
    """初始化

    Args:
        template (List[Union[Type[Element], Element]]): \
            匹配模板,可以为 `Element` 类或其 `Union`, `str`, `Plain` 实例
    """
    self.template: List[Union[Tuple[Type[Element], ...], Element, str]] = []
    for element in template:
        if element is Plain:
            element = "*"
        if isinstance(element, type):
            self.template.append((element,))
        elif get_origin(element) in Unions:  # Union
            assert Plain not in get_args(element), "Leaving Plain here leads to ambiguity"  # TODO
            self.template.append(get_args(element))
        elif isinstance(element, Element) and not isinstance(element, Plain):
            self.template.append(element)
        else:
            element = (
                re.escape(element.text)
                if isinstance(element, Plain)
                else fnmatch.translate(element)[:-2]  # truncating the ending \Z
            )
            if self.template and isinstance(self.template[-1], str):
                self.template[-1] += element
            else:
                self.template.append(element)

match(chain) 🔗

匹配消息链

Source code in src/graia/ariadne/message/parser/base.py
265
266
267
268
269
270
271
272
273
274
275
276
277
278
def match(self, chain: MessageChain):
    """匹配消息链"""
    chain = chain.as_sendable()
    if len(self.template) != len(chain):
        return False
    for element, template in zip(chain, self.template):
        if isinstance(template, tuple) and not isinstance(element, template):
            return False
        elif isinstance(template, Element) and element != template:
            return False
        elif isinstance(template, str):
            if not isinstance(element, Plain) or not re.match(template, element.text):
                return False
    return True

Mention 🔗

Bases: ChainDecorator

At 或提到指定账号/名称

Source code in src/graia/ariadne/message/parser/base.py
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
class Mention(ChainDecorator):
    """At 或提到指定账号/名称"""

    def __init__(self, target: Union[int, str]) -> None:
        """
        Args:
            target (Union[int, str]): 要提到的账号或者名称, \
            如果是 int 则是账号, 如果是 str 则是名称
        """
        self.person: Union[int, str] = target

    async def __call__(self, chain: MessageChain, _) -> Optional[MessageChain]:
        first: Element = chain[0]
        if (
            chain
            and isinstance(first, Plain)
            and isinstance(self.person, str)
            and str(first).startswith(self.person)
        ):
            return chain.removeprefix(self.person).removeprefix(" ")
        if isinstance(first, At) and isinstance(self.person, int) and first.target == self.person:
            return MessageChain(chain.__root__[1:], inline=True).removeprefix(" ")

        raise ExecutionStop

__init__(target) 🔗

Parameters:

Name Type Description Default
target Union[int, str]

要提到的账号或者名称, 如果是 int 则是账号, 如果是 str 则是名称

required
Source code in src/graia/ariadne/message/parser/base.py
106
107
108
109
110
111
112
def __init__(self, target: Union[int, str]) -> None:
    """
    Args:
        target (Union[int, str]): 要提到的账号或者名称, \
        如果是 int 则是账号, 如果是 str 则是名称
    """
    self.person: Union[int, str] = target

MentionMe 🔗

Bases: ChainDecorator

At 账号或者提到账号群昵称

Source code in src/graia/ariadne/message/parser/base.py
 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
class MentionMe(ChainDecorator):
    """At 账号或者提到账号群昵称"""

    def __init__(self, name: Union[bool, str] = True) -> None:
        """
        Args:
            name (Union[bool, str]): 是否提取昵称, 如果为 True, 则自动提取昵称, \
            如果为 False 则禁用昵称, 为 str 则将参数作为昵称
        """
        self.name = name

    async def __call__(self, chain: MessageChain, interface: DispatcherInterface) -> Optional[MessageChain]:
        ariadne = Ariadne.current()
        name: Optional[str] = self.name if isinstance(self.name, str) else None
        if self.name is True:
            if isinstance(interface.event, GroupMessage):
                name = (await ariadne.get_member(interface.event.sender.group, ariadne.account)).name
            else:
                name = (await ariadne.get_bot_profile()).nickname
        first: Element = chain[0]
        if isinstance(name, str) and isinstance(first, Plain) and str(first).startswith(name):
            return chain.removeprefix(name).removeprefix(" ")
        if isinstance(first, At) and first.target == ariadne.account:
            return MessageChain(chain.__root__[1:], inline=True).removeprefix(" ")
        raise ExecutionStop

__init__(name=True) 🔗

Parameters:

Name Type Description Default
name Union[bool, str]

是否提取昵称, 如果为 True, 则自动提取昵称, 如果为 False 则禁用昵称, 为 str 则将参数作为昵称

True
Source code in src/graia/ariadne/message/parser/base.py
79
80
81
82
83
84
85
def __init__(self, name: Union[bool, str] = True) -> None:
    """
    Args:
        name (Union[bool, str]): 是否提取昵称, 如果为 True, 则自动提取昵称, \
        如果为 False 则禁用昵称, 为 str 则将参数作为昵称
    """
    self.name = name

RegexGroup 🔗

Bases: Decorator

正则表达式组的标志 以 Annotated[MessageChain, RegexGroup("xxx")] 的形式使用, 或者作为 Decorator 使用.

Source code in src/graia/ariadne/message/parser/base.py
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
class RegexGroup(Decorator):
    """正则表达式组的标志
    以 `Annotated[MessageChain, RegexGroup("xxx")]` 的形式使用,
    或者作为 Decorator 使用.
    """

    def __init__(self, target: Union[int, str]) -> None:
        """初始化

        Args:
            target (Union[int, str]): 目标的组名或序号
        """
        self.assign_target = target

    async def __call__(self, _, interface: DispatcherInterface[MessageEvent]):
        _res: re.Match = interface.local_storage["__parser_regex_match_obj__"]
        match_group: Tuple[str] = _res.groups()
        match_group_dict: Dict[str, str] = _res.groupdict()
        origin: Optional[str] = None
        if isinstance(self.assign_target, str) and self.assign_target in match_group_dict:
            origin = match_group_dict[self.assign_target]
        elif isinstance(self.assign_target, int) and self.assign_target < len(match_group):
            origin = match_group[self.assign_target]

        return (
            MessageChain._from_mapping_string(origin, interface.local_storage["__parser_regex_match_map__"])
            if origin is not None
            else None
        )

    async def target(self, interface: DecoratorInterface):
        return self("", interface.dispatcher_interface)

__init__(target) 🔗

初始化

Parameters:

Name Type Description Default
target Union[int, str]

目标的组名或序号

required
Source code in src/graia/ariadne/message/parser/base.py
205
206
207
208
209
210
211
def __init__(self, target: Union[int, str]) -> None:
    """初始化

    Args:
        target (Union[int, str]): 目标的组名或序号
    """
    self.assign_target = target