aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/python/Pygments/py3/pygments/sphinxext.py
blob: 614fcef1737cec6e4e3d844da63842c961f87e9d (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
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
""" 
    pygments.sphinxext 
    ~~~~~~~~~~~~~~~~~~ 
 
    Sphinx extension to generate automatic documentation of lexers, 
    formatters and filters. 
 
    :copyright: Copyright 2006-2021 by the Pygments team, see AUTHORS.
    :license: BSD, see LICENSE for details. 
""" 
 
import sys 
 
from docutils import nodes 
from docutils.statemachine import ViewList 
from docutils.parsers.rst import Directive
from sphinx.util.nodes import nested_parse_with_titles 
 
 
MODULEDOC = ''' 
.. module:: %s 
 
%s 
%s 
''' 
 
LEXERDOC = ''' 
.. class:: %s 
 
    :Short names: %s 
    :Filenames:   %s 
    :MIME types:  %s 
 
    %s 
 
''' 
 
FMTERDOC = ''' 
.. class:: %s 
 
    :Short names: %s 
    :Filenames: %s 
 
    %s 
 
''' 
 
FILTERDOC = ''' 
.. class:: %s 
 
    :Name: %s 
 
    %s 
 
''' 
 

class PygmentsDoc(Directive): 
    """ 
    A directive to collect all lexers/formatters/filters and generate 
    autoclass directives for them. 
    """ 
    has_content = False 
    required_arguments = 1 
    optional_arguments = 0 
    final_argument_whitespace = False 
    option_spec = {} 
 
    def run(self): 
        self.filenames = set() 
        if self.arguments[0] == 'lexers': 
            out = self.document_lexers() 
        elif self.arguments[0] == 'formatters': 
            out = self.document_formatters() 
        elif self.arguments[0] == 'filters': 
            out = self.document_filters() 
        else: 
            raise Exception('invalid argument for "pygmentsdoc" directive') 
        node = nodes.compound() 
        vl = ViewList(out.split('\n'), source='') 
        nested_parse_with_titles(self.state, vl, node) 
        for fn in self.filenames: 
            self.state.document.settings.record_dependencies.add(fn) 
        return node.children 
 
    def document_lexers(self): 
        from pygments.lexers._mapping import LEXERS 
        out = [] 
        modules = {} 
        moduledocstrings = {} 
        for classname, data in sorted(LEXERS.items(), key=lambda x: x[0]): 
            module = data[0] 
            mod = __import__(module, None, None, [classname]) 
            self.filenames.add(mod.__file__) 
            cls = getattr(mod, classname) 
            if not cls.__doc__: 
                print("Warning: %s does not have a docstring." % classname) 
            docstring = cls.__doc__ 
            if isinstance(docstring, bytes): 
                docstring = docstring.decode('utf8') 
            modules.setdefault(module, []).append(( 
                classname, 
                ', '.join(data[2]) or 'None', 
                ', '.join(data[3]).replace('*', '\\*').replace('_', '\\') or 'None', 
                ', '.join(data[4]) or 'None', 
                docstring)) 
            if module not in moduledocstrings: 
                moddoc = mod.__doc__ 
                if isinstance(moddoc, bytes): 
                    moddoc = moddoc.decode('utf8') 
                moduledocstrings[module] = moddoc 
 
        for module, lexers in sorted(modules.items(), key=lambda x: x[0]): 
            if moduledocstrings[module] is None: 
                raise Exception("Missing docstring for %s" % (module,)) 
            heading = moduledocstrings[module].splitlines()[4].strip().rstrip('.') 
            out.append(MODULEDOC % (module, heading, '-'*len(heading))) 
            for data in lexers: 
                out.append(LEXERDOC % data) 
 
        return ''.join(out) 
 
    def document_formatters(self): 
        from pygments.formatters import FORMATTERS 
 
        out = [] 
        for classname, data in sorted(FORMATTERS.items(), key=lambda x: x[0]): 
            module = data[0] 
            mod = __import__(module, None, None, [classname]) 
            self.filenames.add(mod.__file__) 
            cls = getattr(mod, classname) 
            docstring = cls.__doc__ 
            if isinstance(docstring, bytes): 
                docstring = docstring.decode('utf8') 
            heading = cls.__name__ 
            out.append(FMTERDOC % (heading, ', '.join(data[2]) or 'None', 
                                   ', '.join(data[3]).replace('*', '\\*') or 'None', 
                                   docstring)) 
        return ''.join(out) 
 
    def document_filters(self): 
        from pygments.filters import FILTERS 
 
        out = [] 
        for name, cls in FILTERS.items(): 
            self.filenames.add(sys.modules[cls.__module__].__file__) 
            docstring = cls.__doc__ 
            if isinstance(docstring, bytes): 
                docstring = docstring.decode('utf8') 
            out.append(FILTERDOC % (cls.__name__, name, docstring)) 
        return ''.join(out) 
 
 
def setup(app): 
    app.add_directive('pygmentsdoc', PygmentsDoc)