Helper class, used by [[MarkdownIt#core]], [[MarkdownIt#block]] and
[[MarkdownIt#inline]] to manage sequences of functions (rules):
- keep rules in defined order
- assign the name to each rule
- enable/disable rules
- add/replace rules
- allow assign rules to additional named chains (in the same)
- caching lists of active rules
You will not need use this class directly until write plugins. For simple
rules control use [[MarkdownIt.disable]], [[MarkdownIt.enable]] and
[[MarkdownIt.use]]. """ from __future__ import annotations
from collections.abc import Iterable from dataclasses import dataclass, field from typing import TYPE_CHECKING, Generic, TypedDict, TypeVar import warnings
from markdown_it._compat import DATACLASS_KWARGS
from .utils import EnvType
if TYPE_CHECKING: from markdown_it import MarkdownIt
@property def srcCharCode(self) -> tuple[int, ...]:
warnings.warn( "StateBase.srcCharCode is deprecated. Use StateBase.src instead.",
DeprecationWarning,
stacklevel=2,
) if self._srcCharCode isNone:
self._srcCharCode = tuple(ord(c) for c in self._src) return self._srcCharCode
class RuleOptionsType(TypedDict, total=False):
alt: list[str]
RuleFuncTv = TypeVar("RuleFuncTv") """A rule function, whose signature is dependent on the state type."""
class Ruler(Generic[RuleFuncTv]): def __init__(self) -> None: # List of added rules.
self.__rules__: list[Rule[RuleFuncTv]] = [] # Cached rule chains. # First level - chain name, '' for default. # Second level - diginal anchor for fast filtering by charcodes.
self.__cache__: dict[str, list[RuleFuncTv]] | None = None
def __find__(self, name: str) -> int: """Find rule index by name""" for i, rule in enumerate(self.__rules__): if rule.name == name: return i return -1
def __compile__(self) -> None: """Build rules lookup cache"""
chains = {""} # collect unique names for rule in self.__rules__: ifnot rule.enabled: continue for name in rule.alt:
chains.add(name)
self.__cache__ = {} for chain in chains:
self.__cache__[chain] = [] for rule in self.__rules__: ifnot rule.enabled: continue if chain and (chain notin rule.alt): continue
self.__cache__[chain].append(rule.fn)
def at(
self, ruleName: str, fn: RuleFuncTv, options: RuleOptionsType | None = None
) -> None: """Replace rule by name with new function & options.
:param ruleName: rule name to replace.
:param fn: new rule function.
:param options: new rule options (not mandatory).
:raises: KeyError if name not found """
index = self.__find__(ruleName)
options = options or {} if index == -1: raise KeyError(f"Parser rule not found: {ruleName}")
self.__rules__[index].fn = fn
self.__rules__[index].alt = options.get("alt", [])
self.__cache__ = None
def before(
self,
beforeName: str,
ruleName: str,
fn: RuleFuncTv,
options: RuleOptionsType | None = None,
) -> None: """Add new rule to chain before one with given name.
:param beforeName: new rule will be added before this one.
:param ruleName: new rule will be added before this one.
:param fn: new rule function.
:param options: new rule options (not mandatory).
:raises: KeyError if name not found """
index = self.__find__(beforeName)
options = options or {} if index == -1: raise KeyError(f"Parser rule not found: {beforeName}")
self.__rules__.insert(
index, Rule[RuleFuncTv](ruleName, True, fn, options.get("alt", []))
)
self.__cache__ = None
def after(
self,
afterName: str,
ruleName: str,
fn: RuleFuncTv,
options: RuleOptionsType | None = None,
) -> None: """Add new rule to chain after one with given name.
:param afterName: new rule will be added after this one.
:param ruleName: new rule will be added after this one.
:param fn: new rule function.
:param options: new rule options (not mandatory).
:raises: KeyError if name not found """
index = self.__find__(afterName)
options = options or {} if index == -1: raise KeyError(f"Parser rule not found: {afterName}")
self.__rules__.insert(
index + 1, Rule[RuleFuncTv](ruleName, True, fn, options.get("alt", []))
)
self.__cache__ = None
def push(
self, ruleName: str, fn: RuleFuncTv, options: RuleOptionsType | None = None
) -> None: """Push new rule to the end of chain.
:param ruleName: new rule will be added to the end of chain.
:param fn: new rule function.
:param options: new rule options (not mandatory).
def enable(
self, names: str | Iterable[str], ignoreInvalid: bool = False
) -> list[str]: """Enable rules with given names.
:param names: name or list of rule names to enable.
:param ignoreInvalid: ignore errors when rule not found
:raises: KeyError if name not found andnot ignoreInvalid
:return: list of found rule names """ if isinstance(names, str):
names = [names]
result: list[str] = [] for name in names:
idx = self.__find__(name) if (idx < 0) and ignoreInvalid: continue if (idx < 0) andnot ignoreInvalid: raise KeyError(f"Rules manager: invalid rule name {name}")
self.__rules__[idx].enabled = True
result.append(name)
self.__cache__ = None return result
def enableOnly(
self, names: str | Iterable[str], ignoreInvalid: bool = False
) -> list[str]: """Enable rules with given names, and disable everything else.
:param names: name or list of rule names to enable.
:param ignoreInvalid: ignore errors when rule not found
:raises: KeyError if name not found andnot ignoreInvalid
:return: list of found rule names """ if isinstance(names, str):
names = [names] for rule in self.__rules__:
rule.enabled = False return self.enable(names, ignoreInvalid)
def disable(
self, names: str | Iterable[str], ignoreInvalid: bool = False
) -> list[str]: """Disable rules with given names.
:param names: name or list of rule names to enable.
:param ignoreInvalid: ignore errors when rule not found
:raises: KeyError if name not found andnot ignoreInvalid
:return: list of found rule names """ if isinstance(names, str):
names = [names]
result = [] for name in names:
idx = self.__find__(name) if (idx < 0) and ignoreInvalid: continue if (idx < 0) andnot ignoreInvalid: raise KeyError(f"Rules manager: invalid rule name {name}")
self.__rules__[idx].enabled = False
result.append(name)
self.__cache__ = None return result
def getRules(self, chainName: str = "") -> list[RuleFuncTv]: """Return array of active functions (rules) for given chain name.
It analyzes rules configuration, compiles caches ifnot exists and returns result.
Default chain name is `''` (empty string). It can't be skipped.
That's done intentionally, to keep signature monomorphic for high speed.
""" if self.__cache__ isNone:
self.__compile__() assert self.__cache__ isnotNone # Chain can be empty, if rules disabled. But we still have to return Array. return self.__cache__.get(chainName, []) or []
def get_all_rules(self) -> list[str]: """Return all available rule names.""" return [r.name for r in self.__rules__]
def get_active_rules(self) -> list[str]: """Return the active rule names.""" return [r.name for r in self.__rules__ if r.enabled]
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.