gobject-introspection 2.2.0-x86-mingw32 → 2.2.1-x86-mingw32
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/Rakefile +7 -7
- data/ext/gobject-introspection/rb-gi-argument.c +28 -3
- data/ext/gobject-introspection/rb-gi-constructor-info.c +6 -1
- data/ext/gobject-introspection/rb-gi-function-info.c +29 -4
- data/ext/gobject-introspection/rb-gi-loader.c +19 -3
- data/ext/gobject-introspection/rb-gi-struct-info.c +28 -16
- data/lib/2.0/gobject_introspection.so +0 -0
- data/lib/gobject-introspection/callable-info.rb +16 -7
- data/lib/gobject-introspection/loader.rb +47 -19
- data/test/test-object-info.rb +1 -1
- data/test/test-signal-info.rb +2 -2
- data/test/test-struct-info.rb +1 -1
- data/vendor/local/bin/g-ir-compiler.exe +0 -0
- data/vendor/local/bin/g-ir-generate.exe +0 -0
- data/vendor/local/bin/libgirepository-1.0-1.dll +0 -0
- data/vendor/local/include/gobject-introspection-1.0/giarginfo.h +30 -0
- data/vendor/local/include/gobject-introspection-1.0/gibaseinfo.h +31 -0
- data/vendor/local/include/gobject-introspection-1.0/gicallableinfo.h +36 -0
- data/vendor/local/include/gobject-introspection-1.0/giconstantinfo.h +12 -0
- data/vendor/local/include/gobject-introspection-1.0/gienuminfo.h +26 -0
- data/vendor/local/include/gobject-introspection-1.0/gifieldinfo.h +19 -0
- data/vendor/local/include/gobject-introspection-1.0/gifunctioninfo.h +23 -0
- data/vendor/local/include/gobject-introspection-1.0/giinterfaceinfo.h +38 -0
- data/vendor/local/include/gobject-introspection-1.0/giobjectinfo.h +72 -2
- data/vendor/local/include/gobject-introspection-1.0/gipropertyinfo.h +12 -0
- data/vendor/local/include/gobject-introspection-1.0/giregisteredtypeinfo.h +11 -0
- data/vendor/local/include/gobject-introspection-1.0/girepository.h +68 -8
- data/vendor/local/include/gobject-introspection-1.0/girffi.h +25 -1
- data/vendor/local/include/gobject-introspection-1.0/gisignalinfo.h +12 -0
- data/vendor/local/include/gobject-introspection-1.0/gistructinfo.h +24 -0
- data/vendor/local/include/gobject-introspection-1.0/gitypeinfo.h +31 -0
- data/vendor/local/include/gobject-introspection-1.0/gitypelib.h +25 -0
- data/vendor/local/include/gobject-introspection-1.0/gitypes.h +84 -13
- data/vendor/local/include/gobject-introspection-1.0/giunioninfo.h +27 -0
- data/vendor/local/include/gobject-introspection-1.0/giversionmacros.h +128 -0
- data/vendor/local/include/gobject-introspection-1.0/givfuncinfo.h +17 -0
- data/vendor/local/lib/girepository-1.0/GIRepository-2.0.typelib +0 -0
- data/vendor/local/lib/girepository-1.0/GLib-2.0.typelib +0 -0
- data/vendor/local/lib/girepository-1.0/GObject-2.0.typelib +0 -0
- data/vendor/local/lib/girepository-1.0/Gio-2.0.typelib +0 -0
- data/vendor/local/lib/girepository-1.0/libxml2-2.0.typelib +0 -0
- data/vendor/local/lib/girepository-1.0/win32-1.0.typelib +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/__init__.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/__init__.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/annotationmain.py +5 -4
- data/vendor/local/lib/gobject-introspection/giscanner/annotationmain.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/annotationmain.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/annotationparser.py +1865 -913
- data/vendor/local/lib/gobject-introspection/giscanner/annotationparser.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/annotationparser.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/ast.py +49 -16
- data/vendor/local/lib/gobject-introspection/giscanner/ast.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/ast.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/cachestore.py +10 -4
- data/vendor/local/lib/gobject-introspection/giscanner/cachestore.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/cachestore.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/ccompiler.py +202 -0
- data/vendor/local/lib/gobject-introspection/giscanner/ccompiler.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/ccompiler.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/codegen.py +32 -1
- data/vendor/local/lib/gobject-introspection/giscanner/codegen.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/codegen.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/__init__.py +1 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/__init__.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/__init__.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/counter.py +305 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/counter.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/counter.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/ordereddict.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/collections/ordereddict.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/docmain.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/docmain.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/C/callback.tmpl +4 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/C/field.tmpl +1 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/C/function.tmpl +8 -9
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/C/interface.tmpl +2 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/callback.tmpl +27 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/class.tmpl +17 -5
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/enum.tmpl +8 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/field.tmpl +9 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/function.tmpl +12 -13
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/interface.tmpl +17 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/property.tmpl +3 -4
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/signal.tmpl +10 -9
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Gjs/vfunc.tmpl +7 -7
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/callback.tmpl +27 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/class.tmpl +5 -4
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/field.tmpl +1 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/function.tmpl +8 -14
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/interface.tmpl +16 -0
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/property.tmpl +2 -3
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/signal.tmpl +6 -7
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/Python/vfunc.tmpl +7 -13
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/base.tmpl +10 -19
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/class.tmpl +24 -3
- data/vendor/local/lib/gobject-introspection/giscanner/doctemplates/namespace.tmpl +4 -7
- data/vendor/local/lib/gobject-introspection/giscanner/docwriter.py +375 -61
- data/vendor/local/lib/gobject-introspection/giscanner/docwriter.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/docwriter.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/dumper.py +43 -75
- data/vendor/local/lib/gobject-introspection/giscanner/dumper.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/dumper.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/gdumpparser.py +1 -20
- data/vendor/local/lib/gobject-introspection/giscanner/gdumpparser.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/gdumpparser.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/girparser.py +33 -15
- data/vendor/local/lib/gobject-introspection/giscanner/girparser.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/girparser.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/girwriter.py +51 -22
- data/vendor/local/lib/gobject-introspection/giscanner/girwriter.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/girwriter.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/introspectablepass.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/introspectablepass.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/libtoolimporter.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/libtoolimporter.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/maintransformer.py +285 -254
- data/vendor/local/lib/gobject-introspection/giscanner/maintransformer.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/maintransformer.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/message.py +41 -25
- data/vendor/local/lib/gobject-introspection/giscanner/message.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/message.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/scannermain.py +67 -15
- data/vendor/local/lib/gobject-introspection/giscanner/scannermain.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/scannermain.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/sectionparser.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/sectionparser.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/shlibs.py +10 -6
- data/vendor/local/lib/gobject-introspection/giscanner/shlibs.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/shlibs.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/sourcescanner.py +14 -8
- data/vendor/local/lib/gobject-introspection/giscanner/sourcescanner.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/sourcescanner.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/testcodegen.py +16 -2
- data/vendor/local/lib/gobject-introspection/giscanner/testcodegen.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/testcodegen.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/transformer.py +150 -169
- data/vendor/local/lib/gobject-introspection/giscanner/transformer.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/transformer.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/utils.py +64 -3
- data/vendor/local/lib/gobject-introspection/giscanner/utils.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/utils.pyo +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/xmlwriter.py +5 -44
- data/vendor/local/lib/gobject-introspection/giscanner/xmlwriter.pyc +0 -0
- data/vendor/local/lib/gobject-introspection/giscanner/xmlwriter.pyo +0 -0
- data/vendor/local/lib/libgirepository-1.0.a +0 -0
- data/vendor/local/lib/libgirepository-1.0.dll.a +0 -0
- data/vendor/local/lib/libgirepository-1.0.la +1 -1
- data/vendor/local/lib/pkgconfig/gobject-introspection-1.0.pc +4 -4
- data/vendor/local/lib/pkgconfig/gobject-introspection-no-export-1.0.pc +5 -5
- data/vendor/local/share/gir-1.0/GIRepository-2.0.gir +352 -143
- data/vendor/local/share/gir-1.0/GLib-2.0.gir +6402 -3872
- data/vendor/local/share/gir-1.0/GModule-2.0.gir +42 -24
- data/vendor/local/share/gir-1.0/GObject-2.0.gir +1543 -887
- data/vendor/local/share/gir-1.0/Gio-2.0.gir +10859 -3705
- data/vendor/local/share/gobject-introspection-1.0/tests/annotation.c +10 -14
- data/vendor/local/share/gobject-introspection-1.0/tests/annotation.h +112 -9
- data/vendor/local/share/gobject-introspection-1.0/tests/drawable.c +2 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/drawable.h +11 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/everything.c +106 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/everything.h +107 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/foo.c +3 -3
- data/vendor/local/share/gobject-introspection-1.0/tests/foo.h +145 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/gimarshallingtests.c +166 -10
- data/vendor/local/share/gobject-introspection-1.0/tests/gimarshallingtests.h +790 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/gitestmacros.h +10 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/regress.c +79 -28
- data/vendor/local/share/gobject-introspection-1.0/tests/regress.h +462 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/utility.c +2 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/utility.h +7 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/warnlib.c +14 -0
- data/vendor/local/share/gobject-introspection-1.0/tests/warnlib.h +10 -0
- data/vendor/local/share/man/man1/g-ir-compiler.1 +1 -10
- metadata +38 -23
- data/lib/1.9/gobject_introspection.so +0 -0
- data/lib/2.1/gobject_introspection.so +0 -0
@@ -1,7 +1,9 @@
|
|
1
|
+
# -*- coding: utf-8 -*-
|
1
2
|
# -*- Mode: Python -*-
|
3
|
+
|
2
4
|
# GObject-Introspection - a framework for introspecting GObject libraries
|
3
5
|
# Copyright (C) 2008-2010 Johan Dahlin
|
4
|
-
# Copyright (C) 2012 Dieter Verfaillie <dieterv@optionexplicit.be>
|
6
|
+
# Copyright (C) 2012-2013 Dieter Verfaillie <dieterv@optionexplicit.be>
|
5
7
|
#
|
6
8
|
# This program is free software; you can redistribute it and/or
|
7
9
|
# modify it under the terms of the GNU General Public License
|
@@ -20,1004 +22,1402 @@
|
|
20
22
|
#
|
21
23
|
|
22
24
|
|
23
|
-
|
25
|
+
'''
|
26
|
+
GTK-Doc comment block format
|
27
|
+
----------------------------
|
28
|
+
|
29
|
+
A GTK-Doc comment block is built out of multiple parts. Each part can be further
|
30
|
+
divided into fields which are separated by a colon ("``:``") delimiter.
|
31
|
+
|
32
|
+
Known parts and the fields they are constructed from look like the following
|
33
|
+
(optional fields are enclosed in square brackets)::
|
34
|
+
|
35
|
+
┌───────────────────────────────────────────────────────────┐
|
36
|
+
│ /** │ ─▷ start token
|
37
|
+
├────────────────────┬──────────────────────────────────────┤
|
38
|
+
│ * identifier_name │ [: annotations] │ ─▷ identifier part
|
39
|
+
├────────────────────┼─────────────────┬────────────────────┤
|
40
|
+
│ * @parameter_name │ [: annotations] │ : description │ ─▷ parameter part
|
41
|
+
├────────────────────┴─────────────────┴────────────────────┤
|
42
|
+
│ * │ ─▷ comment block description
|
43
|
+
│ * comment_block_description │
|
44
|
+
├─────────────┬─────────────────┬───────────┬───────────────┤
|
45
|
+
│ * tag_name │ [: annotations] │ [: value] │ : description │ ─▷ tag part
|
46
|
+
├─────────────┴─────────────────┴───────────┴───────────────┤
|
47
|
+
│ */ │ ─▷ end token
|
48
|
+
└───────────────────────────────────────────────────────────┘
|
49
|
+
|
50
|
+
There are two conditions that must be met before a comment block is recognized
|
51
|
+
as a GTK-Doc comment block:
|
52
|
+
|
53
|
+
#. The comment block is opened with a GTK-Doc start token ("``/**``")
|
54
|
+
#. The first line following the start token contains a valid identifier part
|
55
|
+
|
56
|
+
Once a GTK-Doc comment block has been identified as such and has been stripped
|
57
|
+
from its start and end tokens the remaining parts have to be written in a
|
58
|
+
specific order:
|
59
|
+
|
60
|
+
#. There must be exactly 1 `identifier` part on the first line of the
|
61
|
+
comment block which consists of:
|
62
|
+
|
63
|
+
* a required `identifier_name` field
|
64
|
+
* an optional `annotations` field
|
65
|
+
|
66
|
+
#. Zero or more `parameter` parts, each consisting of:
|
67
|
+
|
68
|
+
* a required `parameter_name` field
|
69
|
+
* an optional `annotations` field
|
70
|
+
* a required `description` field (can be the empty string)
|
71
|
+
|
72
|
+
#. One optional `comment block description` part which must begin with at
|
73
|
+
least 1 empty line signaling the start of this part.
|
74
|
+
|
75
|
+
#. Zero or more `tag` parts, each consisting of:
|
76
|
+
|
77
|
+
* a required `tag_name` field
|
78
|
+
* an optional `annotations` field
|
79
|
+
* an optional `value` field
|
80
|
+
* a required `description` field (can be the empty string)
|
81
|
+
|
82
|
+
Additionally, the following restrictions are in effect:
|
24
83
|
|
84
|
+
#. Separating parts with an empty line:
|
25
85
|
|
86
|
+
* `identifier` and `parameter` parts cannot be separated from each other by
|
87
|
+
an empty line as this would signal the start of the
|
88
|
+
`comment block description` part (see above).
|
89
|
+
* it is required to separate the `comment block description` part from the
|
90
|
+
`identifier` or `parameter` parts with an empty line (see above)
|
91
|
+
* `comment block description` and `tag` parts can optionally be separated
|
92
|
+
by an empty line
|
93
|
+
|
94
|
+
#. Parts and fields cannot span multiple lines, except for:
|
95
|
+
|
96
|
+
* the `comment_block_description` part
|
97
|
+
* `parameter description` and `tag description` fields
|
98
|
+
|
99
|
+
#. Taking the above restrictions into account, spanning multiple paragraphs is
|
100
|
+
limited to the `comment block description` part and `tag description` fields.
|
101
|
+
|
102
|
+
Refer to the `GTK-Doc manual`_ for more detailed usage information.
|
103
|
+
|
104
|
+
.. _GTK-Doc manual:
|
105
|
+
http://developer.gnome.org/gtk-doc-manual/1.18/documenting.html.en
|
106
|
+
'''
|
107
|
+
|
108
|
+
|
109
|
+
from __future__ import absolute_import
|
110
|
+
|
111
|
+
import os
|
26
112
|
import re
|
27
113
|
|
28
|
-
from
|
29
|
-
from
|
114
|
+
from collections import namedtuple
|
115
|
+
from operator import ne, gt, lt
|
116
|
+
|
117
|
+
from .collections import Counter, OrderedDict
|
118
|
+
from .message import Position, warn, error
|
30
119
|
|
31
120
|
|
32
121
|
# GTK-Doc comment block parts
|
33
|
-
PART_IDENTIFIER =
|
34
|
-
PART_PARAMETERS =
|
35
|
-
PART_DESCRIPTION =
|
36
|
-
PART_TAGS =
|
37
|
-
|
38
|
-
#
|
39
|
-
|
40
|
-
|
41
|
-
IDENTIFIER_PROPERTY = 'property'
|
42
|
-
IDENTIFIER_SIGNAL = 'signal'
|
43
|
-
|
44
|
-
# Tags - annotations applied to comment blocks
|
45
|
-
TAG_VFUNC = 'virtual'
|
46
|
-
TAG_SINCE = 'since'
|
47
|
-
TAG_STABILITY = 'stability'
|
122
|
+
PART_IDENTIFIER = 0
|
123
|
+
PART_PARAMETERS = 1
|
124
|
+
PART_DESCRIPTION = 2
|
125
|
+
PART_TAGS = 3
|
126
|
+
|
127
|
+
# GTK-Doc comment block tags
|
128
|
+
# 1) Basic GTK-Doc tags.
|
129
|
+
# Note: This list cannot be extended unless the GTK-Doc project defines new tags.
|
48
130
|
TAG_DEPRECATED = 'deprecated'
|
49
131
|
TAG_RETURNS = 'returns'
|
50
|
-
|
132
|
+
TAG_SINCE = 'since'
|
133
|
+
TAG_STABILITY = 'stability'
|
134
|
+
|
135
|
+
GTKDOC_TAGS = [TAG_DEPRECATED,
|
136
|
+
TAG_RETURNS,
|
137
|
+
TAG_SINCE,
|
138
|
+
TAG_STABILITY]
|
139
|
+
|
140
|
+
# 2) Deprecated basic GTK-Doc tags.
|
141
|
+
# Note: This list cannot be extended unless the GTK-Doc project defines new deprecated tags.
|
51
142
|
TAG_DESCRIPTION = 'description'
|
143
|
+
TAG_RETURN_VALUE = 'return value'
|
144
|
+
|
145
|
+
DEPRECATED_GTKDOC_TAGS = [TAG_DESCRIPTION,
|
146
|
+
TAG_RETURN_VALUE]
|
147
|
+
|
148
|
+
# 3) Deprecated GObject-Introspection tags.
|
149
|
+
# Unfortunately, these where accepted by old versions of this module.
|
150
|
+
TAG_RETURN = 'return'
|
151
|
+
TAG_RETURNS_VALUE = 'returns value'
|
152
|
+
|
153
|
+
DEPRECATED_GI_TAGS = [TAG_RETURN,
|
154
|
+
TAG_RETURNS_VALUE]
|
155
|
+
|
156
|
+
# 4) Deprecated GObject-Introspection annotation tags.
|
157
|
+
# Accepted by old versions of this module while they should have been
|
158
|
+
# annotations on the identifier part instead.
|
159
|
+
# Note: This list can not be extended ever again. The GObject-Introspection project is not
|
160
|
+
# allowed to invent GTK-Doc tags. Please create new annotations instead.
|
52
161
|
TAG_ATTRIBUTES = 'attributes'
|
53
|
-
|
54
|
-
TAG_TYPE = 'type'
|
55
|
-
TAG_UNREF_FUNC = 'unref func'
|
162
|
+
TAG_GET_VALUE_FUNC = 'get value func'
|
56
163
|
TAG_REF_FUNC = 'ref func'
|
164
|
+
TAG_RENAME_TO = 'rename to'
|
57
165
|
TAG_SET_VALUE_FUNC = 'set value func'
|
58
|
-
TAG_GET_VALUE_FUNC = 'get value func'
|
59
166
|
TAG_TRANSFER = 'transfer'
|
167
|
+
TAG_TYPE = 'type'
|
168
|
+
TAG_UNREF_FUNC = 'unref func'
|
60
169
|
TAG_VALUE = 'value'
|
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
|
-
|
170
|
+
TAG_VFUNC = 'virtual'
|
171
|
+
|
172
|
+
DEPRECATED_GI_ANN_TAGS = [TAG_ATTRIBUTES,
|
173
|
+
TAG_GET_VALUE_FUNC,
|
174
|
+
TAG_REF_FUNC,
|
175
|
+
TAG_RENAME_TO,
|
176
|
+
TAG_SET_VALUE_FUNC,
|
177
|
+
TAG_TRANSFER,
|
178
|
+
TAG_TYPE,
|
179
|
+
TAG_UNREF_FUNC,
|
180
|
+
TAG_VALUE,
|
181
|
+
TAG_VFUNC]
|
182
|
+
|
183
|
+
ALL_TAGS = GTKDOC_TAGS + DEPRECATED_GTKDOC_TAGS + DEPRECATED_GI_TAGS + DEPRECATED_GI_ANN_TAGS
|
184
|
+
|
185
|
+
# GObject-Introspection annotation start/end tokens
|
186
|
+
ANN_LPAR = '('
|
187
|
+
ANN_RPAR = ')'
|
188
|
+
|
189
|
+
# GObject-Introspection annotations
|
190
|
+
# 1) Supported annotations
|
191
|
+
# Note: when adding new annotations, GTK-Doc project's gtkdoc-mkdb needs to be modified too!
|
192
|
+
ANN_ALLOW_NONE = 'allow-none'
|
193
|
+
ANN_ARRAY = 'array'
|
194
|
+
ANN_ATTRIBUTES = 'attributes'
|
195
|
+
ANN_CLOSURE = 'closure'
|
196
|
+
ANN_CONSTRUCTOR = 'constructor'
|
197
|
+
ANN_DESTROY = 'destroy'
|
198
|
+
ANN_ELEMENT_TYPE = 'element-type'
|
199
|
+
ANN_FOREIGN = 'foreign'
|
200
|
+
ANN_GET_VALUE_FUNC = 'get-value-func'
|
201
|
+
ANN_IN = 'in'
|
202
|
+
ANN_INOUT = 'inout'
|
203
|
+
ANN_METHOD = 'method'
|
204
|
+
ANN_NULLABLE = 'nullable'
|
205
|
+
ANN_OPTIONAL = 'optional'
|
206
|
+
ANN_OUT = 'out'
|
207
|
+
ANN_REF_FUNC = 'ref-func'
|
208
|
+
ANN_RENAME_TO = 'rename-to'
|
209
|
+
ANN_SCOPE = 'scope'
|
210
|
+
ANN_SET_VALUE_FUNC = 'set-value-func'
|
211
|
+
ANN_SKIP = 'skip'
|
212
|
+
ANN_TRANSFER = 'transfer'
|
213
|
+
ANN_TYPE = 'type'
|
214
|
+
ANN_UNREF_FUNC = 'unref-func'
|
215
|
+
ANN_VFUNC = 'virtual'
|
216
|
+
ANN_VALUE = 'value'
|
217
|
+
|
218
|
+
GI_ANNS = [ANN_ALLOW_NONE,
|
219
|
+
ANN_NULLABLE,
|
220
|
+
ANN_OPTIONAL,
|
221
|
+
ANN_ARRAY,
|
222
|
+
ANN_ATTRIBUTES,
|
223
|
+
ANN_CLOSURE,
|
224
|
+
ANN_CONSTRUCTOR,
|
225
|
+
ANN_DESTROY,
|
226
|
+
ANN_ELEMENT_TYPE,
|
227
|
+
ANN_FOREIGN,
|
228
|
+
ANN_GET_VALUE_FUNC,
|
229
|
+
ANN_IN,
|
230
|
+
ANN_INOUT,
|
231
|
+
ANN_METHOD,
|
232
|
+
ANN_OUT,
|
233
|
+
ANN_REF_FUNC,
|
234
|
+
ANN_RENAME_TO,
|
235
|
+
ANN_SCOPE,
|
236
|
+
ANN_SET_VALUE_FUNC,
|
237
|
+
ANN_SKIP,
|
238
|
+
ANN_TRANSFER,
|
239
|
+
ANN_TYPE,
|
240
|
+
ANN_UNREF_FUNC,
|
241
|
+
ANN_VFUNC,
|
242
|
+
ANN_VALUE]
|
243
|
+
|
244
|
+
# 2) Deprecated GObject-Introspection annotations
|
245
|
+
ANN_ATTRIBUTE = 'attribute'
|
246
|
+
ANN_INOUT_ALT = 'in-out'
|
247
|
+
|
248
|
+
DEPRECATED_GI_ANNS = [ANN_ATTRIBUTE,
|
249
|
+
ANN_INOUT_ALT]
|
250
|
+
|
251
|
+
ALL_ANNOTATIONS = GI_ANNS + DEPRECATED_GI_ANNS
|
252
|
+
DICT_ANNOTATIONS = [ANN_ARRAY, ANN_ATTRIBUTES]
|
253
|
+
LIST_ANNOTATIONS = [ann for ann in ALL_ANNOTATIONS if ann not in DICT_ANNOTATIONS]
|
254
|
+
|
255
|
+
# (array) annotation options
|
117
256
|
OPT_ARRAY_FIXED_SIZE = 'fixed-size'
|
118
257
|
OPT_ARRAY_LENGTH = 'length'
|
119
258
|
OPT_ARRAY_ZERO_TERMINATED = 'zero-terminated'
|
120
259
|
|
121
|
-
|
122
|
-
|
260
|
+
ARRAY_OPTIONS = [OPT_ARRAY_FIXED_SIZE,
|
261
|
+
OPT_ARRAY_LENGTH,
|
262
|
+
OPT_ARRAY_ZERO_TERMINATED]
|
263
|
+
|
264
|
+
# (out) annotation options
|
123
265
|
OPT_OUT_CALLEE_ALLOCATES = 'callee-allocates'
|
266
|
+
OPT_OUT_CALLER_ALLOCATES = 'caller-allocates'
|
267
|
+
|
268
|
+
OUT_OPTIONS = [OPT_OUT_CALLEE_ALLOCATES,
|
269
|
+
OPT_OUT_CALLER_ALLOCATES]
|
124
270
|
|
125
|
-
#
|
271
|
+
# (scope) annotation options
|
126
272
|
OPT_SCOPE_ASYNC = 'async'
|
127
273
|
OPT_SCOPE_CALL = 'call'
|
128
274
|
OPT_SCOPE_NOTIFIED = 'notified'
|
129
275
|
|
130
|
-
|
131
|
-
|
276
|
+
SCOPE_OPTIONS = [OPT_SCOPE_ASYNC,
|
277
|
+
OPT_SCOPE_CALL,
|
278
|
+
OPT_SCOPE_NOTIFIED]
|
279
|
+
|
280
|
+
# (transfer) annotation options
|
132
281
|
OPT_TRANSFER_CONTAINER = 'container'
|
133
|
-
OPT_TRANSFER_FULL = 'full'
|
134
282
|
OPT_TRANSFER_FLOATING = 'floating'
|
283
|
+
OPT_TRANSFER_FULL = 'full'
|
284
|
+
OPT_TRANSFER_NONE = 'none'
|
135
285
|
|
286
|
+
TRANSFER_OPTIONS = [OPT_TRANSFER_CONTAINER,
|
287
|
+
OPT_TRANSFER_FLOATING,
|
288
|
+
OPT_TRANSFER_FULL,
|
289
|
+
OPT_TRANSFER_NONE]
|
136
290
|
|
137
|
-
#The following regular expression programs are built to:
|
138
|
-
# - match (or substitute) a single comment block line at a time;
|
139
|
-
# - support (but remains untested) LOCALE and UNICODE modes.
|
140
291
|
|
141
|
-
#
|
142
|
-
|
143
|
-
|
144
|
-
|
292
|
+
# Pattern used to normalize different types of line endings
|
293
|
+
LINE_BREAK_RE = re.compile(r'\r\n|\r|\n', re.UNICODE)
|
294
|
+
|
295
|
+
# Pattern matching the start token of a comment block.
|
296
|
+
COMMENT_BLOCK_START_RE = re.compile(
|
145
297
|
r'''
|
146
|
-
^
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
298
|
+
^ # start
|
299
|
+
(?P<code>.*?) # whitespace, code, ...
|
300
|
+
\s* # 0 or more whitespace characters
|
301
|
+
(?P<token>/\*{2}(?![\*/])) # 1 forward slash character followed
|
302
|
+
# by exactly 2 asterisk characters
|
303
|
+
# and not followed by a slash character
|
304
|
+
\s* # 0 or more whitespace characters
|
305
|
+
(?P<comment>.*?) # GTK-Doc comment text
|
306
|
+
\s* # 0 or more whitespace characters
|
307
|
+
$ # end
|
152
308
|
''',
|
153
|
-
re.VERBOSE)
|
309
|
+
re.UNICODE | re.VERBOSE)
|
154
310
|
|
155
|
-
#
|
156
|
-
|
157
|
-
# reasons. See https://bugzilla.gnome.org/show_bug.cgi?id=689354
|
158
|
-
#
|
159
|
-
# Results in 1 symbolic group:
|
160
|
-
# - group 1 = description
|
161
|
-
COMMENT_END_RE = re.compile(
|
311
|
+
# Pattern matching the end token of a comment block.
|
312
|
+
COMMENT_BLOCK_END_RE = re.compile(
|
162
313
|
r'''
|
163
|
-
^
|
164
|
-
|
165
|
-
(?P<
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
314
|
+
^ # start
|
315
|
+
\s* # 0 or more whitespace characters
|
316
|
+
(?P<comment>.*?) # GTK-Doc comment text
|
317
|
+
\s* # 0 or more whitespace characters
|
318
|
+
(?P<token>\*+/) # 1 or more asterisk characters followed
|
319
|
+
# by exactly 1 forward slash character
|
320
|
+
(?P<code>.*?) # whitespace, code, ...
|
321
|
+
\s* # 0 or more whitespace characters
|
322
|
+
$ # end
|
171
323
|
''',
|
172
|
-
re.VERBOSE)
|
324
|
+
re.UNICODE | re.VERBOSE)
|
173
325
|
|
174
|
-
#
|
326
|
+
# Pattern matching the ' * ' at the beginning of every
|
175
327
|
# line inside a comment block.
|
176
|
-
#
|
177
|
-
# Results in 0 symbolic groups.
|
178
328
|
COMMENT_ASTERISK_RE = re.compile(
|
179
329
|
r'''
|
180
|
-
^
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
330
|
+
^ # start
|
331
|
+
\s* # 0 or more whitespace characters
|
332
|
+
(?P<comment>.*?) # invalid comment text
|
333
|
+
\s* # 0 or more whitespace characters
|
334
|
+
\* # 1 asterisk character
|
335
|
+
\s? # 0 or 1 whitespace characters
|
336
|
+
# WARNING: removing more than 1
|
337
|
+
# whitespace character breaks
|
338
|
+
# embedded example program indentation
|
187
339
|
''',
|
188
|
-
re.VERBOSE)
|
340
|
+
re.UNICODE | re.VERBOSE)
|
189
341
|
|
190
|
-
#
|
191
|
-
#
|
192
|
-
|
193
|
-
# Results in 1 symbolic group:
|
194
|
-
# - group 1 = indentation
|
195
|
-
COMMENT_INDENTATION_RE = re.compile(
|
342
|
+
# Pattern matching the indentation level of a line (used
|
343
|
+
# to get the indentation before and after the ' * ').
|
344
|
+
INDENTATION_RE = re.compile(
|
196
345
|
r'''
|
197
346
|
^
|
198
|
-
(?P<indentation
|
347
|
+
(?P<indentation>\s*) # 0 or more whitespace characters
|
199
348
|
.*
|
200
349
|
$
|
201
350
|
''',
|
202
|
-
re.VERBOSE)
|
351
|
+
re.UNICODE | re.VERBOSE)
|
203
352
|
|
204
|
-
#
|
205
|
-
#
|
206
|
-
# Results in 0 symbolic groups.
|
353
|
+
# Pattern matching an empty line.
|
207
354
|
EMPTY_LINE_RE = re.compile(
|
208
355
|
r'''
|
209
|
-
^
|
210
|
-
|
211
|
-
$
|
356
|
+
^ # start
|
357
|
+
\s* # 0 or more whitespace characters
|
358
|
+
$ # end
|
212
359
|
''',
|
213
|
-
re.VERBOSE)
|
360
|
+
re.UNICODE | re.VERBOSE)
|
214
361
|
|
215
|
-
#
|
216
|
-
#
|
217
|
-
# Results in 2 symbolic groups:
|
218
|
-
# - group 1 = colon
|
219
|
-
# - group 2 = section_name
|
362
|
+
# Pattern matching SECTION identifiers.
|
220
363
|
SECTION_RE = re.compile(
|
221
364
|
r'''
|
222
|
-
^
|
223
|
-
|
224
|
-
SECTION
|
225
|
-
|
226
|
-
(?P<
|
227
|
-
|
228
|
-
(?P<section_name>\w\S
|
229
|
-
|
365
|
+
^ # start
|
366
|
+
\s* # 0 or more whitespace characters
|
367
|
+
SECTION # SECTION
|
368
|
+
\s* # 0 or more whitespace characters
|
369
|
+
(?P<delimiter>:?) # delimiter
|
370
|
+
\s* # 0 or more whitespace characters
|
371
|
+
(?P<section_name>\w\S+?) # section name
|
372
|
+
\s* # 0 or more whitespace characters
|
373
|
+
:? # invalid delimiter
|
374
|
+
\s* # 0 or more whitespace characters
|
230
375
|
$
|
231
376
|
''',
|
232
|
-
re.VERBOSE)
|
377
|
+
re.UNICODE | re.VERBOSE)
|
233
378
|
|
234
|
-
#
|
235
|
-
#
|
236
|
-
# Results in 3 symbolic groups:
|
237
|
-
# - group 1 = symbol_name
|
238
|
-
# - group 2 = colon
|
239
|
-
# - group 3 = annotations
|
379
|
+
# Pattern matching symbol (function, constant, struct and enum) identifiers.
|
240
380
|
SYMBOL_RE = re.compile(
|
241
381
|
r'''
|
242
|
-
^
|
243
|
-
|
244
|
-
(?P<symbol_name>[\w-]*\w)
|
245
|
-
|
246
|
-
(?P<
|
247
|
-
|
248
|
-
(?P<
|
249
|
-
|
250
|
-
|
382
|
+
^ # start
|
383
|
+
\s* # 0 or more whitespace characters
|
384
|
+
(?P<symbol_name>[\w-]*\w) # symbol name
|
385
|
+
\s* # 0 or more whitespace characters
|
386
|
+
(?P<delimiter>:?) # delimiter
|
387
|
+
\s* # 0 or more whitespace characters
|
388
|
+
(?P<fields>.*?) # annotations + description
|
389
|
+
\s* # 0 or more whitespace characters
|
390
|
+
:? # invalid delimiter
|
391
|
+
\s* # 0 or more whitespace characters
|
392
|
+
$ # end
|
251
393
|
''',
|
252
|
-
re.VERBOSE)
|
394
|
+
re.UNICODE | re.VERBOSE)
|
253
395
|
|
254
|
-
#
|
255
|
-
#
|
256
|
-
# Results in 4 symbolic groups:
|
257
|
-
# - group 1 = class_name
|
258
|
-
# - group 2 = property_name
|
259
|
-
# - group 3 = colon
|
260
|
-
# - group 4 = annotations
|
396
|
+
# Pattern matching property identifiers.
|
261
397
|
PROPERTY_RE = re.compile(
|
262
398
|
r'''
|
263
|
-
^
|
264
|
-
|
265
|
-
(?P<class_name>[\w]+)
|
266
|
-
|
267
|
-
:{1}
|
268
|
-
|
269
|
-
(?P<property_name>[\w-]*\w)
|
270
|
-
|
271
|
-
(?P<
|
272
|
-
|
273
|
-
(?P<
|
274
|
-
|
275
|
-
|
399
|
+
^ # start
|
400
|
+
\s* # 0 or more whitespace characters
|
401
|
+
(?P<class_name>[\w]+) # class name
|
402
|
+
\s* # 0 or more whitespace characters
|
403
|
+
:{1} # 1 required colon
|
404
|
+
\s* # 0 or more whitespace characters
|
405
|
+
(?P<property_name>[\w-]*\w) # property name
|
406
|
+
\s* # 0 or more whitespace characters
|
407
|
+
(?P<delimiter>:?) # delimiter
|
408
|
+
\s* # 0 or more whitespace characters
|
409
|
+
(?P<fields>.*?) # annotations + description
|
410
|
+
\s* # 0 or more whitespace characters
|
411
|
+
:? # invalid delimiter
|
412
|
+
\s* # 0 or more whitespace characters
|
413
|
+
$ # end
|
276
414
|
''',
|
277
|
-
re.VERBOSE)
|
415
|
+
re.UNICODE | re.VERBOSE)
|
278
416
|
|
279
|
-
#
|
280
|
-
#
|
281
|
-
# Results in 4 symbolic groups:
|
282
|
-
# - group 1 = class_name
|
283
|
-
# - group 2 = signal_name
|
284
|
-
# - group 3 = colon
|
285
|
-
# - group 4 = annotations
|
417
|
+
# Pattern matching signal identifiers.
|
286
418
|
SIGNAL_RE = re.compile(
|
287
419
|
r'''
|
288
|
-
^
|
289
|
-
|
290
|
-
(?P<class_name>[\w]+)
|
291
|
-
|
292
|
-
:{2}
|
293
|
-
|
294
|
-
(?P<signal_name>[\w-]*\w)
|
295
|
-
|
296
|
-
(?P<
|
297
|
-
|
298
|
-
(?P<
|
299
|
-
|
300
|
-
|
420
|
+
^ # start
|
421
|
+
\s* # 0 or more whitespace characters
|
422
|
+
(?P<class_name>[\w]+) # class name
|
423
|
+
\s* # 0 or more whitespace characters
|
424
|
+
:{2} # 2 required colons
|
425
|
+
\s* # 0 or more whitespace characters
|
426
|
+
(?P<signal_name>[\w-]*\w) # signal name
|
427
|
+
\s* # 0 or more whitespace characters
|
428
|
+
(?P<delimiter>:?) # delimiter
|
429
|
+
\s* # 0 or more whitespace characters
|
430
|
+
(?P<fields>.*?) # annotations + description
|
431
|
+
\s* # 0 or more whitespace characters
|
432
|
+
:? # invalid delimiter
|
433
|
+
\s* # 0 or more whitespace characters
|
434
|
+
$ # end
|
301
435
|
''',
|
302
|
-
re.VERBOSE)
|
436
|
+
re.UNICODE | re.VERBOSE)
|
303
437
|
|
304
|
-
#
|
305
|
-
#
|
306
|
-
# Results in 4 symbolic groups:
|
307
|
-
# - group 1 = parameter_name
|
308
|
-
# - group 2 = annotations
|
309
|
-
# - group 3 = colon
|
310
|
-
# - group 4 = description
|
438
|
+
# Pattern matching parameters.
|
311
439
|
PARAMETER_RE = re.compile(
|
312
440
|
r'''
|
313
|
-
^
|
314
|
-
|
315
|
-
@
|
316
|
-
(?P<parameter_name>[\w-]*\w
|
317
|
-
|
318
|
-
:{1}
|
319
|
-
|
320
|
-
(?P<
|
321
|
-
|
322
|
-
|
323
|
-
(?P<description>.*?) # description
|
324
|
-
[^\S\n\r]* # 0 or more whitespace characters
|
325
|
-
$ # end
|
441
|
+
^ # start
|
442
|
+
\s* # 0 or more whitespace characters
|
443
|
+
@ # @ character
|
444
|
+
(?P<parameter_name>[\w-]*\w|.*?\.\.\.) # parameter name
|
445
|
+
\s* # 0 or more whitespace characters
|
446
|
+
:{1} # 1 required delimiter
|
447
|
+
\s* # 0 or more whitespace characters
|
448
|
+
(?P<fields>.*?) # annotations + description
|
449
|
+
\s* # 0 or more whitespace characters
|
450
|
+
$ # end
|
326
451
|
''',
|
327
|
-
re.VERBOSE)
|
452
|
+
re.UNICODE | re.VERBOSE)
|
328
453
|
|
329
|
-
#
|
330
|
-
|
331
|
-
# Results in 4 symbolic groups:
|
332
|
-
# - group 1 = tag_name
|
333
|
-
# - group 2 = annotations
|
334
|
-
# - group 3 = colon
|
335
|
-
# - group 4 = description
|
336
|
-
_all_tags = '|'.join(_ALL_TAGS).replace(' ', '\\ ')
|
454
|
+
# Pattern matching tags.
|
455
|
+
_all_tags = '|'.join(ALL_TAGS).replace(' ', r'\s')
|
337
456
|
TAG_RE = re.compile(
|
338
457
|
r'''
|
339
|
-
^
|
340
|
-
|
341
|
-
(?P<tag_name>''' + _all_tags + r''')
|
342
|
-
|
343
|
-
:{1}
|
344
|
-
|
345
|
-
(?P<
|
346
|
-
|
347
|
-
|
348
|
-
(?P<description>.*?) # description
|
349
|
-
[^\S\n\r]* # 0 or more whitespace characters
|
350
|
-
$ # end
|
458
|
+
^ # start
|
459
|
+
\s* # 0 or more whitespace characters
|
460
|
+
(?P<tag_name>''' + _all_tags + r''') # tag name
|
461
|
+
\s* # 0 or more whitespace characters
|
462
|
+
:{1} # 1 required delimiter
|
463
|
+
\s* # 0 or more whitespace characters
|
464
|
+
(?P<fields>.*?) # annotations + value + description
|
465
|
+
\s* # 0 or more whitespace characters
|
466
|
+
$ # end
|
351
467
|
''',
|
352
|
-
re.VERBOSE | re.IGNORECASE)
|
468
|
+
re.UNICODE | re.VERBOSE | re.IGNORECASE)
|
353
469
|
|
354
|
-
#
|
355
|
-
|
356
|
-
# generate warnings about invalid annotations spanning multiple lines.
|
357
|
-
#
|
358
|
-
# Results in 3 symbolic groups:
|
359
|
-
# - group 2 = annotations
|
360
|
-
# - group 3 = colon
|
361
|
-
# - group 4 = description
|
362
|
-
MULTILINE_ANNOTATION_CONTINUATION_RE = re.compile(
|
470
|
+
# Pattern matching value and description fields for TAG_DEPRECATED & TAG_SINCE tags.
|
471
|
+
TAG_VALUE_VERSION_RE = re.compile(
|
363
472
|
r'''
|
364
|
-
^
|
365
|
-
|
366
|
-
(?P<
|
367
|
-
|
368
|
-
|
369
|
-
|
370
|
-
|
371
|
-
|
473
|
+
^ # start
|
474
|
+
\s* # 0 or more whitespace characters
|
475
|
+
(?P<value>([0-9\.])*) # value
|
476
|
+
\s* # 0 or more whitespace characters
|
477
|
+
(?P<delimiter>:?) # delimiter
|
478
|
+
\s* # 0 or more whitespace characters
|
479
|
+
(?P<description>.*?) # description
|
480
|
+
\s* # 0 or more whitespace characters
|
481
|
+
$ # end
|
372
482
|
''',
|
373
|
-
re.VERBOSE)
|
483
|
+
re.UNICODE | re.VERBOSE)
|
374
484
|
|
485
|
+
# Pattern matching value and description fields for TAG_STABILITY tags.
|
486
|
+
TAG_VALUE_STABILITY_RE = re.compile(
|
487
|
+
r'''
|
488
|
+
^ # start
|
489
|
+
\s* # 0 or more whitespace characters
|
490
|
+
(?P<value>(stable|unstable|private|internal)?) # value
|
491
|
+
\s* # 0 or more whitespace characters
|
492
|
+
(?P<delimiter>:?) # delimiter
|
493
|
+
\s* # 0 or more whitespace characters
|
494
|
+
(?P<description>.*?) # description
|
495
|
+
\s* # 0 or more whitespace characters
|
496
|
+
$ # end
|
497
|
+
''',
|
498
|
+
re.UNICODE | re.VERBOSE | re.IGNORECASE)
|
375
499
|
|
376
|
-
class DocBlock(object):
|
377
500
|
|
378
|
-
|
379
|
-
|
380
|
-
|
381
|
-
|
382
|
-
|
383
|
-
|
384
|
-
self.params = OrderedDict()
|
385
|
-
self.position = None
|
501
|
+
class GtkDocAnnotations(OrderedDict):
|
502
|
+
'''
|
503
|
+
An ordered dictionary mapping annotation names to annotation options (if any). Annotation
|
504
|
+
options can be either a :class:`list`, a :class:`giscanner.collections.OrderedDict`
|
505
|
+
(depending on the annotation name)or :const:`None`.
|
506
|
+
'''
|
386
507
|
|
387
|
-
|
388
|
-
return cmp(self.name, other.name)
|
508
|
+
__slots__ = ('position')
|
389
509
|
|
390
|
-
def
|
391
|
-
|
392
|
-
|
393
|
-
def to_gtk_doc(self):
|
394
|
-
options = ''
|
395
|
-
if self.options:
|
396
|
-
options += ' '
|
397
|
-
options += ' '.join('(%s)' % o for o in self.options)
|
398
|
-
lines = [self.name]
|
399
|
-
if 'SECTION' not in self.name:
|
400
|
-
lines[0] += ':'
|
401
|
-
lines[0] += options
|
402
|
-
for param in self.params.values():
|
403
|
-
lines.append(param.to_gtk_doc_param())
|
404
|
-
if self.comment:
|
405
|
-
lines.append('')
|
406
|
-
for l in self.comment.split('\n'):
|
407
|
-
lines.append(l)
|
408
|
-
if self.tags:
|
409
|
-
lines.append('')
|
410
|
-
for tag in self.tags.values():
|
411
|
-
lines.append(tag.to_gtk_doc_tag())
|
412
|
-
|
413
|
-
comment = ''
|
414
|
-
comment += '/**\n'
|
415
|
-
for line in lines:
|
416
|
-
line = line.rstrip()
|
417
|
-
if line:
|
418
|
-
comment += ' * %s\n' % (line, )
|
419
|
-
else:
|
420
|
-
comment += ' *\n'
|
421
|
-
comment += ' */\n'
|
422
|
-
return comment
|
510
|
+
def __init__(self, position=None):
|
511
|
+
OrderedDict.__init__(self)
|
423
512
|
|
424
|
-
|
425
|
-
|
426
|
-
|
513
|
+
#: A :class:`giscanner.message.Position` instance specifying the location of the
|
514
|
+
#: annotations in the source file or :const:`None`.
|
515
|
+
self.position = position
|
427
516
|
|
428
|
-
for tag in self.tags.values():
|
429
|
-
tag.validate()
|
430
517
|
|
518
|
+
class GtkDocAnnotatable(object):
|
519
|
+
'''
|
520
|
+
Base class for GTK-Doc comment block parts that can be annotated.
|
521
|
+
'''
|
431
522
|
|
432
|
-
|
523
|
+
__slots__ = ('position', 'annotations')
|
433
524
|
|
434
|
-
|
435
|
-
|
436
|
-
|
437
|
-
|
438
|
-
|
439
|
-
|
440
|
-
|
525
|
+
#: A :class:`tuple` of annotation name constants that are valid for this object. Annotation
|
526
|
+
#: names not in this :class:`tuple` will be reported as *unknown* by :func:`validate`. The
|
527
|
+
#: :attr:`valid_annotations` class attribute should be overridden by subclasses.
|
528
|
+
valid_annotations = ()
|
529
|
+
|
530
|
+
def __init__(self, position=None):
|
531
|
+
#: A :class:`giscanner.message.Position` instance specifying the location of the
|
532
|
+
#: annotatable comment block part in the source file or :const:`None`.
|
533
|
+
self.position = position
|
534
|
+
|
535
|
+
#: A :class:`GtkDocAnnotations` instance representing the annotations
|
536
|
+
#: applied to this :class:`GtkDocAnnotatable` instance.
|
537
|
+
self.annotations = GtkDocAnnotations()
|
441
538
|
|
442
539
|
def __repr__(self):
|
443
|
-
return '<
|
540
|
+
return '<GtkDocAnnotatable %r %r>' % (self.annotations, )
|
444
541
|
|
445
|
-
def
|
446
|
-
|
447
|
-
|
448
|
-
|
449
|
-
name, ), self.position)
|
450
|
-
return
|
542
|
+
def validate(self):
|
543
|
+
'''
|
544
|
+
Validate annotations stored by the :class:`GtkDocAnnotatable` instance, if any.
|
545
|
+
'''
|
451
546
|
|
452
|
-
if
|
453
|
-
|
454
|
-
|
455
|
-
|
456
|
-
|
457
|
-
|
458
|
-
|
459
|
-
|
460
|
-
|
461
|
-
|
462
|
-
|
547
|
+
if self.annotations:
|
548
|
+
position = self.annotations.position
|
549
|
+
|
550
|
+
for ann_name, options in self.annotations.items():
|
551
|
+
if ann_name in self.valid_annotations:
|
552
|
+
validate = getattr(self, '_do_validate_' + ann_name.replace('-', '_'))
|
553
|
+
validate(position, ann_name, options)
|
554
|
+
elif ann_name in ALL_ANNOTATIONS:
|
555
|
+
# Not error() as ann_name might be valid in some newer
|
556
|
+
# GObject-Instrospection version.
|
557
|
+
warn('unexpected annotation: %s' % (ann_name, ), position)
|
463
558
|
else:
|
464
|
-
|
465
|
-
|
466
|
-
|
467
|
-
|
468
|
-
|
469
|
-
|
470
|
-
|
471
|
-
|
472
|
-
|
473
|
-
|
474
|
-
|
475
|
-
|
476
|
-
|
477
|
-
|
559
|
+
# Not error() as ann_name might be valid in some newer
|
560
|
+
# GObject-Instrospection version.
|
561
|
+
warn('unknown annotation: %s' % (ann_name, ), position)
|
562
|
+
|
563
|
+
def _validate_options(self, position, ann_name, n_options, expected_n_options, operator,
|
564
|
+
message):
|
565
|
+
'''
|
566
|
+
Validate the number of options held by an annotation according to the test
|
567
|
+
``operator(n_options, expected_n_options)``.
|
568
|
+
|
569
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
570
|
+
containing the annotation to be validated
|
571
|
+
:param ann_name: name of the annotation holding the options to validate
|
572
|
+
:param n_options: number of options held by the annotation
|
573
|
+
:param expected_n_options: number of expected options
|
574
|
+
:param operator: an operator function from python's :mod:`operator` module, for example
|
575
|
+
:func:`operator.ne` or :func:`operator.lt`
|
576
|
+
:param message: warning message used when the test
|
577
|
+
``operator(n_options, expected_n_options)`` fails.
|
578
|
+
'''
|
579
|
+
|
580
|
+
if n_options == 0:
|
581
|
+
t = 'none'
|
582
|
+
else:
|
583
|
+
t = '%d' % (n_options, )
|
584
|
+
|
585
|
+
if expected_n_options == 0:
|
586
|
+
s = 'no options'
|
587
|
+
elif expected_n_options == 1:
|
588
|
+
s = 'one option'
|
589
|
+
else:
|
590
|
+
s = '%d options' % (expected_n_options, )
|
591
|
+
|
592
|
+
if operator(n_options, expected_n_options):
|
593
|
+
warn('"%s" annotation %s %s, %s given' % (ann_name, message, s, t), position)
|
594
|
+
|
595
|
+
def _validate_annotation(self, position, ann_name, options, choices=None,
|
596
|
+
exact_n_options=None, min_n_options=None, max_n_options=None):
|
597
|
+
'''
|
598
|
+
Validate an annotation.
|
599
|
+
|
600
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
601
|
+
containing the annotation to be validated
|
602
|
+
:param ann_name: name of the annotation holding the options to validate
|
603
|
+
:param options: annotation options to be validated
|
604
|
+
:param choices: an iterable of allowed option names or :const:`None` to skip this test
|
605
|
+
:param exact_n_options: exact number of expected options or :const:`None` to skip this test
|
606
|
+
:param min_n_options: minimum number of expected options or :const:`None` to skip this test
|
607
|
+
:param max_n_options: maximum number of expected options or :const:`None` to skip this test
|
608
|
+
'''
|
609
|
+
|
610
|
+
n_options = len(options)
|
611
|
+
|
612
|
+
if exact_n_options is not None:
|
613
|
+
self._validate_options(position,
|
614
|
+
ann_name, n_options, exact_n_options, ne, 'needs')
|
615
|
+
|
616
|
+
if min_n_options is not None:
|
617
|
+
self._validate_options(position,
|
618
|
+
ann_name, n_options, min_n_options, lt, 'takes at least')
|
619
|
+
|
620
|
+
if max_n_options is not None:
|
621
|
+
self._validate_options(position,
|
622
|
+
ann_name, n_options, max_n_options, gt, 'takes at most')
|
623
|
+
|
624
|
+
if options and choices is not None:
|
625
|
+
option = options[0]
|
626
|
+
if option not in choices:
|
627
|
+
warn('invalid "%s" annotation option: "%s"' % (ann_name, option), position)
|
628
|
+
|
629
|
+
def _do_validate_allow_none(self, position, ann_name, options):
|
630
|
+
'''
|
631
|
+
Validate the ``(allow-none)`` annotation.
|
632
|
+
|
633
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
634
|
+
containing the annotation to be validated
|
635
|
+
:param ann_name: name of the annotation holding the options to validate
|
636
|
+
:param options: annotation options held by the annotation
|
637
|
+
'''
|
638
|
+
|
639
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
640
|
+
|
641
|
+
def _do_validate_array(self, position, ann_name, options):
|
642
|
+
'''
|
643
|
+
Validate the ``(array)`` annotation.
|
644
|
+
|
645
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
646
|
+
containing the annotation to be validated
|
647
|
+
:param ann_name: name of the annotation holding the options to validate
|
648
|
+
:param options: annotation options held by the annotation
|
649
|
+
'''
|
650
|
+
|
651
|
+
if len(options) == 0:
|
478
652
|
return
|
479
653
|
|
480
|
-
for
|
481
|
-
if
|
654
|
+
for option, value in options.items():
|
655
|
+
if option in [OPT_ARRAY_ZERO_TERMINATED, OPT_ARRAY_FIXED_SIZE]:
|
482
656
|
try:
|
483
|
-
int(
|
657
|
+
int(value)
|
484
658
|
except (TypeError, ValueError):
|
485
|
-
if
|
486
|
-
|
487
|
-
|
488
|
-
name, ),
|
489
|
-
positions=self.position)
|
659
|
+
if value is None:
|
660
|
+
warn('"%s" annotation option "%s" needs a value' % (ann_name, option),
|
661
|
+
position)
|
490
662
|
else:
|
491
|
-
|
492
|
-
|
493
|
-
|
494
|
-
|
495
|
-
|
496
|
-
|
497
|
-
|
498
|
-
'array option length needs a value',
|
499
|
-
positions=self.position)
|
663
|
+
warn('invalid "%s" annotation option "%s" value "%s", must be an integer' %
|
664
|
+
(ann_name, option, value),
|
665
|
+
position)
|
666
|
+
elif option == OPT_ARRAY_LENGTH:
|
667
|
+
if value is None:
|
668
|
+
warn('"%s" annotation option "length" needs a value' % (ann_name, ),
|
669
|
+
position)
|
500
670
|
else:
|
501
|
-
|
502
|
-
|
503
|
-
name, ), self.position)
|
504
|
-
|
505
|
-
def _validate_closure(self, option, value):
|
506
|
-
if value is not None and value.length() > 1:
|
507
|
-
message.warn(
|
508
|
-
'closure takes at most 1 value, %d given' % (
|
509
|
-
value.length(), ), self.position)
|
510
|
-
|
511
|
-
def _validate_element_type(self, option, value):
|
512
|
-
self._validate_option(option, value, required=True)
|
513
|
-
if value is None:
|
514
|
-
message.warn(
|
515
|
-
'element-type takes at least one value, none given',
|
516
|
-
self.position)
|
517
|
-
return
|
518
|
-
if value.length() > 2:
|
519
|
-
message.warn(
|
520
|
-
'element-type takes at most 2 values, %d given' % (
|
521
|
-
value.length(), ), self.position)
|
522
|
-
return
|
671
|
+
warn('invalid "%s" annotation option: "%s"' % (ann_name, option),
|
672
|
+
position)
|
523
673
|
|
524
|
-
def
|
525
|
-
|
526
|
-
|
527
|
-
if value.length() > 1:
|
528
|
-
message.warn(
|
529
|
-
'out annotation takes at most 1 value, %d given' % (
|
530
|
-
value.length(), ), self.position)
|
531
|
-
return
|
532
|
-
value_str = value.one()
|
533
|
-
if value_str not in [OPT_OUT_CALLEE_ALLOCATES,
|
534
|
-
OPT_OUT_CALLER_ALLOCATES]:
|
535
|
-
message.warn("out annotation value is invalid: %r" % (
|
536
|
-
value_str, ), self.position)
|
537
|
-
return
|
674
|
+
def _do_validate_attributes(self, position, ann_name, options):
|
675
|
+
'''
|
676
|
+
Validate the ``(attributes)`` annotation.
|
538
677
|
|
539
|
-
|
540
|
-
|
541
|
-
|
542
|
-
|
543
|
-
|
544
|
-
for k, v in value.all().items()))
|
545
|
-
return fmt % (option, value)
|
546
|
-
else:
|
547
|
-
return fmt2 % (option, )
|
548
|
-
annotations = []
|
549
|
-
for option, value in self.options.items():
|
550
|
-
annotations.append(
|
551
|
-
serialize_one(option, value, '(%s %s)', '(%s)'))
|
552
|
-
if annotations:
|
553
|
-
return ' '.join(annotations) + ': '
|
554
|
-
else:
|
555
|
-
return self.value
|
678
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
679
|
+
containing the annotation to be validated
|
680
|
+
:param ann_name: name of the annotation holding the options to validate
|
681
|
+
:param options: annotation options to validate
|
682
|
+
'''
|
556
683
|
|
557
|
-
|
558
|
-
|
684
|
+
# The 'attributes' annotation allows free form annotations.
|
685
|
+
pass
|
559
686
|
|
560
|
-
def
|
561
|
-
|
562
|
-
|
563
|
-
self.comment or '')
|
687
|
+
def _do_validate_closure(self, position, ann_name, options):
|
688
|
+
'''
|
689
|
+
Validate the ``(closure)`` annotation.
|
564
690
|
|
565
|
-
|
566
|
-
|
567
|
-
|
568
|
-
|
569
|
-
|
691
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
692
|
+
containing the annotation to be validated
|
693
|
+
:param ann_name: name of the annotation holding the options to validate
|
694
|
+
:param options: annotation options to validate
|
695
|
+
'''
|
570
696
|
|
571
|
-
|
572
|
-
if option == OPT_ALLOW_NONE:
|
573
|
-
self._validate_option(option, value, n_params=0)
|
574
|
-
elif option == OPT_ARRAY:
|
575
|
-
self._validate_array(option, value)
|
576
|
-
elif option == OPT_ATTRIBUTE:
|
577
|
-
self._validate_option(option, value, n_params=2)
|
578
|
-
elif option == OPT_CLOSURE:
|
579
|
-
self._validate_closure(option, value)
|
580
|
-
elif option == OPT_DESTROY:
|
581
|
-
self._validate_option(option, value, n_params=1)
|
582
|
-
elif option == OPT_ELEMENT_TYPE:
|
583
|
-
self._validate_element_type(option, value)
|
584
|
-
elif option == OPT_FOREIGN:
|
585
|
-
self._validate_option(option, value, n_params=0)
|
586
|
-
elif option == OPT_IN:
|
587
|
-
self._validate_option(option, value, n_params=0)
|
588
|
-
elif option in [OPT_INOUT, OPT_INOUT_ALT]:
|
589
|
-
self._validate_option(option, value, n_params=0)
|
590
|
-
elif option == OPT_OUT:
|
591
|
-
self._validate_out(option, value)
|
592
|
-
elif option == OPT_SCOPE:
|
593
|
-
self._validate_option(
|
594
|
-
option, value, required=True,
|
595
|
-
n_params=1,
|
596
|
-
choices=[OPT_SCOPE_ASYNC,
|
597
|
-
OPT_SCOPE_CALL,
|
598
|
-
OPT_SCOPE_NOTIFIED])
|
599
|
-
elif option == OPT_SKIP:
|
600
|
-
self._validate_option(option, value, n_params=0)
|
601
|
-
elif option == OPT_TRANSFER:
|
602
|
-
self._validate_option(
|
603
|
-
option, value, required=True,
|
604
|
-
n_params=1,
|
605
|
-
choices=[OPT_TRANSFER_FULL,
|
606
|
-
OPT_TRANSFER_CONTAINER,
|
607
|
-
OPT_TRANSFER_NONE,
|
608
|
-
OPT_TRANSFER_FLOATING])
|
609
|
-
elif option == OPT_TYPE:
|
610
|
-
self._validate_option(option, value, required=True,
|
611
|
-
n_params=1)
|
612
|
-
elif option == OPT_CONSTRUCTOR:
|
613
|
-
self._validate_option(option, value, n_params=0)
|
614
|
-
elif option == OPT_METHOD:
|
615
|
-
self._validate_option(option, value, n_params=0)
|
616
|
-
else:
|
617
|
-
message.warn('invalid annotation option: %s' % (option, ),
|
618
|
-
self.position)
|
697
|
+
self._validate_annotation(position, ann_name, options, max_n_options=1)
|
619
698
|
|
699
|
+
def _do_validate_constructor(self, position, ann_name, options):
|
700
|
+
'''
|
701
|
+
Validate the ``(constructor)`` annotation.
|
702
|
+
|
703
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
704
|
+
containing the annotation to be validated
|
705
|
+
:param ann_name: name of the annotation holding the options to validate
|
706
|
+
:param options: annotation options to validate
|
707
|
+
'''
|
708
|
+
|
709
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
620
710
|
|
621
|
-
|
622
|
-
|
623
|
-
|
624
|
-
|
711
|
+
def _do_validate_destroy(self, position, ann_name, options):
|
712
|
+
'''
|
713
|
+
Validate the ``(destroy)`` annotation.
|
714
|
+
|
715
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
716
|
+
containing the annotation to be validated
|
717
|
+
:param ann_name: name of the annotation holding the options to validate
|
718
|
+
:param options: annotation options to validate
|
719
|
+
'''
|
720
|
+
|
721
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
722
|
+
|
723
|
+
def _do_validate_element_type(self, position, ann_name, options):
|
724
|
+
'''
|
725
|
+
Validate the ``(element)`` annotation.
|
726
|
+
|
727
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
728
|
+
containing the annotation to be validated
|
729
|
+
:param ann_name: name of the annotation holding the options to validate
|
730
|
+
:param options: annotation options to validate
|
731
|
+
'''
|
732
|
+
|
733
|
+
self._validate_annotation(position, ann_name, options, min_n_options=1, max_n_options=2)
|
734
|
+
|
735
|
+
def _do_validate_foreign(self, position, ann_name, options):
|
736
|
+
'''
|
737
|
+
Validate the ``(foreign)`` annotation.
|
738
|
+
|
739
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
740
|
+
containing the annotation to be validated
|
741
|
+
:param ann_name: name of the annotation holding the options to validate
|
742
|
+
:param options: annotation options to validate
|
743
|
+
'''
|
744
|
+
|
745
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
746
|
+
|
747
|
+
def _do_validate_get_value_func(self, position, ann_name, options):
|
748
|
+
'''
|
749
|
+
Validate the ``(value-func)`` annotation.
|
750
|
+
|
751
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
752
|
+
containing the annotation to be validated
|
753
|
+
:param ann_name: name of the annotation holding the options to validate
|
754
|
+
:param options: annotation options to validate
|
755
|
+
'''
|
756
|
+
|
757
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
758
|
+
|
759
|
+
def _do_validate_in(self, position, ann_name, options):
|
760
|
+
'''
|
761
|
+
Validate the ``(in)`` annotation.
|
762
|
+
|
763
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
764
|
+
containing the annotation to be validated
|
765
|
+
:param ann_name: name of the annotation holding the options to validate
|
766
|
+
:param options: annotation options to validate
|
767
|
+
'''
|
768
|
+
|
769
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
770
|
+
|
771
|
+
def _do_validate_inout(self, position, ann_name, options):
|
772
|
+
'''
|
773
|
+
Validate the ``(in-out)`` annotation.
|
774
|
+
|
775
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
776
|
+
containing the annotation to be validated
|
777
|
+
:param ann_name: name of the annotation holding the options to validate
|
778
|
+
:param options: annotation options to validate
|
779
|
+
'''
|
780
|
+
|
781
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
782
|
+
|
783
|
+
def _do_validate_method(self, position, ann_name, options):
|
784
|
+
'''
|
785
|
+
Validate the ``(method)`` annotation.
|
786
|
+
|
787
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
788
|
+
containing the annotation to be validated
|
789
|
+
:param ann_name: name of the annotation holding the options to validate
|
790
|
+
:param options: annotation options to validate
|
791
|
+
'''
|
792
|
+
|
793
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
794
|
+
|
795
|
+
def _do_validate_nullable(self, position, ann_name, options):
|
796
|
+
'''
|
797
|
+
Validate the ``(nullable)`` annotation.
|
798
|
+
|
799
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
800
|
+
containing the annotation to be validated
|
801
|
+
:param ann_name: name of the annotation holding the options to validate
|
802
|
+
:param options: annotation options held by the annotation
|
803
|
+
'''
|
804
|
+
|
805
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
806
|
+
|
807
|
+
def _do_validate_optional(self, position, ann_name, options):
|
808
|
+
'''
|
809
|
+
Validate the ``(optional)`` annotation.
|
810
|
+
|
811
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
812
|
+
containing the annotation to be validated
|
813
|
+
:param ann_name: name of the annotation holding the options to validate
|
814
|
+
:param options: annotation options held by the annotation
|
815
|
+
'''
|
816
|
+
|
817
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
818
|
+
|
819
|
+
def _do_validate_out(self, position, ann_name, options):
|
820
|
+
'''
|
821
|
+
Validate the ``(out)`` annotation.
|
822
|
+
|
823
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
824
|
+
containing the annotation to be validated
|
825
|
+
:param ann_name: name of the annotation holding the options to validate
|
826
|
+
:param options: annotation options to validate
|
827
|
+
'''
|
828
|
+
|
829
|
+
self._validate_annotation(position, ann_name, options, max_n_options=1,
|
830
|
+
choices=OUT_OPTIONS)
|
831
|
+
|
832
|
+
def _do_validate_ref_func(self, position, ann_name, options):
|
833
|
+
'''
|
834
|
+
Validate the ``(ref-func)`` annotation.
|
835
|
+
|
836
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
837
|
+
containing the annotation to be validated
|
838
|
+
:param ann_name: name of the annotation holding the options to validate
|
839
|
+
:param options: annotation options to validate
|
840
|
+
'''
|
841
|
+
|
842
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
843
|
+
|
844
|
+
def _do_validate_rename_to(self, position, ann_name, options):
|
845
|
+
'''
|
846
|
+
Validate the ``(rename-to)`` annotation.
|
847
|
+
|
848
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
849
|
+
containing the annotation to be validated
|
850
|
+
:param ann_name: name of the annotation holding the options to validate
|
851
|
+
:param options: annotation options to validate
|
852
|
+
'''
|
853
|
+
|
854
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
855
|
+
|
856
|
+
def _do_validate_scope(self, position, ann_name, options):
|
857
|
+
'''
|
858
|
+
Validate the ``(scope)`` annotation.
|
859
|
+
|
860
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
861
|
+
containing the annotation to be validated
|
862
|
+
:param ann_name: name of the annotation holding the options to validate
|
863
|
+
:param options: annotation options to validate
|
864
|
+
'''
|
865
|
+
|
866
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1,
|
867
|
+
choices=SCOPE_OPTIONS)
|
868
|
+
|
869
|
+
def _do_validate_set_value_func(self, position, ann_name, options):
|
870
|
+
'''
|
871
|
+
Validate the ``(value-func)`` annotation.
|
872
|
+
|
873
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
874
|
+
containing the annotation to be validated
|
875
|
+
:param ann_name: name of the annotation holding the options to validate
|
876
|
+
:param options: annotation options to validate
|
877
|
+
'''
|
878
|
+
|
879
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
880
|
+
|
881
|
+
def _do_validate_skip(self, position, ann_name, options):
|
882
|
+
'''
|
883
|
+
Validate the ``(skip)`` annotation.
|
884
|
+
|
885
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
886
|
+
containing the annotation to be validated
|
887
|
+
:param ann_name: name of the annotation holding the options to validate
|
888
|
+
:param options: annotation options to validate
|
889
|
+
'''
|
890
|
+
|
891
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=0)
|
892
|
+
|
893
|
+
def _do_validate_transfer(self, position, ann_name, options):
|
894
|
+
'''
|
895
|
+
Validate the ``(transfer)`` annotation.
|
896
|
+
|
897
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
898
|
+
containing the annotation to be validated
|
899
|
+
:param ann_name: name of the annotation holding the options to validate
|
900
|
+
:param options: annotation options to validate
|
901
|
+
'''
|
902
|
+
|
903
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1,
|
904
|
+
choices=TRANSFER_OPTIONS)
|
905
|
+
|
906
|
+
def _do_validate_type(self, position, ann_name, options):
|
907
|
+
'''
|
908
|
+
Validate the ``(type)`` annotation.
|
909
|
+
|
910
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
911
|
+
containing the annotation to be validated
|
912
|
+
:param ann_name: name of the annotation holding the options to validate
|
913
|
+
:param options: annotation options to validate
|
914
|
+
'''
|
915
|
+
|
916
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
917
|
+
|
918
|
+
def _do_validate_unref_func(self, position, ann_name, options):
|
919
|
+
'''
|
920
|
+
Validate the ``(unref-func)`` annotation.
|
921
|
+
|
922
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
923
|
+
containing the annotation to be validated
|
924
|
+
:param ann_name: name of the annotation holding the options to validate
|
925
|
+
:param options: annotation options to validate
|
926
|
+
'''
|
927
|
+
|
928
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
929
|
+
|
930
|
+
def _do_validate_value(self, position, ann_name, options):
|
931
|
+
'''
|
932
|
+
Validate the ``(value)`` annotation.
|
933
|
+
|
934
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
935
|
+
containing the annotation to be validated
|
936
|
+
:param ann_name: name of the annotation holding the options to validate
|
937
|
+
:param options: annotation options to validate
|
938
|
+
'''
|
939
|
+
|
940
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
941
|
+
|
942
|
+
def _do_validate_virtual(self, position, ann_name, options):
|
943
|
+
'''
|
944
|
+
Validate the ``(virtual)`` annotation.
|
945
|
+
|
946
|
+
:param position: :class:`giscanner.message.Position` of the line in the source file
|
947
|
+
containing the annotation to be validated
|
948
|
+
:param ann_name: name of the annotation holding the options to validate
|
949
|
+
:param options: annotation options to validate
|
950
|
+
'''
|
951
|
+
|
952
|
+
self._validate_annotation(position, ann_name, options, exact_n_options=1)
|
953
|
+
|
954
|
+
|
955
|
+
class GtkDocParameter(GtkDocAnnotatable):
|
956
|
+
'''
|
957
|
+
Represents a GTK-Doc parameter part.
|
958
|
+
'''
|
959
|
+
|
960
|
+
__slots__ = ('name', 'description')
|
961
|
+
|
962
|
+
valid_annotations = (ANN_ALLOW_NONE, ANN_ARRAY, ANN_ATTRIBUTES, ANN_CLOSURE, ANN_DESTROY,
|
963
|
+
ANN_ELEMENT_TYPE, ANN_IN, ANN_INOUT, ANN_OUT, ANN_SCOPE, ANN_SKIP,
|
964
|
+
ANN_TRANSFER, ANN_TYPE, ANN_OPTIONAL, ANN_NULLABLE)
|
965
|
+
|
966
|
+
def __init__(self, name, position=None):
|
967
|
+
GtkDocAnnotatable.__init__(self, position)
|
968
|
+
|
969
|
+
#: Parameter name.
|
970
|
+
self.name = name
|
971
|
+
|
972
|
+
#: Parameter description or :const:`None`.
|
973
|
+
self.description = None
|
625
974
|
|
626
975
|
def __repr__(self):
|
627
|
-
return '<
|
976
|
+
return '<GtkDocParameter %r %r>' % (self.name, self.annotations)
|
977
|
+
|
978
|
+
|
979
|
+
class GtkDocTag(GtkDocAnnotatable):
|
980
|
+
'''
|
981
|
+
Represents a GTK-Doc tag part.
|
982
|
+
'''
|
628
983
|
|
629
|
-
|
630
|
-
for key, value in self.values:
|
631
|
-
if key == item:
|
632
|
-
return value
|
633
|
-
raise KeyError
|
984
|
+
__slots__ = ('name', 'value', 'description')
|
634
985
|
|
635
|
-
|
636
|
-
|
986
|
+
valid_annotations = (ANN_ALLOW_NONE, ANN_ARRAY, ANN_ATTRIBUTES, ANN_ELEMENT_TYPE, ANN_SKIP,
|
987
|
+
ANN_TRANSFER, ANN_TYPE, ANN_NULLABLE, ANN_OPTIONAL)
|
637
988
|
|
638
|
-
def
|
639
|
-
|
989
|
+
def __init__(self, name, position=None):
|
990
|
+
GtkDocAnnotatable.__init__(self, position)
|
640
991
|
|
641
|
-
|
642
|
-
self.
|
992
|
+
#: Tag name.
|
993
|
+
self.name = name
|
643
994
|
|
644
|
-
|
645
|
-
|
646
|
-
if key == item:
|
647
|
-
return value
|
648
|
-
return default
|
995
|
+
#: Tag value or :const:`None`.
|
996
|
+
self.value = None
|
649
997
|
|
650
|
-
|
651
|
-
|
652
|
-
if key == item:
|
653
|
-
yield value
|
998
|
+
#: Tag description or :const:`None`.
|
999
|
+
self.description = None
|
654
1000
|
|
655
|
-
def
|
656
|
-
return
|
1001
|
+
def __repr__(self):
|
1002
|
+
return '<GtkDocTag %r %r>' % (self.name, self.annotations)
|
657
1003
|
|
658
1004
|
|
659
|
-
class
|
1005
|
+
class GtkDocCommentBlock(GtkDocAnnotatable):
|
1006
|
+
'''
|
1007
|
+
Represents a GTK-Doc comment block.
|
1008
|
+
'''
|
660
1009
|
|
661
|
-
|
662
|
-
|
663
|
-
|
664
|
-
|
665
|
-
|
666
|
-
|
667
|
-
|
668
|
-
|
669
|
-
|
670
|
-
|
671
|
-
|
672
|
-
|
673
|
-
|
674
|
-
|
675
|
-
|
676
|
-
|
1010
|
+
__slots__ = ('code_before', 'code_after', 'indentation',
|
1011
|
+
'name', 'params', 'description', 'tags')
|
1012
|
+
|
1013
|
+
#: Valid annotation names for the GTK-Doc comment block identifier part.
|
1014
|
+
valid_annotations = (ANN_ATTRIBUTES, ANN_CONSTRUCTOR, ANN_FOREIGN, ANN_GET_VALUE_FUNC,
|
1015
|
+
ANN_METHOD, ANN_REF_FUNC, ANN_RENAME_TO, ANN_SET_VALUE_FUNC,
|
1016
|
+
ANN_SKIP, ANN_TRANSFER, ANN_TYPE, ANN_UNREF_FUNC, ANN_VALUE, ANN_VFUNC)
|
1017
|
+
|
1018
|
+
def __init__(self, name, position=None):
|
1019
|
+
GtkDocAnnotatable.__init__(self, position)
|
1020
|
+
|
1021
|
+
#: Code preceding the GTK-Doc comment block start token ("``/**``"), if any.
|
1022
|
+
self.code_before = None
|
1023
|
+
|
1024
|
+
#: Code following the GTK-Doc comment block end token ("``*/``"), if any.
|
1025
|
+
self.code_after = None
|
1026
|
+
|
1027
|
+
#: List of indentation levels (preceding the "``*``") for all lines in the comment
|
1028
|
+
#: block's source text.
|
1029
|
+
self.indentation = []
|
1030
|
+
|
1031
|
+
#: Identifier name.
|
1032
|
+
self.name = name
|
1033
|
+
|
1034
|
+
#: Ordered dictionary mapping parameter names to :class:`GtkDocParameter` instances
|
1035
|
+
#: applied to this :class:`GtkDocCommentBlock`.
|
1036
|
+
self.params = OrderedDict()
|
1037
|
+
|
1038
|
+
#: The GTK-Doc comment block description part.
|
1039
|
+
self.description = None
|
1040
|
+
|
1041
|
+
#: Ordered dictionary mapping tag names to :class:`GtkDocTag` instances
|
1042
|
+
#: applied to this :class:`GtkDocCommentBlock`.
|
1043
|
+
self.tags = OrderedDict()
|
1044
|
+
|
1045
|
+
def __cmp__(self, other):
|
1046
|
+
# Note: This is used by g-ir-annotation-tool, which does a ``sorted(blocks.values())``,
|
1047
|
+
# meaning that keeping this around makes update-glib-annotations.py patches
|
1048
|
+
# easier to review.
|
1049
|
+
return cmp(self.name, other.name)
|
677
1050
|
|
678
1051
|
def __repr__(self):
|
679
|
-
return '<
|
680
|
-
|
681
|
-
def
|
682
|
-
|
683
|
-
|
684
|
-
|
685
|
-
|
686
|
-
|
687
|
-
|
688
|
-
|
689
|
-
|
690
|
-
|
691
|
-
|
692
|
-
|
693
|
-
|
694
|
-
|
695
|
-
class
|
696
|
-
|
697
|
-
|
698
|
-
|
699
|
-
|
700
|
-
|
701
|
-
|
702
|
-
|
703
|
-
|
704
|
-
|
705
|
-
|
706
|
-
|
707
|
-
|
708
|
-
|
709
|
-
|
710
|
-
|
711
|
-
|
712
|
-
|
713
|
-
|
714
|
-
|
715
|
-
|
716
|
-
|
717
|
-
|
718
|
-
|
719
|
-
|
720
|
-
|
721
|
-
|
722
|
-
|
723
|
-
The order in which the different parts have to be specified is important::
|
724
|
-
|
725
|
-
- There has to be exactly 1 `identifier` part on the first line of the
|
726
|
-
comment block which consists of:
|
727
|
-
* an `identifier_name` field
|
728
|
-
* an optional `annotations` field
|
729
|
-
- Followed by 0 or more `parameters` parts, each consisting of:
|
730
|
-
* a `parameter_name` field
|
731
|
-
* an optional `annotations` field
|
732
|
-
* an optional `description` field
|
733
|
-
- Followed by at least 1 empty line signaling the beginning of
|
734
|
-
the `comment_block_description` part
|
735
|
-
- Followed by an optional `comment block description` part.
|
736
|
-
- Followed by 0 or more `tag` parts, each consisting of:
|
737
|
-
* a `tag_name` field
|
738
|
-
* an optional `annotations` field
|
739
|
-
* an optional `description` field
|
740
|
-
|
741
|
-
Additionally, the following restrictions are in effect::
|
742
|
-
|
743
|
-
- Parts can optionally be separated by an empty line, except between
|
744
|
-
the `parameter` parts and the `comment block description` part where
|
745
|
-
an empty line is required (see above).
|
746
|
-
- Parts and fields cannot span multiple lines, except for
|
747
|
-
`parameter descriptions`, `tag descriptions` and the
|
748
|
-
`comment_block_description` fields.
|
749
|
-
- `parameter descriptions` fields can not span multiple paragraphs.
|
750
|
-
- `tag descriptions` and `comment block description` fields can
|
751
|
-
span multiple paragraphs.
|
752
|
-
|
753
|
-
.. NOTE:: :class:`AnnotationParser` functionality is heavily based on gtkdoc-mkdb's
|
1052
|
+
return '<GtkDocCommentBlock %r %r>' % (self.name, self.annotations)
|
1053
|
+
|
1054
|
+
def validate(self):
|
1055
|
+
'''
|
1056
|
+
Validate annotations applied to the :class:`GtkDocCommentBlock` identifier, parameters
|
1057
|
+
and tags.
|
1058
|
+
'''
|
1059
|
+
GtkDocAnnotatable.validate(self)
|
1060
|
+
|
1061
|
+
for param in self.params.values():
|
1062
|
+
param.validate()
|
1063
|
+
|
1064
|
+
for tag in self.tags.values():
|
1065
|
+
tag.validate()
|
1066
|
+
|
1067
|
+
|
1068
|
+
#: Result object returned by :class:`GtkDocCommentBlockParser`._parse_annotations()
|
1069
|
+
_ParseAnnotationsResult = namedtuple('Result', ['success', 'annotations', 'start_pos', 'end_pos'])
|
1070
|
+
|
1071
|
+
#: Result object returned by :class:`GtkDocCommentBlockParser`._parse_fields()
|
1072
|
+
_ParseFieldsResult = namedtuple('Result', ['success', 'annotations', 'description'])
|
1073
|
+
|
1074
|
+
|
1075
|
+
class GtkDocCommentBlockParser(object):
|
1076
|
+
'''
|
1077
|
+
Parse GTK-Doc comment blocks into a parse tree built out of :class:`GtkDocCommentBlock`,
|
1078
|
+
:class:`GtkDocParameter`, :class:`GtkDocTag` and :class:`GtkDocAnnotations`
|
1079
|
+
objects. This parser tries to accept malformed input whenever possible and does
|
1080
|
+
not cause the process to exit on syntax errors. It does however emit:
|
1081
|
+
|
1082
|
+
* warning messages at the slightest indication of recoverable malformed input and
|
1083
|
+
* error messages for unrecoverable malformed input
|
1084
|
+
|
1085
|
+
whenever possible. Recoverable, in this context, means that we can serialize the
|
1086
|
+
:class:`GtkDocCommentBlock` instance using a :class:`GtkDocCommentBlockWriter` without
|
1087
|
+
information being lost. It is usually a good idea to heed these warning and error messages
|
1088
|
+
as malformed input can result in both:
|
1089
|
+
|
1090
|
+
* invalid GTK-Doc output (HTML, pdf, ...) when the comment blocks are parsed
|
1091
|
+
with GTK-Doc's gtkdoc-mkdb
|
1092
|
+
* unexpected introspection behavior, for example missing parameters in the
|
1093
|
+
generated .gir and .typelib files
|
1094
|
+
|
1095
|
+
.. NOTE:: :class:`GtkDocCommentBlockParser` functionality is heavily based on gtkdoc-mkdb's
|
754
1096
|
`ScanSourceFile()`_ function and is currently in sync with GTK-Doc
|
755
1097
|
commit `47abcd5`_.
|
756
1098
|
|
757
|
-
.. _GTK-Doc's documentation:
|
758
|
-
http://developer.gnome.org/gtk-doc-manual/1.18/documenting.html.en
|
759
1099
|
.. _ScanSourceFile():
|
760
|
-
|
761
|
-
.. _47abcd5:
|
762
|
-
|
1100
|
+
http://git.gnome.org/browse/gtk-doc/tree/gtkdoc-mkdb.in#n3722
|
1101
|
+
.. _47abcd5:
|
1102
|
+
https://git.gnome.org/browse/gtk-doc/commit/?id=47abcd53b8489ebceec9e394676512a181c1f1f6
|
1103
|
+
'''
|
763
1104
|
|
764
|
-
def
|
765
|
-
|
766
|
-
|
1105
|
+
def parse_comment_blocks(self, comments):
|
1106
|
+
'''
|
1107
|
+
Parse multiple GTK-Doc comment blocks.
|
767
1108
|
|
768
|
-
:param comments:
|
769
|
-
:returns: a dictionary mapping identifier names to :class:`
|
770
|
-
|
1109
|
+
:param comments: an iterable of ``(comment, filename, lineno)`` tuples
|
1110
|
+
:returns: a dictionary mapping identifier names to :class:`GtkDocCommentBlock` objects
|
1111
|
+
'''
|
771
1112
|
|
772
1113
|
comment_blocks = {}
|
773
1114
|
|
774
|
-
for comment in comments:
|
1115
|
+
for (comment, filename, lineno) in comments:
|
775
1116
|
try:
|
776
|
-
comment_block = self.parse_comment_block(comment)
|
1117
|
+
comment_block = self.parse_comment_block(comment, filename, lineno)
|
777
1118
|
except Exception:
|
778
|
-
|
779
|
-
|
780
|
-
|
1119
|
+
error('unrecoverable parse error, please file a GObject-Introspection bug'
|
1120
|
+
'report including the complete comment block at the indicated location.',
|
1121
|
+
Position(filename, lineno))
|
781
1122
|
continue
|
782
1123
|
|
783
1124
|
if comment_block is not None:
|
784
|
-
# Note: previous versions of this parser did not check
|
785
|
-
#
|
786
|
-
#
|
787
|
-
#
|
788
|
-
# Keep this behavior for backwards compatibility, but
|
789
|
-
# emit a warning.
|
1125
|
+
# Note: previous versions of this parser did not check if an identifier was
|
1126
|
+
# already stored in comment_blocks, so when different comment blocks where
|
1127
|
+
# encountered documenting the same identifier the last comment block seen
|
1128
|
+
# "wins". Keep this behavior for backwards compatibility, but emit a warning.
|
790
1129
|
if comment_block.name in comment_blocks:
|
791
|
-
|
792
|
-
|
793
|
-
|
1130
|
+
firstseen = comment_blocks[comment_block.name]
|
1131
|
+
path = os.path.dirname(firstseen.position.filename)
|
1132
|
+
warn('multiple comment blocks documenting \'%s:\' identifier '
|
1133
|
+
'(already seen at %s).' %
|
1134
|
+
(comment_block.name, firstseen.position.format(path)),
|
1135
|
+
comment_block.position)
|
794
1136
|
|
795
1137
|
comment_blocks[comment_block.name] = comment_block
|
796
1138
|
|
797
1139
|
return comment_blocks
|
798
1140
|
|
799
|
-
def parse_comment_block(self, comment):
|
800
|
-
|
801
|
-
|
802
|
-
|
803
|
-
:param comment: a (comment, filename, lineno) tuple
|
804
|
-
:returns: a :class:`DocBlock` object or ``None``
|
805
|
-
"""
|
1141
|
+
def parse_comment_block(self, comment, filename, lineno):
|
1142
|
+
'''
|
1143
|
+
Parse a single GTK-Doc comment block.
|
806
1144
|
|
807
|
-
comment
|
1145
|
+
:param comment: string representing the GTK-Doc comment block including it's
|
1146
|
+
start ("``/**``") and end ("``*/``") tokens.
|
1147
|
+
:param filename: source file name where the comment block originated from
|
1148
|
+
:param lineno: line number in the source file where the comment block starts
|
1149
|
+
:returns: a :class:`GtkDocCommentBlock` object or ``None``
|
1150
|
+
'''
|
808
1151
|
|
809
|
-
|
810
|
-
|
811
|
-
|
812
|
-
comment_lines =
|
1152
|
+
code_before = ''
|
1153
|
+
code_after = ''
|
1154
|
+
comment_block_pos = Position(filename, lineno)
|
1155
|
+
comment_lines = re.sub(LINE_BREAK_RE, '\n', comment).split('\n')
|
1156
|
+
comment_lines_len = len(comment_lines)
|
813
1157
|
|
814
|
-
# Check for the start the comment block.
|
815
|
-
|
816
|
-
|
1158
|
+
# Check for the start of the comment block.
|
1159
|
+
result = COMMENT_BLOCK_START_RE.match(comment_lines[0])
|
1160
|
+
if result:
|
1161
|
+
# Skip single line comment blocks
|
1162
|
+
if comment_lines_len == 1:
|
1163
|
+
position = Position(filename, lineno)
|
1164
|
+
marker = ' ' * result.end('code') + '^'
|
1165
|
+
error('Skipping invalid GTK-Doc comment block:'
|
1166
|
+
'\n%s\n%s' % (comment_lines[0], marker),
|
1167
|
+
position)
|
1168
|
+
return None
|
1169
|
+
|
1170
|
+
code_before = result.group('code')
|
1171
|
+
comment = result.group('comment')
|
1172
|
+
|
1173
|
+
if code_before:
|
1174
|
+
position = Position(filename, lineno)
|
1175
|
+
marker = ' ' * result.end('code') + '^'
|
1176
|
+
warn('GTK-Doc comment block start token "/**" should '
|
1177
|
+
'not be preceded by code:\n%s\n%s' % (comment_lines[0], marker),
|
1178
|
+
position)
|
1179
|
+
|
1180
|
+
if comment:
|
1181
|
+
position = Position(filename, lineno)
|
1182
|
+
marker = ' ' * result.start('comment') + '^'
|
1183
|
+
warn('GTK-Doc comment block start token "/**" should '
|
1184
|
+
'not be followed by comment text:\n%s\n%s' % (comment_lines[0], marker),
|
1185
|
+
position)
|
1186
|
+
|
1187
|
+
comment_lines[0] = comment
|
1188
|
+
else:
|
1189
|
+
del comment_lines[0]
|
817
1190
|
else:
|
818
1191
|
# Not a GTK-Doc comment block.
|
819
1192
|
return None
|
820
1193
|
|
821
|
-
# Check for the end the comment block.
|
822
|
-
|
823
|
-
result = COMMENT_END_RE.match(line)
|
1194
|
+
# Check for the end of the comment block.
|
1195
|
+
result = COMMENT_BLOCK_END_RE.match(comment_lines[-1])
|
824
1196
|
if result:
|
825
|
-
|
826
|
-
|
827
|
-
|
828
|
-
position =
|
829
|
-
marker = ' ' * result.end('
|
830
|
-
|
831
|
-
|
832
|
-
|
1197
|
+
code_after = result.group('code')
|
1198
|
+
comment = result.group('comment')
|
1199
|
+
if code_after:
|
1200
|
+
position = Position(filename, lineno + comment_lines_len - 1)
|
1201
|
+
marker = ' ' * result.end('code') + '^'
|
1202
|
+
warn('GTK-Doc comment block end token "*/" should '
|
1203
|
+
'not be followed by code:\n%s\n%s' % (comment_lines[-1], marker),
|
1204
|
+
position)
|
1205
|
+
|
1206
|
+
if comment:
|
1207
|
+
position = Position(filename, lineno + comment_lines_len - 1)
|
1208
|
+
marker = ' ' * result.end('comment') + '^'
|
1209
|
+
warn('GTK-Doc comment block end token "*/" should '
|
1210
|
+
'not be preceded by comment text:\n%s\n%s' % (comment_lines[-1], marker),
|
1211
|
+
position)
|
1212
|
+
|
1213
|
+
comment_lines[-1] = comment
|
833
1214
|
else:
|
834
1215
|
del comment_lines[-1]
|
835
1216
|
else:
|
836
1217
|
# Not a GTK-Doc comment block.
|
837
1218
|
return None
|
838
1219
|
|
839
|
-
# If we get this far, we
|
840
|
-
|
841
|
-
|
842
|
-
def _parse_comment_block(self, comment_lines, filename, lineno):
|
843
|
-
"""
|
844
|
-
Parses a single GTK-Doc comment block already stripped from its
|
845
|
-
comment start (/**) and comment end (*/) marker lines.
|
846
|
-
|
847
|
-
:param comment_lines: list of (line_offset, line) tuples representing a
|
848
|
-
GTK-Doc comment block already stripped from it's
|
849
|
-
start (/**) and end (*/) marker lines
|
850
|
-
:param filename: source file name where the comment block originated from
|
851
|
-
:param lineno: line in the source file where the comment block starts
|
852
|
-
:returns: a :class:`DocBlock` object or ``None``
|
853
|
-
|
854
|
-
.. NOTE:: If you are tempted to refactor this method and split it
|
855
|
-
further up (for example into _parse_identifier(), _parse_parameters(),
|
856
|
-
_parse_description(), _parse_tags() methods) then please resist the
|
857
|
-
urge. It is considered important that this method should be more or
|
858
|
-
less easily comparable with gtkdoc-mkdb's `ScanSourceFile()`_ function.
|
859
|
-
|
860
|
-
The different parsing steps are marked with a comment surrounded
|
861
|
-
by `#` characters in an attempt to make it clear what is going on.
|
862
|
-
|
863
|
-
.. _ScanSourceFile():
|
864
|
-
http://git.gnome.org/browse/gtk-doc/tree/gtkdoc-mkdb.in#n3722
|
865
|
-
"""
|
1220
|
+
# If we get this far, we must be inside something
|
1221
|
+
# that looks like a GTK-Doc comment block.
|
866
1222
|
comment_block = None
|
867
|
-
|
1223
|
+
identifier_warned = False
|
1224
|
+
block_indent = []
|
868
1225
|
line_indent = None
|
1226
|
+
part_indent = None
|
869
1227
|
in_part = None
|
870
|
-
|
871
|
-
current_param = None
|
872
|
-
current_tag = None
|
1228
|
+
current_part = None
|
873
1229
|
returns_seen = False
|
874
1230
|
|
875
|
-
for
|
876
|
-
|
1231
|
+
for line in comment_lines:
|
1232
|
+
lineno += 1
|
1233
|
+
position = Position(filename, lineno)
|
877
1234
|
|
878
1235
|
# Store the original line (without \n) and column offset
|
879
1236
|
# so we can generate meaningful warnings later on.
|
880
1237
|
original_line = line
|
881
1238
|
column_offset = 0
|
882
1239
|
|
883
|
-
#
|
1240
|
+
# Store indentation level of the comment (before the ' * ')
|
1241
|
+
result = INDENTATION_RE.match(line)
|
1242
|
+
block_indent.append(result.group('indentation'))
|
1243
|
+
|
1244
|
+
# Get rid of the ' * ' at the start of the line.
|
884
1245
|
result = COMMENT_ASTERISK_RE.match(line)
|
885
1246
|
if result:
|
1247
|
+
comment = result.group('comment')
|
1248
|
+
if comment:
|
1249
|
+
marker = ' ' * result.start('comment') + '^'
|
1250
|
+
error('invalid comment text:\n%s\n%s' %
|
1251
|
+
(original_line, marker),
|
1252
|
+
position)
|
1253
|
+
|
886
1254
|
column_offset = result.end(0)
|
887
1255
|
line = line[result.end(0):]
|
888
1256
|
|
889
|
-
# Store indentation level of the line.
|
890
|
-
result =
|
1257
|
+
# Store indentation level of the line (after the ' * ').
|
1258
|
+
result = INDENTATION_RE.match(line)
|
891
1259
|
line_indent = len(result.group('indentation').replace('\t', ' '))
|
892
1260
|
|
893
1261
|
####################################################################
|
894
1262
|
# Check for GTK-Doc comment block identifier.
|
895
1263
|
####################################################################
|
896
|
-
if
|
897
|
-
|
898
|
-
|
899
|
-
|
900
|
-
|
901
|
-
|
902
|
-
|
903
|
-
|
904
|
-
|
905
|
-
result = SYMBOL_RE.match(line)
|
906
|
-
if result:
|
907
|
-
identifier = IDENTIFIER_SYMBOL
|
908
|
-
identifier_name = '%s' % (result.group('symbol_name'), )
|
909
|
-
column = result.start('symbol_name') + column_offset
|
910
|
-
|
911
|
-
if not identifier:
|
1264
|
+
if comment_block is None:
|
1265
|
+
result = SECTION_RE.match(line)
|
1266
|
+
|
1267
|
+
if result:
|
1268
|
+
identifier_name = 'SECTION:%s' % (result.group('section_name'), )
|
1269
|
+
identifier_delimiter = None
|
1270
|
+
identifier_fields = None
|
1271
|
+
identifier_fields_start = None
|
1272
|
+
else:
|
912
1273
|
result = PROPERTY_RE.match(line)
|
1274
|
+
|
913
1275
|
if result:
|
914
|
-
identifier = IDENTIFIER_PROPERTY
|
915
1276
|
identifier_name = '%s:%s' % (result.group('class_name'),
|
916
1277
|
result.group('property_name'))
|
917
|
-
|
1278
|
+
identifier_delimiter = result.group('delimiter')
|
1279
|
+
identifier_fields = result.group('fields')
|
1280
|
+
identifier_fields_start = result.start('fields')
|
1281
|
+
else:
|
1282
|
+
result = SIGNAL_RE.match(line)
|
1283
|
+
|
1284
|
+
if result:
|
1285
|
+
identifier_name = '%s::%s' % (result.group('class_name'),
|
1286
|
+
result.group('signal_name'))
|
1287
|
+
identifier_delimiter = result.group('delimiter')
|
1288
|
+
identifier_fields = result.group('fields')
|
1289
|
+
identifier_fields_start = result.start('fields')
|
1290
|
+
else:
|
1291
|
+
result = SYMBOL_RE.match(line)
|
918
1292
|
|
919
|
-
|
920
|
-
|
921
|
-
|
922
|
-
|
923
|
-
|
924
|
-
result.group('signal_name'))
|
925
|
-
column = result.start('signal_name') + column_offset
|
1293
|
+
if result:
|
1294
|
+
identifier_name = '%s' % (result.group('symbol_name'), )
|
1295
|
+
identifier_delimiter = result.group('delimiter')
|
1296
|
+
identifier_fields = result.group('fields')
|
1297
|
+
identifier_fields_start = result.start('fields')
|
926
1298
|
|
927
|
-
if
|
1299
|
+
if result:
|
928
1300
|
in_part = PART_IDENTIFIER
|
929
1301
|
part_indent = line_indent
|
930
1302
|
|
931
|
-
comment_block =
|
932
|
-
comment_block.
|
933
|
-
|
934
|
-
|
935
|
-
|
936
|
-
|
937
|
-
|
938
|
-
|
939
|
-
|
940
|
-
|
941
|
-
|
942
|
-
|
943
|
-
|
944
|
-
|
1303
|
+
comment_block = GtkDocCommentBlock(identifier_name, comment_block_pos)
|
1304
|
+
comment_block.code_before = code_before
|
1305
|
+
comment_block.code_after = code_after
|
1306
|
+
|
1307
|
+
if identifier_fields:
|
1308
|
+
res = self._parse_annotations(position,
|
1309
|
+
column_offset + identifier_fields_start,
|
1310
|
+
original_line,
|
1311
|
+
identifier_fields)
|
1312
|
+
|
1313
|
+
if res.success:
|
1314
|
+
if identifier_fields[res.end_pos:].strip():
|
1315
|
+
# Not an identifier due to invalid trailing description field
|
1316
|
+
result = None
|
1317
|
+
in_part = None
|
1318
|
+
part_indent = None
|
1319
|
+
comment_block = None
|
1320
|
+
else:
|
1321
|
+
comment_block.annotations = res.annotations
|
1322
|
+
|
1323
|
+
if not identifier_delimiter and res.annotations:
|
1324
|
+
marker_position = column_offset + result.start('delimiter')
|
1325
|
+
marker = ' ' * marker_position + '^'
|
1326
|
+
warn('missing ":" at column %s:\n%s\n%s' %
|
1327
|
+
(marker_position + 1, original_line, marker),
|
1328
|
+
position)
|
945
1329
|
|
946
|
-
|
947
|
-
|
948
|
-
|
949
|
-
|
950
|
-
|
951
|
-
|
952
|
-
|
953
|
-
|
954
|
-
|
955
|
-
# descriptions will contain something matching an identifier
|
956
|
-
# pattern by accident.
|
957
|
-
marker = ' ' * column_offset + '^'
|
958
|
-
message.warn('ignoring unrecognized GTK-Doc comment block, identifier not '
|
959
|
-
'found:\n%s\n%s' % (original_line, marker),
|
960
|
-
position)
|
961
|
-
|
962
|
-
return None
|
1330
|
+
if not result:
|
1331
|
+
# Emit a single warning when the identifier is not found on the first line
|
1332
|
+
if not identifier_warned:
|
1333
|
+
identifier_warned = True
|
1334
|
+
marker = ' ' * column_offset + '^'
|
1335
|
+
error('identifier not found on the first line:\n%s\n%s' %
|
1336
|
+
(original_line, marker),
|
1337
|
+
position)
|
1338
|
+
continue
|
963
1339
|
|
964
1340
|
####################################################################
|
965
1341
|
# Check for comment block parameters.
|
966
1342
|
####################################################################
|
967
1343
|
result = PARAMETER_RE.match(line)
|
968
1344
|
if result:
|
969
|
-
param_name = result.group('parameter_name')
|
970
|
-
param_annotations = result.group('annotations')
|
971
|
-
param_description = result.group('description')
|
972
|
-
|
973
|
-
if in_part == PART_IDENTIFIER:
|
974
|
-
in_part = PART_PARAMETERS
|
975
|
-
|
976
1345
|
part_indent = line_indent
|
1346
|
+
param_name = result.group('parameter_name')
|
1347
|
+
param_name_lower = param_name.lower()
|
1348
|
+
param_fields = result.group('fields')
|
1349
|
+
param_fields_start = result.start('fields')
|
1350
|
+
marker = ' ' * (result.start('parameter_name') + column_offset) + '^'
|
1351
|
+
|
1352
|
+
if in_part not in [PART_IDENTIFIER, PART_PARAMETERS]:
|
1353
|
+
warn('"@%s" parameter unexpected at this location:\n%s\n%s' %
|
1354
|
+
(param_name, original_line, marker),
|
1355
|
+
position)
|
977
1356
|
|
978
|
-
|
979
|
-
column = result.start('parameter_name') + column_offset
|
980
|
-
marker = ' ' * column + '^'
|
981
|
-
message.warn("'@%s' parameter unexpected at this location:\n%s\n%s" %
|
982
|
-
(param_name, original_line, marker),
|
983
|
-
position)
|
1357
|
+
in_part = PART_PARAMETERS
|
984
1358
|
|
985
|
-
|
986
|
-
|
987
|
-
if param_name.lower() == TAG_RETURNS:
|
1359
|
+
if param_name_lower == TAG_RETURNS:
|
1360
|
+
# Deprecated return value as parameter instead of tag
|
988
1361
|
param_name = TAG_RETURNS
|
989
1362
|
|
990
1363
|
if not returns_seen:
|
991
1364
|
returns_seen = True
|
992
1365
|
else:
|
993
|
-
|
994
|
-
|
995
|
-
|
996
|
-
|
997
|
-
|
998
|
-
|
999
|
-
|
1000
|
-
|
1001
|
-
|
1002
|
-
|
1003
|
-
|
1004
|
-
|
1005
|
-
|
1006
|
-
|
1007
|
-
|
1008
|
-
|
1009
|
-
|
1010
|
-
|
1011
|
-
|
1012
|
-
|
1366
|
+
error('encountered multiple "Returns" parameters or tags for "%s".' %
|
1367
|
+
(comment_block.name, ),
|
1368
|
+
position)
|
1369
|
+
|
1370
|
+
tag = GtkDocTag(TAG_RETURNS, position)
|
1371
|
+
|
1372
|
+
if param_fields:
|
1373
|
+
result = self._parse_fields(position,
|
1374
|
+
column_offset + param_fields_start,
|
1375
|
+
original_line,
|
1376
|
+
param_fields)
|
1377
|
+
if result.success:
|
1378
|
+
tag.annotations = result.annotations
|
1379
|
+
tag.description = result.description
|
1380
|
+
comment_block.tags[TAG_RETURNS] = tag
|
1381
|
+
current_part = tag
|
1382
|
+
continue
|
1383
|
+
elif (param_name == 'Varargs'
|
1384
|
+
or (param_name.endswith('...') and param_name != '...')):
|
1385
|
+
# Deprecated @Varargs notation or named __VA_ARGS__ instead of @...
|
1386
|
+
warn('"@%s" parameter is deprecated, please use "@..." instead:\n%s\n%s' %
|
1387
|
+
(param_name, original_line, marker),
|
1388
|
+
position)
|
1389
|
+
param_name = '...'
|
1390
|
+
|
1391
|
+
if param_name in comment_block.params.keys():
|
1392
|
+
error('multiple "@%s" parameters for identifier "%s":\n%s\n%s' %
|
1393
|
+
(param_name, comment_block.name, original_line, marker),
|
1394
|
+
position)
|
1395
|
+
|
1396
|
+
parameter = GtkDocParameter(param_name, position)
|
1397
|
+
|
1398
|
+
if param_fields:
|
1399
|
+
result = self._parse_fields(position,
|
1400
|
+
column_offset + param_fields_start,
|
1401
|
+
original_line,
|
1402
|
+
param_fields)
|
1403
|
+
if result.success:
|
1404
|
+
parameter.annotations = result.annotations
|
1405
|
+
parameter.description = result.description
|
1406
|
+
|
1407
|
+
comment_block.params[param_name] = parameter
|
1408
|
+
current_part = parameter
|
1013
1409
|
continue
|
1014
1410
|
|
1015
1411
|
####################################################################
|
1016
1412
|
# Check for comment block description.
|
1017
1413
|
#
|
1018
|
-
# When we are parsing
|
1019
|
-
#
|
1020
|
-
#
|
1414
|
+
# When we are parsing parameter parts or the identifier part (when
|
1415
|
+
# there are no parameters) and encounter an empty line, we must be
|
1416
|
+
# parsing the comment block description.
|
1417
|
+
#
|
1418
|
+
# Note: it is unclear why GTK-Doc does not allow paragraph breaks
|
1419
|
+
# at this location as those might be handy describing
|
1420
|
+
# parameters from time to time...
|
1021
1421
|
####################################################################
|
1022
1422
|
if (EMPTY_LINE_RE.match(line) and in_part in [PART_IDENTIFIER, PART_PARAMETERS]):
|
1023
1423
|
in_part = PART_DESCRIPTION
|
@@ -1029,102 +1429,179 @@ class AnnotationParser(object):
|
|
1029
1429
|
####################################################################
|
1030
1430
|
result = TAG_RE.match(line)
|
1031
1431
|
if result and line_indent <= part_indent:
|
1432
|
+
part_indent = line_indent
|
1032
1433
|
tag_name = result.group('tag_name')
|
1033
|
-
|
1034
|
-
|
1035
|
-
|
1434
|
+
tag_name_lower = tag_name.lower()
|
1435
|
+
tag_fields = result.group('fields')
|
1436
|
+
tag_fields_start = result.start('fields')
|
1036
1437
|
marker = ' ' * (result.start('tag_name') + column_offset) + '^'
|
1037
1438
|
|
1038
|
-
|
1039
|
-
|
1040
|
-
|
1041
|
-
|
1042
|
-
|
1439
|
+
if tag_name_lower in DEPRECATED_GI_ANN_TAGS:
|
1440
|
+
# Deprecated GObject-Introspection specific tags.
|
1441
|
+
# Emit a warning and transform these into annotations on the identifier
|
1442
|
+
# instead, as agreed upon in http://bugzilla.gnome.org/show_bug.cgi?id=676133
|
1443
|
+
warn('GObject-Introspection specific GTK-Doc tag "%s" '
|
1444
|
+
'has been deprecated, please use annotations on the identifier '
|
1445
|
+
'instead:\n%s\n%s' % (tag_name, original_line, marker),
|
1446
|
+
position)
|
1447
|
+
|
1448
|
+
# Translate deprecated tag name into corresponding annotation name
|
1449
|
+
ann_name = tag_name_lower.replace(' ', '-')
|
1450
|
+
|
1451
|
+
if tag_name_lower == TAG_ATTRIBUTES:
|
1452
|
+
transformed = ''
|
1453
|
+
result = self._parse_fields(position,
|
1454
|
+
result.start('tag_name') + column_offset,
|
1455
|
+
line,
|
1456
|
+
tag_fields.strip(),
|
1457
|
+
False,
|
1458
|
+
False)
|
1459
|
+
|
1460
|
+
if result.success:
|
1461
|
+
for annotation in result.annotations:
|
1462
|
+
ann_options = self._parse_annotation_options_list(position, marker,
|
1463
|
+
line, annotation)
|
1464
|
+
n_options = len(ann_options)
|
1465
|
+
if n_options == 1:
|
1466
|
+
transformed = '%s %s' % (transformed, ann_options[0], )
|
1467
|
+
elif n_options == 2:
|
1468
|
+
transformed = '%s %s=%s' % (transformed, ann_options[0],
|
1469
|
+
ann_options[1])
|
1470
|
+
else:
|
1471
|
+
# Malformed Attributes: tag
|
1472
|
+
error('malformed "Attributes:" tag will be ignored:\n%s\n%s' %
|
1473
|
+
(original_line, marker),
|
1474
|
+
position)
|
1475
|
+
transformed = None
|
1476
|
+
|
1477
|
+
if transformed:
|
1478
|
+
transformed = '%s %s' % (ann_name, transformed.strip())
|
1479
|
+
ann_name, docannotation = self._parse_annotation(
|
1480
|
+
position,
|
1481
|
+
column_offset + tag_fields_start,
|
1482
|
+
original_line,
|
1483
|
+
transformed)
|
1484
|
+
stored_annotation = comment_block.annotations.get('attributes')
|
1485
|
+
if stored_annotation:
|
1486
|
+
error('Duplicate "Attributes:" annotation will '
|
1487
|
+
'be ignored:\n%s\n%s' % (original_line, marker),
|
1488
|
+
position)
|
1489
|
+
else:
|
1490
|
+
comment_block.annotations[ann_name] = docannotation
|
1491
|
+
else:
|
1492
|
+
ann_name, options = self._parse_annotation(position,
|
1493
|
+
column_offset + tag_fields_start,
|
1494
|
+
line,
|
1495
|
+
'%s %s' % (ann_name, tag_fields))
|
1496
|
+
comment_block.annotations[ann_name] = options
|
1497
|
+
|
1498
|
+
continue
|
1499
|
+
elif tag_name_lower == TAG_DESCRIPTION:
|
1500
|
+
# Deprecated GTK-Doc Description: tag
|
1501
|
+
warn('GTK-Doc tag "Description:" has been deprecated:\n%s\n%s' %
|
1502
|
+
(original_line, marker),
|
1503
|
+
position)
|
1043
1504
|
|
1044
1505
|
in_part = PART_DESCRIPTION
|
1045
|
-
part_indent = line_indent
|
1046
1506
|
|
1047
|
-
if
|
1048
|
-
comment_block.
|
1507
|
+
if comment_block.description is None:
|
1508
|
+
comment_block.description = tag_fields
|
1049
1509
|
else:
|
1050
|
-
comment_block.
|
1510
|
+
comment_block.description += '\n%s' % (tag_fields, )
|
1051
1511
|
continue
|
1052
1512
|
|
1053
1513
|
# Now that the deprecated stuff is out of the way, continue parsing real tags
|
1054
|
-
if in_part == PART_DESCRIPTION
|
1514
|
+
if (in_part == PART_DESCRIPTION
|
1515
|
+
or (in_part == PART_PARAMETERS and not comment_block.description)
|
1516
|
+
or (in_part == PART_IDENTIFIER and not comment_block.params and not
|
1517
|
+
comment_block.description)):
|
1055
1518
|
in_part = PART_TAGS
|
1056
1519
|
|
1057
|
-
part_indent = line_indent
|
1058
|
-
|
1059
1520
|
if in_part != PART_TAGS:
|
1060
|
-
|
1061
|
-
|
1062
|
-
|
1063
|
-
|
1064
|
-
position)
|
1521
|
+
in_part = PART_TAGS
|
1522
|
+
warn('"%s:" tag unexpected at this location:\n%s\n%s' %
|
1523
|
+
(tag_name, original_line, marker),
|
1524
|
+
position)
|
1065
1525
|
|
1066
|
-
if
|
1526
|
+
if tag_name_lower in [TAG_RETURN, TAG_RETURNS,
|
1527
|
+
TAG_RETURN_VALUE, TAG_RETURNS_VALUE]:
|
1067
1528
|
if not returns_seen:
|
1068
1529
|
returns_seen = True
|
1069
1530
|
else:
|
1070
|
-
|
1071
|
-
|
1072
|
-
|
1073
|
-
|
1074
|
-
tag =
|
1075
|
-
|
1076
|
-
|
1077
|
-
|
1078
|
-
|
1531
|
+
error('encountered multiple return value parameters or tags for "%s".' %
|
1532
|
+
(comment_block.name, ),
|
1533
|
+
position)
|
1534
|
+
|
1535
|
+
tag = GtkDocTag(TAG_RETURNS, position)
|
1536
|
+
|
1537
|
+
if tag_fields:
|
1538
|
+
result = self._parse_fields(position,
|
1539
|
+
column_offset + tag_fields_start,
|
1540
|
+
original_line,
|
1541
|
+
tag_fields)
|
1542
|
+
if result.success:
|
1543
|
+
tag.annotations = result.annotations
|
1544
|
+
tag.description = result.description
|
1545
|
+
|
1079
1546
|
comment_block.tags[TAG_RETURNS] = tag
|
1080
|
-
|
1547
|
+
current_part = tag
|
1081
1548
|
continue
|
1082
1549
|
else:
|
1083
|
-
if
|
1084
|
-
|
1085
|
-
|
1086
|
-
|
1087
|
-
|
1088
|
-
|
1089
|
-
|
1090
|
-
|
1091
|
-
|
1092
|
-
|
1093
|
-
|
1094
|
-
|
1095
|
-
|
1096
|
-
|
1097
|
-
|
1098
|
-
|
1099
|
-
|
1100
|
-
|
1101
|
-
|
1550
|
+
if tag_name_lower in comment_block.tags.keys():
|
1551
|
+
error('multiple "%s:" tags for identifier "%s":\n%s\n%s' %
|
1552
|
+
(tag_name, comment_block.name, original_line, marker),
|
1553
|
+
position)
|
1554
|
+
|
1555
|
+
tag = GtkDocTag(tag_name_lower, position)
|
1556
|
+
|
1557
|
+
if tag_fields:
|
1558
|
+
result = self._parse_fields(position,
|
1559
|
+
column_offset + tag_fields_start,
|
1560
|
+
original_line,
|
1561
|
+
tag_fields)
|
1562
|
+
if result.success:
|
1563
|
+
if result.annotations:
|
1564
|
+
error('annotations not supported for tag "%s:".' % (tag_name, ),
|
1565
|
+
position)
|
1566
|
+
|
1567
|
+
if tag_name_lower in [TAG_DEPRECATED, TAG_SINCE]:
|
1568
|
+
result = TAG_VALUE_VERSION_RE.match(result.description)
|
1569
|
+
tag.value = result.group('value')
|
1570
|
+
tag.description = result.group('description')
|
1571
|
+
elif tag_name_lower == TAG_STABILITY:
|
1572
|
+
result = TAG_VALUE_STABILITY_RE.match(result.description)
|
1573
|
+
tag.value = result.group('value').capitalize()
|
1574
|
+
tag.description = result.group('description')
|
1575
|
+
|
1576
|
+
comment_block.tags[tag_name_lower] = tag
|
1577
|
+
current_part = tag
|
1102
1578
|
continue
|
1103
1579
|
|
1104
1580
|
####################################################################
|
1105
1581
|
# If we get here, we must be in the middle of a multiline
|
1106
1582
|
# comment block, parameter or tag description.
|
1107
1583
|
####################################################################
|
1584
|
+
if EMPTY_LINE_RE.match(line) is None:
|
1585
|
+
line = line.rstrip()
|
1586
|
+
|
1108
1587
|
if in_part in [PART_IDENTIFIER, PART_DESCRIPTION]:
|
1109
|
-
if not comment_block.
|
1110
|
-
|
1588
|
+
if not comment_block.description:
|
1589
|
+
if in_part == PART_IDENTIFIER:
|
1590
|
+
self._validate_multiline_annotation_continuation(line, original_line,
|
1591
|
+
column_offset, position)
|
1592
|
+
if comment_block.description is None:
|
1593
|
+
comment_block.description = line
|
1111
1594
|
else:
|
1112
|
-
comment_block.
|
1113
|
-
continue
|
1114
|
-
elif in_part == PART_PARAMETERS:
|
1115
|
-
self._validate_multiline_annotation_continuation(line, original_line,
|
1116
|
-
column_offset, position)
|
1117
|
-
# Append to parameter description.
|
1118
|
-
current_param.comment += ' ' + line.strip()
|
1595
|
+
comment_block.description += '\n' + line
|
1119
1596
|
continue
|
1120
|
-
elif in_part
|
1121
|
-
|
1122
|
-
|
1123
|
-
|
1124
|
-
if
|
1125
|
-
|
1597
|
+
elif in_part in [PART_PARAMETERS, PART_TAGS]:
|
1598
|
+
if not current_part.description:
|
1599
|
+
self._validate_multiline_annotation_continuation(line, original_line,
|
1600
|
+
column_offset, position)
|
1601
|
+
if current_part.description is None:
|
1602
|
+
current_part.description = line
|
1126
1603
|
else:
|
1127
|
-
|
1604
|
+
current_part.description += '\n' + line
|
1128
1605
|
continue
|
1129
1606
|
|
1130
1607
|
########################################################################
|
@@ -1133,77 +1610,552 @@ class AnnotationParser(object):
|
|
1133
1610
|
if comment_block:
|
1134
1611
|
# We have picked up a couple of \n characters that where not
|
1135
1612
|
# intended. Strip those.
|
1136
|
-
if comment_block.
|
1137
|
-
comment_block.
|
1613
|
+
if comment_block.description:
|
1614
|
+
comment_block.description = comment_block.description.strip()
|
1138
1615
|
|
1139
1616
|
for tag in comment_block.tags.values():
|
1140
|
-
self.
|
1617
|
+
self._clean_description_field(tag)
|
1141
1618
|
|
1142
1619
|
for param in comment_block.params.values():
|
1143
|
-
self.
|
1620
|
+
self._clean_description_field(param)
|
1144
1621
|
|
1145
|
-
|
1622
|
+
comment_block.indentation = block_indent
|
1146
1623
|
comment_block.validate()
|
1147
1624
|
return comment_block
|
1148
1625
|
else:
|
1149
1626
|
return None
|
1150
1627
|
|
1151
|
-
def
|
1152
|
-
|
1153
|
-
|
1154
|
-
else:
|
1155
|
-
part.comment = None
|
1628
|
+
def _clean_description_field(self, part):
|
1629
|
+
'''
|
1630
|
+
Remove extraneous leading and trailing whitespace from description fields.
|
1156
1631
|
|
1157
|
-
|
1158
|
-
|
1159
|
-
|
1160
|
-
|
1632
|
+
:param part: a GTK-Doc comment block part having a description field
|
1633
|
+
'''
|
1634
|
+
|
1635
|
+
if part.description:
|
1636
|
+
if part.description.strip() == '':
|
1637
|
+
part.description = None
|
1638
|
+
else:
|
1639
|
+
if EMPTY_LINE_RE.match(part.description.split('\n', 1)[0]):
|
1640
|
+
part.description = part.description.rstrip()
|
1641
|
+
else:
|
1642
|
+
part.description = part.description.strip()
|
1161
1643
|
|
1162
1644
|
def _validate_multiline_annotation_continuation(self, line, original_line,
|
1163
1645
|
column_offset, position):
|
1164
1646
|
'''
|
1165
|
-
Validate
|
1166
|
-
|
1647
|
+
Validate annotatable parts' source text ensuring annotations don't span multiple lines.
|
1648
|
+
For example, the following comment block would result in a warning being emitted for
|
1649
|
+
the forth line::
|
1650
|
+
|
1651
|
+
/**
|
1652
|
+
* shiny_function:
|
1653
|
+
* @array_: (out caller-allocates) (array)
|
1654
|
+
* (element-type utf8) (transfer full): A beautiful array
|
1655
|
+
*/
|
1656
|
+
|
1657
|
+
:param line: line to validate, stripped from ("``*/``") at start of the line.
|
1658
|
+
:param original_line: original line (including ("``*/``")) being validated
|
1659
|
+
:param column_offset: number of characters stripped from `line` when ("``*/``")
|
1660
|
+
was removed
|
1661
|
+
:param position: :class:`giscanner.message.Position` of `line` in the source file
|
1662
|
+
'''
|
1663
|
+
|
1664
|
+
result = self._parse_annotations(position, column_offset, original_line, line)
|
1665
|
+
|
1666
|
+
if result.success and result.annotations:
|
1667
|
+
marker = ' ' * (result.start_pos + column_offset) + '^'
|
1668
|
+
error('ignoring invalid multiline annotation continuation:\n%s\n%s' %
|
1669
|
+
(original_line, marker),
|
1670
|
+
position)
|
1167
1671
|
|
1168
|
-
|
1169
|
-
|
1170
|
-
|
1171
|
-
|
1672
|
+
def _parse_annotation_options_list(self, position, column, line, options):
|
1673
|
+
'''
|
1674
|
+
Parse annotation options into a list. For example::
|
1675
|
+
|
1676
|
+
┌──────────────────────────────────────────────────────────────┐
|
1677
|
+
│ 'option1 option2 option3' │ ─▷ source
|
1678
|
+
├──────────────────────────────────────────────────────────────┤
|
1679
|
+
│ ['option1', 'option2', 'option3'] │ ◁─ parsed options
|
1680
|
+
└──────────────────────────────────────────────────────────────┘
|
1681
|
+
|
1682
|
+
:param position: :class:`giscanner.message.Position` of `line` in the source file
|
1683
|
+
:param column: start column of the `options` in the source file
|
1684
|
+
:param line: complete source line
|
1685
|
+
:param options: annotation options to parse
|
1686
|
+
:returns: a list of annotation options
|
1172
1687
|
'''
|
1173
1688
|
|
1174
|
-
|
1175
|
-
if result:
|
1176
|
-
column = result.start('annotations') + column_offset
|
1177
|
-
marker = ' ' * column + '^'
|
1178
|
-
message.warn('ignoring invalid multiline annotation continuation:\n'
|
1179
|
-
'%s\n%s' % (original_line, marker),
|
1180
|
-
position)
|
1689
|
+
parsed = []
|
1181
1690
|
|
1182
|
-
|
1183
|
-
|
1184
|
-
|
1185
|
-
|
1186
|
-
|
1187
|
-
|
1188
|
-
|
1189
|
-
|
1190
|
-
|
1191
|
-
|
1192
|
-
|
1193
|
-
|
1194
|
-
|
1195
|
-
|
1196
|
-
|
1197
|
-
|
1198
|
-
|
1199
|
-
|
1200
|
-
|
1201
|
-
|
1691
|
+
if options:
|
1692
|
+
result = options.find('=')
|
1693
|
+
if result >= 0:
|
1694
|
+
marker = ' ' * (column + result) + '^'
|
1695
|
+
warn('invalid annotation options: expected a "list" but '
|
1696
|
+
'received "key=value pairs":\n%s\n%s' % (line, marker),
|
1697
|
+
position)
|
1698
|
+
parsed = self._parse_annotation_options_unknown(position, column, line, options)
|
1699
|
+
else:
|
1700
|
+
parsed = options.split(' ')
|
1701
|
+
|
1702
|
+
return parsed
|
1703
|
+
|
1704
|
+
def _parse_annotation_options_dict(self, position, column, line, options):
|
1705
|
+
'''
|
1706
|
+
Parse annotation options into a dict. For example::
|
1707
|
+
|
1708
|
+
┌──────────────────────────────────────────────────────────────┐
|
1709
|
+
│ 'option1=value1 option2 option3=value2' │ ─▷ source
|
1710
|
+
├──────────────────────────────────────────────────────────────┤
|
1711
|
+
│ {'option1': 'value1', 'option2': None, 'option3': 'value2'} │ ◁─ parsed options
|
1712
|
+
└──────────────────────────────────────────────────────────────┘
|
1713
|
+
|
1714
|
+
:param position: :class:`giscanner.message.Position` of `line` in the source file
|
1715
|
+
:param column: start column of the `options` in the source file
|
1716
|
+
:param line: complete source line
|
1717
|
+
:param options: annotation options to parse
|
1718
|
+
:returns: an ordered dictionary of annotation options
|
1719
|
+
'''
|
1720
|
+
|
1721
|
+
parsed = OrderedDict()
|
1722
|
+
|
1723
|
+
if options:
|
1724
|
+
for p in options.split(' '):
|
1725
|
+
parts = p.split('=', 1)
|
1726
|
+
key = parts[0]
|
1727
|
+
value = parts[1] if len(parts) == 2 else None
|
1728
|
+
parsed[key] = value
|
1729
|
+
|
1730
|
+
return parsed
|
1731
|
+
|
1732
|
+
def _parse_annotation_options_unknown(self, position, column, line, options):
|
1733
|
+
'''
|
1734
|
+
Parse annotation options into a list holding a single item. This is used when the
|
1735
|
+
annotation options to parse in not known to be a list nor dict. For example::
|
1736
|
+
|
1737
|
+
┌──────────────────────────────────────────────────────────────┐
|
1738
|
+
│ ' option1 option2 option3=value1 ' │ ─▷ source
|
1739
|
+
├──────────────────────────────────────────────────────────────┤
|
1740
|
+
│ ['option1 option2 option3=value1'] │ ◁─ parsed options
|
1741
|
+
└──────────────────────────────────────────────────────────────┘
|
1742
|
+
|
1743
|
+
:param position: :class:`giscanner.message.Position` of `line` in the source file
|
1744
|
+
:param column: start column of the `options` in the source file
|
1745
|
+
:param line: complete source line
|
1746
|
+
:param options: annotation options to parse
|
1747
|
+
:returns: a list of annotation options
|
1748
|
+
'''
|
1749
|
+
|
1750
|
+
if options:
|
1751
|
+
return [options.strip()]
|
1752
|
+
|
1753
|
+
def _parse_annotation(self, position, column, line, annotation):
|
1754
|
+
'''
|
1755
|
+
Parse an annotation into the annotation name and a list or dict (depending on the
|
1756
|
+
name of the annotation) holding the options. For example::
|
1757
|
+
|
1758
|
+
┌──────────────────────────────────────────────────────────────┐
|
1759
|
+
│ 'name opt1=value1 opt2=value2 opt3' │ ─▷ source
|
1760
|
+
├──────────────────────────────────────────────────────────────┤
|
1761
|
+
│ 'name', {'opt1': 'value1', 'opt2':'value2', 'opt3':None} │ ◁─ parsed annotation
|
1762
|
+
└──────────────────────────────────────────────────────────────┘
|
1763
|
+
|
1764
|
+
┌──────────────────────────────────────────────────────────────┐
|
1765
|
+
│ 'name opt1 opt2' │ ─▷ source
|
1766
|
+
├──────────────────────────────────────────────────────────────┤
|
1767
|
+
│ 'name', ['opt1', 'opt2'] │ ◁─ parsed annotation
|
1768
|
+
└──────────────────────────────────────────────────────────────┘
|
1769
|
+
|
1770
|
+
┌──────────────────────────────────────────────────────────────┐
|
1771
|
+
│ 'unkownname unknown list of options' │ ─▷ source
|
1772
|
+
├──────────────────────────────────────────────────────────────┤
|
1773
|
+
│ 'unkownname', ['unknown list of options'] │ ◁─ parsed annotation
|
1774
|
+
└──────────────────────────────────────────────────────────────┘
|
1775
|
+
|
1776
|
+
:param position: :class:`giscanner.message.Position` of `line` in the source file
|
1777
|
+
:param column: start column of the `annotation` in the source file
|
1778
|
+
:param line: complete source line
|
1779
|
+
:param annotation: annotation to parse
|
1780
|
+
:returns: a tuple containing the annotation name and options
|
1781
|
+
'''
|
1782
|
+
|
1783
|
+
# Transform deprecated type syntax "tokens"
|
1784
|
+
annotation = annotation.replace('<', ANN_LPAR).replace('>', ANN_RPAR)
|
1785
|
+
|
1786
|
+
parts = annotation.split(' ', 1)
|
1787
|
+
ann_name = parts[0].lower()
|
1788
|
+
ann_options = parts[1] if len(parts) == 2 else None
|
1789
|
+
|
1790
|
+
if ann_name == ANN_INOUT_ALT:
|
1791
|
+
marker = ' ' * (column) + '^'
|
1792
|
+
warn('"%s" annotation has been deprecated, please use "%s" instead:\n%s\n%s' %
|
1793
|
+
(ANN_INOUT_ALT, ANN_INOUT, line, marker),
|
1794
|
+
position)
|
1795
|
+
|
1796
|
+
ann_name = ANN_INOUT
|
1797
|
+
elif ann_name == ANN_ATTRIBUTE:
|
1798
|
+
marker = ' ' * (column) + '^'
|
1799
|
+
warn('"%s" annotation has been deprecated, please use "%s" instead:\n%s\n%s' %
|
1800
|
+
(ANN_ATTRIBUTE, ANN_ATTRIBUTES, line, marker),
|
1801
|
+
position)
|
1802
|
+
|
1803
|
+
ann_name = ANN_ATTRIBUTES
|
1804
|
+
ann_options = self._parse_annotation_options_list(position, column, line, ann_options)
|
1805
|
+
n_options = len(ann_options)
|
1806
|
+
if n_options == 1:
|
1807
|
+
ann_options = ann_options[0]
|
1808
|
+
elif n_options == 2:
|
1809
|
+
ann_options = '%s=%s' % (ann_options[0], ann_options[1])
|
1810
|
+
else:
|
1811
|
+
marker = ' ' * (column) + '^'
|
1812
|
+
error('malformed "(attribute)" annotation will be ignored:\n%s\n%s' %
|
1813
|
+
(line, marker),
|
1814
|
+
position)
|
1815
|
+
return None, None
|
1816
|
+
|
1817
|
+
column += len(ann_name) + 2
|
1818
|
+
|
1819
|
+
if ann_name in LIST_ANNOTATIONS:
|
1820
|
+
ann_options = self._parse_annotation_options_list(position, column, line, ann_options)
|
1821
|
+
elif ann_name in DICT_ANNOTATIONS:
|
1822
|
+
ann_options = self._parse_annotation_options_dict(position, column, line, ann_options)
|
1823
|
+
else:
|
1824
|
+
ann_options = self._parse_annotation_options_unknown(position, column, line,
|
1825
|
+
ann_options)
|
1826
|
+
|
1827
|
+
return ann_name, ann_options
|
1828
|
+
|
1829
|
+
def _parse_annotations(self, position, column, line, fields, parse_options=True):
|
1830
|
+
'''
|
1831
|
+
Parse annotations into a :class:`GtkDocAnnotations` object.
|
1832
|
+
|
1833
|
+
:param position: :class:`giscanner.message.Position` of `line` in the source file
|
1834
|
+
:param column: start column of the `annotations` in the source file
|
1835
|
+
:param line: complete source line
|
1836
|
+
:param fields: string containing the fields to parse
|
1837
|
+
:param parse_options: whether options will be parsed into a :class:`GtkDocAnnotations`
|
1838
|
+
object or into a :class:`list`
|
1839
|
+
:returns: if `parse_options` evaluates to True a :class:`GtkDocAnnotations` object,
|
1840
|
+
a :class:`list` otherwise. If `line` does not contain any annotations,
|
1841
|
+
:const:`None`
|
1842
|
+
'''
|
1843
|
+
|
1844
|
+
if parse_options:
|
1845
|
+
parsed_annotations = GtkDocAnnotations(position)
|
1846
|
+
else:
|
1847
|
+
parsed_annotations = []
|
1848
|
+
|
1849
|
+
i = 0
|
1850
|
+
parens_level = 0
|
1851
|
+
prev_char = ''
|
1852
|
+
char_buffer = []
|
1853
|
+
start_pos = 0
|
1854
|
+
end_pos = 0
|
1855
|
+
|
1856
|
+
for i, cur_char in enumerate(fields):
|
1857
|
+
cur_char_is_space = cur_char.isspace()
|
1858
|
+
|
1859
|
+
if cur_char == ANN_LPAR:
|
1860
|
+
parens_level += 1
|
1861
|
+
|
1862
|
+
if parens_level == 1:
|
1863
|
+
start_pos = i
|
1864
|
+
|
1865
|
+
if prev_char == ANN_LPAR:
|
1866
|
+
marker = ' ' * (column + i) + '^'
|
1867
|
+
error('unexpected parentheses, annotations will be ignored:\n%s\n%s' %
|
1868
|
+
(line, marker),
|
1869
|
+
position)
|
1870
|
+
return _ParseAnnotationsResult(False, None, None, None)
|
1871
|
+
elif parens_level > 1:
|
1872
|
+
char_buffer.append(cur_char)
|
1873
|
+
elif cur_char == ANN_RPAR:
|
1874
|
+
parens_level -= 1
|
1875
|
+
|
1876
|
+
if prev_char == ANN_LPAR:
|
1877
|
+
marker = ' ' * (column + i) + '^'
|
1878
|
+
error('unexpected parentheses, annotations will be ignored:\n%s\n%s' %
|
1879
|
+
(line, marker),
|
1880
|
+
position)
|
1881
|
+
return _ParseAnnotationsResult(False, None, None, None)
|
1882
|
+
elif parens_level < 0:
|
1883
|
+
marker = ' ' * (column + i) + '^'
|
1884
|
+
error('unbalanced parentheses, annotations will be ignored:\n%s\n%s' %
|
1885
|
+
(line, marker),
|
1886
|
+
position)
|
1887
|
+
return _ParseAnnotationsResult(False, None, None, None)
|
1888
|
+
elif parens_level == 0:
|
1889
|
+
end_pos = i + 1
|
1890
|
+
|
1891
|
+
if parse_options is True:
|
1892
|
+
name, options = self._parse_annotation(position,
|
1893
|
+
column + start_pos,
|
1894
|
+
line,
|
1895
|
+
''.join(char_buffer).strip())
|
1896
|
+
if name is not None:
|
1897
|
+
if name in parsed_annotations:
|
1898
|
+
marker = ' ' * (column + i) + '^'
|
1899
|
+
error('multiple "%s" annotations:\n%s\n%s' %
|
1900
|
+
(name, line, marker), position)
|
1901
|
+
parsed_annotations[name] = options
|
1902
|
+
else:
|
1903
|
+
parsed_annotations.append(''.join(char_buffer).strip())
|
1904
|
+
|
1905
|
+
char_buffer = []
|
1906
|
+
else:
|
1907
|
+
char_buffer.append(cur_char)
|
1908
|
+
elif cur_char_is_space:
|
1909
|
+
if parens_level > 0:
|
1910
|
+
char_buffer.append(cur_char)
|
1911
|
+
else:
|
1912
|
+
if parens_level == 0:
|
1913
|
+
break
|
1202
1914
|
else:
|
1203
|
-
|
1204
|
-
|
1205
|
-
|
1206
|
-
|
1207
|
-
|
1915
|
+
char_buffer.append(cur_char)
|
1916
|
+
|
1917
|
+
prev_char = cur_char
|
1918
|
+
|
1919
|
+
if parens_level > 0:
|
1920
|
+
marker = ' ' * (column + i) + '^'
|
1921
|
+
error('unbalanced parentheses, annotations will be ignored:\n%s\n%s' %
|
1922
|
+
(line, marker),
|
1923
|
+
position)
|
1924
|
+
return _ParseAnnotationsResult(False, None, None, None)
|
1925
|
+
else:
|
1926
|
+
return _ParseAnnotationsResult(True, parsed_annotations, start_pos, end_pos)
|
1927
|
+
|
1928
|
+
def _parse_fields(self, position, column, line, fields, parse_options=True,
|
1929
|
+
validate_description_field=True):
|
1930
|
+
'''
|
1931
|
+
Parse annotations out of field data. For example::
|
1932
|
+
|
1933
|
+
┌──────────────────────────────────────────────────────────────┐
|
1934
|
+
│ '(skip): description of some parameter │ ─▷ source
|
1935
|
+
├──────────────────────────────────────────────────────────────┤
|
1936
|
+
│ ({'skip': []}, 'description of some parameter') │ ◁─ annotations and
|
1937
|
+
└──────────────────────────────────────────────────────────────┘ remaining fields
|
1938
|
+
|
1939
|
+
:param position: :class:`giscanner.message.Position` of `line` in the source file
|
1940
|
+
:param column: start column of `fields` in the source file
|
1941
|
+
:param line: complete source line
|
1942
|
+
:param fields: string containing the fields to parse
|
1943
|
+
:param parse_options: whether options will be parsed into a :class:`GtkDocAnnotations`
|
1944
|
+
object or into a :class:`list`
|
1945
|
+
:param validate_description_field: :const:`True` to validate the description field
|
1946
|
+
:returns: if `parse_options` evaluates to True a :class:`GtkDocAnnotations` object,
|
1947
|
+
a :class:`list` otherwise. If `line` does not contain any annotations,
|
1948
|
+
:const:`None` and a string holding the remaining fields
|
1949
|
+
'''
|
1950
|
+
description_field = ''
|
1951
|
+
result = self._parse_annotations(position, column, line, fields, parse_options)
|
1952
|
+
if result.success:
|
1953
|
+
description_field = fields[result.end_pos:].strip()
|
1954
|
+
|
1955
|
+
if description_field and validate_description_field:
|
1956
|
+
if description_field.startswith(':'):
|
1957
|
+
description_field = description_field[1:]
|
1958
|
+
else:
|
1959
|
+
if result.end_pos > 0:
|
1960
|
+
marker_position = column + result.end_pos
|
1961
|
+
marker = ' ' * marker_position + '^'
|
1962
|
+
warn('missing ":" at column %s:\n%s\n%s' %
|
1963
|
+
(marker_position + 1, line, marker),
|
1964
|
+
position)
|
1965
|
+
|
1966
|
+
return _ParseFieldsResult(result.success, result.annotations, description_field)
|
1967
|
+
|
1968
|
+
|
1969
|
+
class GtkDocCommentBlockWriter(object):
|
1970
|
+
'''
|
1971
|
+
Serialized :class:`GtkDocCommentBlock` objects into GTK-Doc comment blocks.
|
1972
|
+
'''
|
1973
|
+
|
1974
|
+
def __init__(self, indent=True):
|
1975
|
+
#: :const:`True` if the original indentation preceding the "``*``" needs to be retained,
|
1976
|
+
#: :const:`False` otherwise. Default value is :const:`True`.
|
1977
|
+
self.indent = indent
|
1978
|
+
|
1979
|
+
def _serialize_annotations(self, annotations):
|
1980
|
+
'''
|
1981
|
+
Serialize an annotation field. For example::
|
1982
|
+
|
1983
|
+
┌──────────────────────────────────────────────────────────────┐
|
1984
|
+
│ {'name': {'opt1': 'value1', 'opt2':'value2', 'opt3':None} │ ◁─ GtkDocAnnotations
|
1985
|
+
├──────────────────────────────────────────────────────────────┤
|
1986
|
+
│ '(name opt1=value1 opt2=value2 opt3)' │ ─▷ serialized
|
1987
|
+
└──────────────────────────────────────────────────────────────┘
|
1988
|
+
|
1989
|
+
┌──────────────────────────────────────────────────────────────┐
|
1990
|
+
│ {'name': ['opt1', 'opt2']} │ ◁─ GtkDocAnnotations
|
1991
|
+
├──────────────────────────────────────────────────────────────┤
|
1992
|
+
│ '(name opt1 opt2)' │ ─▷ serialized
|
1993
|
+
└──────────────────────────────────────────────────────────────┘
|
1994
|
+
|
1995
|
+
┌──────────────────────────────────────────────────────────────┐
|
1996
|
+
│ {'unkownname': ['unknown list of options']} │ ◁─ GtkDocAnnotations
|
1997
|
+
├──────────────────────────────────────────────────────────────┤
|
1998
|
+
│ '(unkownname unknown list of options)' │ ─▷ serialized
|
1999
|
+
└──────────────────────────────────────────────────────────────┘
|
2000
|
+
|
2001
|
+
:param annotations: :class:`GtkDocAnnotations` to be serialized
|
2002
|
+
:returns: a string
|
2003
|
+
'''
|
2004
|
+
|
2005
|
+
serialized = []
|
2006
|
+
|
2007
|
+
for ann_name, options in annotations.items():
|
2008
|
+
if options:
|
2009
|
+
if isinstance(options, list):
|
2010
|
+
serialize_options = ' '.join(options)
|
2011
|
+
else:
|
2012
|
+
serialize_options = ''
|
2013
|
+
|
2014
|
+
for key, value in options.items():
|
2015
|
+
if value:
|
2016
|
+
serialize_options += '%s=%s ' % (key, value)
|
2017
|
+
else:
|
2018
|
+
serialize_options += '%s ' % (key, )
|
2019
|
+
|
2020
|
+
serialize_options = serialize_options.strip()
|
2021
|
+
|
2022
|
+
serialized.append('(%s %s)' % (ann_name, serialize_options))
|
2023
|
+
else:
|
2024
|
+
serialized.append('(%s)' % (ann_name, ))
|
2025
|
+
|
2026
|
+
return ' '.join(serialized)
|
2027
|
+
|
2028
|
+
def _serialize_parameter(self, parameter):
|
2029
|
+
'''
|
2030
|
+
Serialize a parameter.
|
2031
|
+
|
2032
|
+
:param parameter: :class:`GtkDocParameter` to be serialized
|
2033
|
+
:returns: a string
|
2034
|
+
'''
|
2035
|
+
|
2036
|
+
# parameter_name field
|
2037
|
+
serialized = '@%s' % (parameter.name, )
|
2038
|
+
|
2039
|
+
# annotations field
|
2040
|
+
if parameter.annotations:
|
2041
|
+
serialized += ': ' + self._serialize_annotations(parameter.annotations)
|
2042
|
+
|
2043
|
+
# description field
|
2044
|
+
if parameter.description:
|
2045
|
+
if parameter.description.startswith('\n'):
|
2046
|
+
serialized += ':' + parameter.description
|
2047
|
+
else:
|
2048
|
+
serialized += ': ' + parameter.description
|
2049
|
+
else:
|
2050
|
+
serialized += ':'
|
2051
|
+
|
2052
|
+
return serialized.split('\n')
|
2053
|
+
|
2054
|
+
def _serialize_tag(self, tag):
|
2055
|
+
'''
|
2056
|
+
Serialize a tag.
|
2057
|
+
|
2058
|
+
:param tag: :class:`GtkDocTag` to be serialized
|
2059
|
+
:returns: a string
|
2060
|
+
'''
|
2061
|
+
|
2062
|
+
# tag_name field
|
2063
|
+
serialized = tag.name.capitalize()
|
2064
|
+
|
2065
|
+
# annotations field
|
2066
|
+
if tag.annotations:
|
2067
|
+
serialized += ': ' + self._serialize_annotations(tag.annotations)
|
2068
|
+
|
2069
|
+
# value field
|
2070
|
+
if tag.value:
|
2071
|
+
serialized += ': ' + tag.value
|
2072
|
+
|
2073
|
+
# description field
|
2074
|
+
if tag.description:
|
2075
|
+
if tag.description.startswith('\n'):
|
2076
|
+
serialized += ':' + tag.description
|
2077
|
+
else:
|
2078
|
+
serialized += ': ' + tag.description
|
2079
|
+
|
2080
|
+
if not tag.value and not tag.description:
|
2081
|
+
serialized += ':'
|
2082
|
+
|
2083
|
+
return serialized.split('\n')
|
2084
|
+
|
2085
|
+
def write(self, block):
|
2086
|
+
'''
|
2087
|
+
Serialize a :class:`GtkDocCommentBlock` object.
|
2088
|
+
|
2089
|
+
:param block: :class:`GtkDocCommentBlock` to be serialized
|
2090
|
+
:returns: a string
|
2091
|
+
'''
|
2092
|
+
|
2093
|
+
if block is None:
|
2094
|
+
return ''
|
2095
|
+
else:
|
2096
|
+
lines = []
|
2097
|
+
|
2098
|
+
# Identifier part
|
2099
|
+
if block.name.startswith('SECTION'):
|
2100
|
+
lines.append(block.name)
|
2101
|
+
else:
|
2102
|
+
if block.annotations:
|
2103
|
+
annotations = self._serialize_annotations(block.annotations)
|
2104
|
+
lines.append('%s: %s' % (block.name, annotations))
|
2105
|
+
else:
|
2106
|
+
# Note: this delimiter serves no purpose other than most people being used
|
2107
|
+
# to reading/writing it. It is completely legal to ommit this.
|
2108
|
+
lines.append('%s:' % (block.name, ))
|
2109
|
+
|
2110
|
+
# Parameter parts
|
2111
|
+
for param in block.params.values():
|
2112
|
+
lines.extend(self._serialize_parameter(param))
|
2113
|
+
|
2114
|
+
# Comment block description part
|
2115
|
+
if block.description:
|
2116
|
+
lines.append('')
|
2117
|
+
for l in block.description.split('\n'):
|
2118
|
+
lines.append(l)
|
2119
|
+
|
2120
|
+
# Tag parts
|
2121
|
+
if block.tags:
|
2122
|
+
# Note: this empty line servers no purpose other than most people being used
|
2123
|
+
# to reading/writing it. It is completely legal to ommit this.
|
2124
|
+
lines.append('')
|
2125
|
+
for tag in block.tags.values():
|
2126
|
+
lines.extend(self._serialize_tag(tag))
|
2127
|
+
|
2128
|
+
# Restore comment block indentation and *
|
2129
|
+
if self.indent:
|
2130
|
+
indent = Counter(block.indentation).most_common(1)[0][0] or ' '
|
2131
|
+
if indent.endswith('\t'):
|
2132
|
+
start_indent = indent
|
2133
|
+
line_indent = indent + ' '
|
2134
|
+
else:
|
2135
|
+
start_indent = indent[:-1]
|
2136
|
+
line_indent = indent
|
2137
|
+
else:
|
2138
|
+
start_indent = ''
|
2139
|
+
line_indent = ' '
|
2140
|
+
|
2141
|
+
i = 0
|
2142
|
+
while i < len(lines):
|
2143
|
+
line = lines[i]
|
2144
|
+
if line:
|
2145
|
+
lines[i] = '%s* %s\n' % (line_indent, line)
|
2146
|
+
else:
|
2147
|
+
lines[i] = '%s*\n' % (line_indent, )
|
2148
|
+
i += 1
|
2149
|
+
|
2150
|
+
# Restore comment block start and end tokens
|
2151
|
+
lines.insert(0, '%s/**\n' % (start_indent, ))
|
2152
|
+
lines.append('%s*/\n' % (line_indent, ))
|
2153
|
+
|
2154
|
+
# Restore code before and after comment block start and end tokens
|
2155
|
+
if block.code_before:
|
2156
|
+
lines.insert(0, '%s\n' % (block.code_before, ))
|
2157
|
+
|
2158
|
+
if block.code_after:
|
2159
|
+
lines.append('%s\n' % (block.code_after, ))
|
1208
2160
|
|
1209
|
-
|
2161
|
+
return ''.join(lines)
|