1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-04 05:18:06 +03:00
samba-mirror/source4/scripting/bin/w32err_code.py
2010-03-25 14:37:19 +11:00

362 lines
13 KiB
Python
Executable File

#!/usr/bin/env python
# Unix SMB/CIFS implementation.
# Copyright (C) Kamen Mazdrashki <kamen.mazdrashki@postpath.com> 2009
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""Import generete werror.h/doserr.c files from WSPP HTML"""
import re
import os
import sys
import urllib
import pprint
from xml.dom import minidom
from optparse import OptionParser, OptionGroup
_wspp_werror_url = 'http://msdn.microsoft.com/en-us/library/cc231199%28PROT.10%29.aspx'
class WerrorHtmlParser(object):
"""
Parses HTML from WSPP documentation generating dictionary of
dictionaries with following keys:
- "err_hex" - hex number (as string)
- "err_name" - error name
- "err_desc" - error long description
For the key of returned dictionary err_hex is used,
i.e. "hex-error-code-str" => {error dictionary object}
"""
ERROR_PREFIX = ['ERROR_', 'NERR_', 'FRS_', 'RPC_', 'EPT_', 'OR_', 'WAIT_TIMEOUT']
ERROR_REPLACE = ['ERROR_']
def __init__(self, opt):
self.opt = opt
self._errors_skipped = []
pass
def _is_error_code_name(self, err_name):
for pref in self.ERROR_PREFIX:
if err_name.startswith(pref):
return True
return False
def _make_werr_name(self, err_name):
err_name = err_name.upper()
for pref in self.ERROR_REPLACE:
if err_name.startswith(pref):
return err_name.replace(pref, 'WERR_', 1)
return 'WERR_' + err_name
def parse_url(self, url):
errors = {}
html = self._load_url(url)
# let minidom to parse the tree, should be:
# table -> tr -> td
# p -> [hex code, br, error code]
# p -> [description]
table_node = minidom.parseString(html)
for row_node in table_node.getElementsByTagName("tr"):
# verify we got right number of td elements
td_nodes = row_node.getElementsByTagName('td')
if len(td_nodes) != 2:
continue
# now get the real data
p_nodes = row_node.getElementsByTagName('p')
if len(p_nodes) != 2: continue
if len(p_nodes[0].childNodes) != 3: continue
if len(p_nodes[1].childNodes) != 1: continue
err_hex = str(p_nodes[0].childNodes[0].nodeValue)
err_name = str(p_nodes[0].childNodes[2].nodeValue)
err_desc = p_nodes[1].childNodes[0].nodeValue.encode('utf-8')
err_desc = err_desc.replace('"', '\\"').replace("\'", "\\'")
# do some checking
if not err_hex.startswith('0x'): continue
if not self._is_error_code_name(err_name):
self._errors_skipped.append("%s - %s - %d" % (err_name, err_hex, int(err_hex, 16)))
continue
# create entry
err_name = self._make_werr_name(err_name)
err_def = {'err_hex': err_hex,
'err_name': err_name,
'err_desc': err_desc,
'code': int(err_hex, 16)}
errors[err_def['code']] = err_def
# print skipped errors
if self.opt.print_skipped and len(self._errors_skipped):
print "\nErrors skipped during HTML parsing:"
pprint.pprint(self._errors_skipped)
print "\n"
return errors
def _load_url(self, url):
html_str = ""
try:
fp = urllib.urlopen(url)
for line in fp:
html_str += line.strip()
fp.close()
except IOError, e:
print "error loading url: " + e.strerror
pass
# currently ERROR codes are rendered as table
# locate table chunk with ERROR_SUCCESS
html = [x for x in html_str.split('<table ') if "ERROR_SUCCESS" in x]
html = '<table ' + html[0]
pos = html.find('</table>')
if pos == -1:
return '';
html = html[:pos] + '</table>'
# html clean up
html = re.sub(r'<a[^>]*>(.*?)</a>', r'\1', html)
return html
class WerrorGenerator(object):
"""
provides methods to generate parts of werror.h and doserr.c files
"""
FNAME_WERRORS = 'w32errors.lst'
FNAME_WERROR_DEFS = 'werror_defs.h'
FNAME_DOSERR_DEFS = 'doserr_defs.c'
FNAME_DOSERR_DESC = 'doserr_desc.c'
def __init__(self, opt):
self.opt = opt
self._out_dir = opt.out_dir
pass
def _open_out_file(self, fname):
fname = os.path.join(self._out_dir, fname)
return open(fname, "w")
def _gen_werrors_list(self, errors):
"""uses 'errors' dictionary to display list of Win32 Errors"""
fp = self._open_out_file(self.FNAME_WERRORS)
for err_code in sorted(errors.keys()):
err_name = errors[err_code]['err_name']
fp.write(err_name)
fp.write("\n")
fp.close()
def _gen_werror_defs(self, errors):
"""uses 'errors' dictionary to generate werror.h file"""
fp = self._open_out_file(self.FNAME_WERROR_DEFS)
for err_code in sorted(errors.keys()):
err_name = errors[err_code]['err_name']
err_hex = errors[err_code]['err_hex']
fp.write('#define %s\tW_ERROR(%s)' % (err_name, err_hex))
fp.write("\n")
fp.close()
def _gen_doserr_defs(self, errors):
"""uses 'errors' dictionary to generate defines in doserr.c file"""
fp = self._open_out_file(self.FNAME_DOSERR_DEFS)
for err_code in sorted(errors.keys()):
err_name = errors[err_code]['err_name']
fp.write('\t{ "%s", %s },' % (err_name, err_name))
fp.write("\n")
fp.close()
def _gen_doserr_descriptions(self, errors):
"""uses 'errors' dictionary to generate descriptions in doserr.c file"""
fp = self._open_out_file(self.FNAME_DOSERR_DESC)
for err_code in sorted(errors.keys()):
err_name = errors[err_code]['err_name']
fp.write('\t{ %s, "%s" },' % (err_name, errors[err_code]['err_desc']))
fp.write("\n")
fp.close()
def _lookup_error_by_name(self, err_name, defined_errors):
for err in defined_errors.itervalues():
if err['err_name'] == err_name:
return err
return None
def _filter_errors(self, errors, defined_errors):
"""
returns tuple (new_erros, diff_code_errors, diff_name_errors)
new_errors - dictionary of errors not in defined_errors
diff_code_errors - list of errors found in defined_errors
but with different value
diff_name_errors - list of errors found with same code in
defined_errors, but with different name
Most critical is diff_code_errors list to be empty!
"""
new_errors = {}
diff_code_errors = []
diff_name_errors = []
for err_def in errors.itervalues():
add_error = True
# try get defined error by code
if defined_errors.has_key(err_def['code']):
old_err = defined_errors[err_def['code']]
if err_def['err_name'] != old_err['err_name']:
warning = {'msg': 'New and Old errors has different error names',
'err_new': err_def,
'err_old': old_err}
diff_name_errors.append(warning)
# sanity check for errors with same name but different values
old_err = self._lookup_error_by_name(err_def['err_name'], defined_errors)
if old_err:
if err_def['code'] != old_err['code']:
warning = {'msg': 'New and Old error defs has different error value',
'err_new': err_def,
'err_old': old_err}
diff_code_errors.append(warning)
# exclude error already defined with same name
add_error = False
# do add the error in new_errors if everything is fine
if add_error:
new_errors[err_def['code']] = err_def
pass
return (new_errors, diff_code_errors, diff_name_errors)
def generate(self, errors):
# load already defined error codes
werr_parser = WerrorParser(self.opt)
(defined_errors,
no_value_errors) = werr_parser.load_err_codes(self.opt.werror_file)
if not defined_errors:
print "\nUnable to load existing errors file: %s" % self.opt.werror_file
sys.exit(1)
if self.opt.verbose and len(no_value_errors):
print "\nWarning: there are errors defines using macro value:"
pprint.pprint(no_value_errors)
print ""
# filter generated error codes
(new_errors,
diff_code_errors,
diff_name_errors) = self._filter_errors(errors, defined_errors)
if diff_code_errors:
print("\nFound %d errors with same names but different error values! Aborting."
% len(diff_code_errors))
pprint.pprint(diff_code_errors)
sys.exit(2)
if diff_name_errors:
print("\nFound %d errors with same values but different names (should be normal)"
% len(diff_name_errors))
pprint.pprint(diff_name_errors)
# finally generate output files
self._gen_werror_defs(new_errors)
self._gen_doserr_defs(new_errors)
self._gen_werrors_list(errors)
self._gen_doserr_descriptions(errors)
pass
class WerrorParser(object):
"""
Parses errors defined in werror.h file
"""
def __init__(self, opt):
self.opt = opt
pass
def _parse_werror_line(self, line):
m = re.match('#define[ \t]*(.*?)[ \t]*W_ERROR\((.*?)\)', line)
if not m or (len(m.groups()) != 2):
return None
if len(m.group(1)) == 0:
return None
if str(m.group(2)).startswith('0x'):
err_code = int(m.group(2), 16)
elif m.group(2).isdigit():
err_code = int(m.group(2))
else:
self.err_no_values.append(line)
return None
return {'err_name': str(m.group(1)),
'err_hex': "0x%08X" % err_code,
'code': err_code}
pass
def load_err_codes(self, fname):
"""
Returns tuple of:
dictionary of "hex_err_code" => {code, name}
"hex_err_code" is string
"code" is int value for the error
list of errors that was ignored for some reason
"""
# reset internal variables
self.err_no_values = []
err_codes = {}
fp = open(fname)
for line in fp.readlines():
err_def = self._parse_werror_line(line)
if err_def:
err_codes[err_def['code']] = err_def
fp.close();
return (err_codes, self.err_no_values)
def _generate_files(opt):
parser = WerrorHtmlParser(opt)
errors = parser.parse_url(opt.url)
out = WerrorGenerator(opt)
out.generate(errors)
pass
if __name__ == '__main__':
_cur_dir = os.path.abspath(os.path.dirname(__file__))
opt_parser = OptionParser(usage="usage: %prog [options]", version="%prog 0.3")
opt_group = OptionGroup(opt_parser, "Main options")
opt_group.add_option("--url", dest="url",
default=_wspp_werror_url,
help="url for w32 error codes html - may be local file")
opt_group.add_option("--out", dest="out_dir",
default=_cur_dir,
help="output dir for generated files")
opt_group.add_option("--werror", dest="werror_file",
default=os.path.join(_cur_dir, 'werror.h'),
help="path to werror.h file")
opt_group.add_option("--print_skipped",
action="store_true", dest="print_skipped", default=False,
help="print errors skipped during HTML parsing")
opt_group.add_option("-q", "--quiet",
action="store_false", dest="verbose", default=True,
help="don't print warnings to stdout")
opt_parser.add_option_group(opt_group)
(options, args) = opt_parser.parse_args()
# add some options to be used internally
options.err_defs_file = os.path.join(options.out_dir, WerrorGenerator.FNAME_WERROR_DEFS)
options.dos_defs_file = os.path.join(options.out_dir, WerrorGenerator.FNAME_DOSERR_DEFS)
options.dos_desc_file = os.path.join(options.out_dir, WerrorGenerator.FNAME_DOSERR_DESC)
# check options
_generate_files(options)