2020-04-07 16:58:58 +02:00
#!/usr/bin/env python3
2020-11-09 13:23:58 +09:00
# SPDX-License-Identifier: LGPL-2.1-or-later
2023-07-17 21:11:04 +02:00
# pylint: disable=superfluous-parens,consider-using-with
2020-04-07 16:58:58 +02:00
2020-08-27 19:27:18 +02:00
import argparse
2020-04-07 16:58:58 +02:00
import collections
import sys
2020-04-24 12:09:07 +02:00
import os
2020-04-07 16:58:58 +02:00
import subprocess
import io
2020-09-18 18:51:42 +02:00
try :
from lxml import etree
except ModuleNotFoundError as e :
etree = e
2020-04-07 16:58:58 +02:00
2020-09-20 13:15:44 +02:00
try :
from shlex import join as shlex_join
except ImportError as e :
shlex_join = e
2021-01-19 11:57:46 +00:00
try :
from shlex import quote as shlex_quote
except ImportError as e :
shlex_quote = e
2020-04-07 16:58:58 +02:00
class NoCommand ( Exception ) :
pass
BORING_INTERFACES = [
' org.freedesktop.DBus.Peer ' ,
' org.freedesktop.DBus.Introspectable ' ,
' org.freedesktop.DBus.Properties ' ,
]
2021-02-06 10:02:37 +01:00
RED = ' \x1b [31m '
GREEN = ' \x1b [32m '
YELLOW = ' \x1b [33m '
RESET = ' \x1b [39m '
2020-04-07 16:58:58 +02:00
2023-07-17 21:11:04 +02:00
arguments = None
2020-09-18 18:51:42 +02:00
def xml_parser ( ) :
return etree . XMLParser ( no_network = True ,
remove_comments = False ,
strip_cdata = False ,
resolve_entities = False )
2020-04-07 16:58:58 +02:00
def print_method ( declarations , elem , * , prefix , file , is_signal = False ) :
name = elem . get ( ' name ' )
klass = ' signal ' if is_signal else ' method '
declarations [ klass ] . append ( name )
2021-12-24 14:52:44 +09:00
# @org.freedesktop.systemd1.Privileged("true")
# SetShowStatus(in s mode);
for anno in elem . findall ( ' ./annotation ' ) :
anno_name = anno . get ( ' name ' )
anno_value = anno . get ( ' value ' )
print ( f ''' { prefix } @ { anno_name } ( " { anno_value } " ) ''' , file = file )
2020-04-07 16:58:58 +02:00
print ( f ''' { prefix } { name } ( ''' , file = file , end = ' ' )
lead = ' , \n ' + prefix + ' ' * len ( name ) + ' '
for num , arg in enumerate ( elem . findall ( ' ./arg ' ) ) :
argname = arg . get ( ' name ' )
if argname is None :
2023-07-17 21:11:04 +02:00
if arguments . print_errors :
2020-04-07 16:58:58 +02:00
print ( f ' method { name } : argument { num + 1 } has no name ' , file = sys . stderr )
argname = ' UNNAMED '
2023-07-17 21:11:04 +02:00
argtype = arg . get ( ' type ' )
2020-04-07 16:58:58 +02:00
if not is_signal :
direction = arg . get ( ' direction ' )
2023-07-17 21:11:04 +02:00
print ( f ''' { lead if num > 0 else ' ' } { direction : 3 } { argtype } { argname } ''' , file = file , end = ' ' )
2020-04-07 16:58:58 +02:00
else :
2023-07-17 21:11:04 +02:00
print ( f ''' { lead if num > 0 else ' ' } { argtype } { argname } ''' , file = file , end = ' ' )
2020-04-07 16:58:58 +02:00
2023-07-17 21:11:04 +02:00
print ( ' ); ' , file = file )
2020-04-07 16:58:58 +02:00
ACCESS_MAP = {
' read ' : ' readonly ' ,
' write ' : ' readwrite ' ,
}
2023-07-17 21:11:04 +02:00
def value_ellipsis ( prop_type ) :
if prop_type == ' s ' :
return " ' ... ' "
if prop_type [ 0 ] == ' a ' :
inner = value_ellipsis ( prop_type [ 1 : ] )
return f " [ { inner } { ' , ... ' if inner != ' ... ' else ' ' } ] "
2020-04-07 16:58:58 +02:00
return ' ... '
def print_property ( declarations , elem , * , prefix , file ) :
2023-07-17 21:11:04 +02:00
prop_name = elem . get ( ' name ' )
prop_type = elem . get ( ' type ' )
prop_access = elem . get ( ' access ' )
2020-04-07 16:58:58 +02:00
2023-07-17 21:11:04 +02:00
declarations [ ' property ' ] . append ( prop_name )
2020-04-07 16:58:58 +02:00
# @org.freedesktop.DBus.Property.EmitsChangedSignal("false")
# @org.freedesktop.systemd1.Privileged("true")
# readwrite b EnableWallMessages = false;
for anno in elem . findall ( ' ./annotation ' ) :
anno_name = anno . get ( ' name ' )
anno_value = anno . get ( ' value ' )
print ( f ''' { prefix } @ { anno_name } ( " { anno_value } " ) ''' , file = file )
2023-07-17 21:11:04 +02:00
prop_access = ACCESS_MAP . get ( prop_access , prop_access )
print ( f ''' { prefix } { prop_access } { prop_type } { prop_name } = { value_ellipsis ( prop_type ) } ; ''' , file = file )
2020-04-07 16:58:58 +02:00
2020-04-10 14:46:44 +02:00
def print_interface ( iface , * , prefix , file , print_boring , only_interface , declarations ) :
2020-04-07 16:58:58 +02:00
name = iface . get ( ' name ' )
2020-04-10 14:46:44 +02:00
is_boring = ( name in BORING_INTERFACES or
only_interface is not None and name != only_interface )
2020-04-07 16:58:58 +02:00
if is_boring and print_boring :
print ( f ''' { prefix } interface { name } {{ ... }} ; ''' , file = file )
2020-04-10 14:46:44 +02:00
2020-04-07 16:58:58 +02:00
elif not is_boring and not print_boring :
print ( f ''' { prefix } interface { name } {{ ''' , file = file )
prefix2 = prefix + ' '
for num , elem in enumerate ( iface . findall ( ' ./method ' ) ) :
if num == 0 :
print ( f ''' { prefix2 } methods: ''' , file = file )
print_method ( declarations , elem , prefix = prefix2 + ' ' , file = file )
for num , elem in enumerate ( iface . findall ( ' ./signal ' ) ) :
if num == 0 :
print ( f ''' { prefix2 } signals: ''' , file = file )
print_method ( declarations , elem , prefix = prefix2 + ' ' , file = file , is_signal = True )
for num , elem in enumerate ( iface . findall ( ' ./property ' ) ) :
if num == 0 :
print ( f ''' { prefix2 } properties: ''' , file = file )
print_property ( declarations , elem , prefix = prefix2 + ' ' , file = file )
print ( f ''' { prefix } }} ; ''' , file = file )
2023-09-24 09:13:45 +01:00
def check_documented ( document , declarations , stats , interface , missing_version ) :
missing = [ ]
2023-09-10 23:41:35 +01:00
2023-09-24 09:13:45 +01:00
sections = document . findall ( " refsect1 " )
history_section = document . find ( " refsect1[title = ' History ' ] " )
if history_section is not None :
sections . remove ( history_section )
2020-04-07 16:58:58 +02:00
for klass , items in declarations . items ( ) :
2020-08-27 19:21:21 +02:00
stats [ ' total ' ] + = len ( items )
2020-04-07 16:58:58 +02:00
for item in items :
2024-01-23 13:06:55 +01:00
if klass in ( ' method ' , ' signal ' ) :
2020-04-07 16:58:58 +02:00
elem = ' function '
item_repr = f ' { item } () '
2024-01-23 13:06:55 +01:00
# Find all functions/signals in <function> elements that are not
# suffixed with '()' and fix them
for section in sections :
element = section . find ( f " .// { elem } [. = ' { item } ' ] " )
if element is not None :
element . text = item_repr
2020-04-07 16:58:58 +02:00
elif klass == ' property ' :
elem = ' varname '
item_repr = item
else :
assert False , ( klass , item )
2023-09-24 09:13:45 +01:00
predicate = f " .// { elem } [. = ' { item_repr } ' ] "
if not any ( section . find ( predicate ) is not None for section in sections ) :
2023-07-17 21:11:04 +02:00
if arguments . print_errors :
2020-04-07 16:58:58 +02:00
print ( f ' { klass } { item } is not documented :( ' )
missing . append ( ( klass , item ) )
2023-09-24 09:13:45 +01:00
if history_section is None or history_section . find ( predicate ) is None :
missing_version . append ( f " { interface } . { item_repr } " )
2020-08-27 19:21:21 +02:00
stats [ ' missing ' ] + = len ( missing )
2020-04-07 16:58:58 +02:00
return missing
2020-04-10 14:46:44 +02:00
def xml_to_text ( destination , xml , * , only_interface = None ) :
2020-04-07 16:58:58 +02:00
file = io . StringIO ( )
declarations = collections . defaultdict ( list )
2020-04-18 20:19:50 +02:00
interfaces = [ ]
2020-04-07 16:58:58 +02:00
print ( f ''' node { destination } {{ ''' , file = file )
for print_boring in [ False , True ] :
for iface in xml . findall ( ' ./interface ' ) :
print_interface ( iface , prefix = ' ' , file = file ,
print_boring = print_boring ,
2020-04-10 14:46:44 +02:00
only_interface = only_interface ,
2020-04-07 16:58:58 +02:00
declarations = declarations )
2020-04-18 20:19:50 +02:00
name = iface . get ( ' name ' )
if not name in BORING_INTERFACES :
interfaces . append ( name )
2020-04-07 16:58:58 +02:00
2023-07-17 21:11:04 +02:00
print ( ''' }; ''' , file = file )
2020-04-07 16:58:58 +02:00
2020-04-18 20:19:50 +02:00
return file . getvalue ( ) , declarations , interfaces
2020-04-07 16:58:58 +02:00
2023-09-24 09:13:45 +01:00
def subst_output ( document , programlisting , stats , missing_version ) :
2020-04-24 12:09:07 +02:00
executable = programlisting . get ( ' executable ' , None )
if executable is None :
# Not our thing
2020-04-07 16:58:58 +02:00
return
2020-04-24 12:09:07 +02:00
executable = programlisting . get ( ' executable ' )
node = programlisting . get ( ' node ' )
interface = programlisting . get ( ' interface ' )
2020-04-07 16:58:58 +02:00
2023-07-17 21:11:04 +02:00
argv = [ f ' { arguments . build_dir } / { executable } ' , f ' --bus-introspect= { interface } ' ]
2021-01-19 11:57:46 +00:00
if isinstance ( shlex_join , Exception ) :
print ( f ' COMMAND: { " " . join ( shlex_quote ( arg ) for arg in argv ) } ' )
else :
print ( f ' COMMAND: { shlex_join ( argv ) } ' )
2020-04-07 16:58:58 +02:00
try :
2021-01-31 19:04:34 -08:00
out = subprocess . check_output ( argv , universal_newlines = True )
2020-04-24 12:09:07 +02:00
except FileNotFoundError :
print ( f ' { executable } not found, ignoring ' , file = sys . stderr )
2020-04-07 16:58:58 +02:00
return
2020-09-18 18:51:42 +02:00
xml = etree . fromstring ( out , parser = xml_parser ( ) )
2020-04-07 16:58:58 +02:00
2020-04-24 12:09:07 +02:00
new_text , declarations , interfaces = xml_to_text ( node , xml , only_interface = interface )
programlisting . text = ' \n ' + new_text + ' '
2020-04-07 16:58:58 +02:00
if declarations :
2023-09-24 09:13:45 +01:00
missing = check_documented ( document , declarations , stats , interface , missing_version )
2020-04-07 16:58:58 +02:00
parent = programlisting . getparent ( )
# delete old comments
for child in parent :
2023-07-17 21:11:04 +02:00
if child . tag is etree . Comment and ' Autogenerated ' in child . text :
2020-04-18 20:19:50 +02:00
parent . remove ( child )
2023-07-17 21:11:04 +02:00
if child . tag is etree . Comment and ' not documented ' in child . text :
2020-04-07 16:58:58 +02:00
parent . remove ( child )
2023-07-17 21:11:04 +02:00
if child . tag == " variablelist " and child . attrib . get ( " generated " , False ) == " True " :
2020-04-18 20:19:50 +02:00
parent . remove ( child )
# insert pointer for systemd-directives generation
the_tail = programlisting . tail #tail is erased by addnext, so save it here.
prev_element = etree . Comment ( " Autogenerated cross-references for systemd.directives, do not edit " )
programlisting . addnext ( prev_element )
programlisting . tail = the_tail
for interface in interfaces :
variablelist = etree . Element ( " variablelist " )
variablelist . attrib [ ' class ' ] = ' dbus-interface '
variablelist . attrib [ ' generated ' ] = ' True '
variablelist . attrib [ ' extra-ref ' ] = interface
prev_element . addnext ( variablelist )
prev_element . tail = the_tail
prev_element = variablelist
for decl_type , decl_list in declarations . items ( ) :
for declaration in decl_list :
variablelist = etree . Element ( " variablelist " )
variablelist . attrib [ ' class ' ] = ' dbus- ' + decl_type
variablelist . attrib [ ' generated ' ] = ' True '
2024-01-23 13:06:55 +01:00
if decl_type in ( ' method ' , ' signal ' ) :
2020-04-18 20:19:50 +02:00
variablelist . attrib [ ' extra-ref ' ] = declaration + ' () '
else :
variablelist . attrib [ ' extra-ref ' ] = declaration
prev_element . addnext ( variablelist )
prev_element . tail = the_tail
prev_element = variablelist
last_element = etree . Comment ( " End of Autogenerated section " )
prev_element . addnext ( last_element )
prev_element . tail = the_tail
last_element . tail = the_tail
2020-04-07 16:58:58 +02:00
# insert comments for undocumented items
for item in reversed ( missing ) :
comment = etree . Comment ( f ' { item [ 0 ] } { item [ 1 ] } is not documented! ' )
comment . tail = programlisting . tail
parent . insert ( parent . index ( programlisting ) + 1 , comment )
2023-09-24 09:13:45 +01:00
def process ( page , missing_version ) :
2020-04-07 16:58:58 +02:00
src = open ( page ) . read ( )
2020-09-18 18:51:42 +02:00
xml = etree . fromstring ( src , parser = xml_parser ( ) )
2020-04-07 16:58:58 +02:00
# print('parsing {}'.format(name), file=sys.stderr)
if xml . tag != ' refentry ' :
2023-07-17 21:11:04 +02:00
return None
2020-04-07 16:58:58 +02:00
2020-08-27 19:21:21 +02:00
stats = collections . Counter ( )
2020-04-07 16:58:58 +02:00
pls = xml . findall ( ' .//programlisting ' )
for pl in pls :
2023-09-24 09:13:45 +01:00
subst_output ( xml , pl , stats , missing_version )
2020-04-07 16:58:58 +02:00
out_text = etree . tostring ( xml , encoding = ' unicode ' )
2020-04-21 20:46:53 +02:00
# massage format to avoid some lxml whitespace handling idiosyncrasies
2020-04-07 16:58:58 +02:00
# https://bugs.launchpad.net/lxml/+bug/526799
out_text = ( src [ : src . find ( ' <refentryinfo ' ) ] +
out_text [ out_text . find ( ' <refentryinfo ' ) : ] +
' \n ' )
2023-07-17 21:11:04 +02:00
if not arguments . test :
2020-08-27 19:55:55 +02:00
with open ( page , ' w ' ) as out :
out . write ( out_text )
2020-04-07 16:58:58 +02:00
2023-07-17 21:11:04 +02:00
return { " stats " : stats , " modified " : out_text != src }
2020-08-27 19:21:21 +02:00
2020-08-27 19:27:18 +02:00
def parse_args ( ) :
p = argparse . ArgumentParser ( )
2020-08-27 19:55:55 +02:00
p . add_argument ( ' --test ' , action = ' store_true ' ,
help = ' only verify that everything is up2date ' )
2020-08-27 19:27:18 +02:00
p . add_argument ( ' --build-dir ' , default = ' build ' )
p . add_argument ( ' pages ' , nargs = ' + ' )
2020-08-27 20:15:30 +02:00
opts = p . parse_args ( )
opts . print_errors = not opts . test
return opts
2020-04-07 16:58:58 +02:00
2023-07-17 21:11:04 +02:00
def main ( ) :
# pylint: disable=global-statement
global arguments
arguments = parse_args ( )
2020-04-24 12:09:07 +02:00
2021-01-19 11:57:46 +00:00
for item in ( etree , shlex_quote ) :
2020-09-20 13:15:44 +02:00
if isinstance ( item , Exception ) :
print ( item , file = sys . stderr )
2023-07-17 21:11:04 +02:00
sys . exit ( 77 if arguments . test else 1 )
2020-09-18 18:51:42 +02:00
2023-07-17 21:11:04 +02:00
if not os . path . exists ( f ' { arguments . build_dir } /systemd ' ) :
sys . exit ( f " { arguments . build_dir } /systemd doesn ' t exist. Use --build-dir=. " )
2020-04-24 12:09:07 +02:00
2023-09-24 09:13:45 +01:00
missing_version = [ ]
stats = { page . split ( ' / ' ) [ - 1 ] : process ( page , missing_version ) for page in arguments . pages }
ignore_list = open ( os . path . join ( os . path . dirname ( __file__ ) , ' dbus_ignorelist ' ) ) . read ( ) . split ( )
missing_version = [ x for x in missing_version if x not in ignore_list ]
for missing in missing_version :
print ( f " { RED } Missing version information for { missing } { RESET } " )
if missing_version :
sys . exit ( 1 )
2020-08-27 19:21:21 +02:00
# Let's print all statistics at the end
mlen = max ( len ( page ) for page in stats )
2021-01-19 11:57:46 +00:00
total = sum ( ( item [ ' stats ' ] for item in stats . values ( ) ) , collections . Counter ( ) )
2023-07-17 21:11:04 +02:00
total = ' total ' , { " stats " : total , " modified " : False }
2021-02-06 09:42:54 +01:00
modified = [ ]
2023-07-17 21:11:04 +02:00
classification = ' OUTDATED ' if arguments . test else ' MODIFIED '
2020-08-27 19:55:55 +02:00
for page , info in sorted ( stats . items ( ) ) + [ total ] :
m = info [ ' stats ' ] [ ' missing ' ]
t = info [ ' stats ' ] [ ' total ' ]
2020-08-27 19:21:21 +02:00
p = page + ' : '
2021-02-06 09:42:54 +01:00
c = classification if info [ ' modified ' ] else ' '
2020-08-27 19:55:55 +02:00
if c :
2021-02-06 09:42:54 +01:00
modified . append ( page )
2021-02-06 10:02:37 +01:00
color = RED if m > t / 2 else ( YELLOW if m else GREEN )
print ( f ' { color } { p : { mlen + 1 } } { t - m } / { t } { c } { RESET } ' )
2020-08-27 19:55:55 +02:00
2023-07-17 21:11:04 +02:00
if arguments . test and modified :
sys . exit ( f ' Outdated pages: { " , " . join ( modified ) } \n '
f ' Hint: ninja -C { arguments . build_dir } update-dbus-docs ' )
if __name__ == ' __main__ ' :
main ( )