#!/usr/bin/python """ = Siteshwar Vashisht = 2012 Copyright (c) 2012, Siteshwar Vashisht All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. """ """ Run me like this: ./create_manpage_completions.py /usr/share/man/man1/* > test2.out """ import sys, re, os.path, gzip, traceback # This gets set to the name of the command that we are currently executing CMDNAME = "" # builtcommand writes into this global variable, yuck built_command_output = [] def compileAndSearch(regex, input): options_section_regex = re.compile(regex , re.DOTALL) options_section_matched = re.search( options_section_regex, input) return options_section_matched def unquoteDoubleQuotes(data): if (len(data) < 2): return data if data[0] == '"' and data[len(data)-1] == '"': data = data[1:len(data)-1] return data def unquoteSingleQuotes(data): if (len(data) < 2): return data if data[0] == '`' and data[len(data)-1] == '\'': data = data[1:len(data)-1] return data def output_complete_command(cmdname, args, description, output_list): comps = ['complete -c', cmdname] comps.extend(args) comps.append('--description') comps.append("'" + description + "'") output_list.append(' '.join(comps)) def builtcommand(options, description): # print "Options are: ", options optionlist = re.split(" |,|\"|=|[|]", options) optionlist = [x for x in optionlist if x.startswith('-')] if len(optionlist) == 0: return for i in range(0, len(optionlist)): if optionlist[i][0:2] == "--": optionlist[i] = "-l " + optionlist[i][2:] else: optionlist[i] = "-s " + optionlist[i][1:] description = description.replace('"','\\"') first_period = description.find(".") # print "First period at: ", first_period if first_period >= 45 or first_period == -1 and len(description) > 45: description = description[:45] + "... [See Man Page]" elif first_period >= 0: description = description[:first_period] output_complete_command(CMDNAME, optionlist, description, built_command_output) def removeGroffFormatting(data): # data = data.replace("\fI","") # data = data.replace("\fP","") data = data.replace("\\fI","") data = data.replace("\\fP","") data = data.replace("\\f1","") data = data.replace("\\fB","") data = data.replace("\\fR","") data = data.replace("\\e","") data = re.sub(".PD( \d+)","",data) data = data.replace(".BI","") data = data.replace(".BR","") data = data.replace("0.5i","") data = data.replace(".rb","") data = data.replace("\\^","") data = data.replace("{ ","") data = data.replace(" }","") data = data.replace("\ ","") data = data.replace("\-","-") data = data.replace("\&","") data = data.replace(".B","") data = data.replace("\-","-") data = data.replace(".I","") data = data.replace("\f","") return data class ManParser: def isMyType(self, manpage): return False def parseManPage(self, manpage): return False def name(self): return "no-name" class Type1ManParser(ManParser): def isMyType(self, manpage): # print manpage options_section_matched = compileAndSearch("\.SH \"OPTIONS\"(.*?)", manpage) if options_section_matched == None: return False else: return True def parseManPage(self, manpage): options_section_regex = re.compile( "\.SH \"OPTIONS\"(.*?)(\.SH|\Z)", re.DOTALL) options_section_matched = re.search( options_section_regex, manpage) options_section = options_section_matched.group(0) # print options_section options_parts_regex = re.compile("\.PP(.*?)\.RE", re.DOTALL) options_matched = re.search(options_parts_regex, options_section) # print options_matched print >> sys.stderr, "Command is ", CMDNAME if options_matched == None: print >> sys.stderr, "Unable to find options" if( self.fallback(options_section) ): return True elif (self.fallback2(options_section) ): return True return False while (options_matched != None): # print len(options_matched.groups()) # print options_matched.group() data = options_matched.group(1) last_dotpp_index = data.rfind(".PP") if (last_dotpp_index != -1): data = data[last_dotpp_index+3:] data = removeGroffFormatting(data) data = data.split(".RS 4") # print data if (len (data) > 1): #and len(data[1]) <= 300): optionName = data[0].strip() if ( optionName.find("-") == -1): print >> sys.stderr, optionName, " doesn't contain - " # return False else: optionName = unquoteDoubleQuotes(optionName) optionName = unquoteSingleQuotes(optionName) optionDescription = data[1].strip().replace("\n"," ") # print >> sys.stderr, "Option: ", optionName," Description: ", optionDescription , '\n' builtcommand(optionName, optionDescription) else: print >> sys.stderr, "Unable to split option from description" return False options_section = options_section[options_matched.end()-3:] options_matched = re.search(options_parts_regex, options_section) def fallback(self, options_section): print >> sys.stderr, "Falling Back" options_parts_regex = re.compile("\.TP( \d+)?(.*?)\.TP", re.DOTALL) options_matched = re.search(options_parts_regex, options_section) if options_matched == None: print >> sys.stderr, "Still not found" return False while options_matched != None: data = options_matched.group(2) data = removeGroffFormatting(data) data = data.strip() data = data.split("\n",1) if (len(data)>1 and len(data[1].strip())>0): # and len(data[1])<400): optionName = data[0].strip() if ( optionName.find("-") == -1): print >> sys.stderr, optionName, "doesn't contains -" else: optionName = unquoteDoubleQuotes(optionName) optionName = unquoteSingleQuotes(optionName) optionDescription = data[1].strip().replace("\n"," ") # print "Option: ", optionName," Description: ", optionDescription , '\n' builtcommand(optionName, optionDescription) else: print >> sys.stderr, data print >> sys.stderr, "Unable to split option from description" return False options_section = options_section[options_matched.end()-3:] options_matched = re.search(options_parts_regex, options_section) return True def fallback2(self, options_section): print >> sys.stderr, "Falling Back2" ix_remover_regex = re.compile("\.IX.*") trailing_num_regex = re.compile('\\d+$') options_parts_regex = re.compile("\.IP (.*?)\.IP", re.DOTALL) options_section = re.sub(ix_remover_regex, "", options_section) options_matched = re.search(options_parts_regex, options_section) if options_matched == None: print >> sys.stderr, "Still not found2" return False while options_matched != None: data = options_matched.group(1) # print "Data is : ", data data = removeGroffFormatting(data) data = data.strip() data = data.split("\n",1) if (len(data)>1 and len(data[1].strip())>0): # and len(data[1])<400): # print "Data[0] is: ", data[0] # data = re.sub(trailing_num_regex, "", data) optionName = re.sub(trailing_num_regex, "", data[0].strip()) if ( optionName.find("-") == -1): print >> sys.stderr, optionName, "doesn't contains -" else: optionName = optionName.strip() optionName = unquoteDoubleQuotes(optionName) optionName = unquoteSingleQuotes(optionName) optionDescription = data[1].strip().replace("\n"," ") # print "Option: ", optionName," Description: ", optionDescription , '\n' builtcommand(optionName, optionDescription) else: # print data print >> sys.stderr, "Unable to split option from description" return False options_section = options_section[options_matched.end()-3:] options_matched = re.search(options_parts_regex, options_section) return True def name(self): return "Type1" class Type2ManParser(ManParser): def isMyType(self, manpage): options_section_matched = compileAndSearch("\.SH OPTIONS(.*?)", manpage) if options_section_matched == None: return False else: return True def parseManPage(self, manpage): options_section_regex = re.compile( "\.SH OPTIONS(.*?)(\.SH|\Z)", re.DOTALL) options_section_matched = re.search( options_section_regex, manpage) # if (options_section_matched == None): # print "Falling Back" # options_section_regex = re.compile( "\.SH OPTIONS(.*?)$", re.DOTALL) # options_section_matched = re.search( options_section_regex, manpage) # print manpage options_section = options_section_matched.group(1) # print options_section # print options_section # sys.exit(1) # options_parts_regex = re.compile("\.TP(.*?)\.TP", re.DOTALL) options_parts_regex = re.compile("\.[I|T]P( \d+(\.\d)?i?)?(.*?)\.[I|T]P", re.DOTALL) # options_parts_regex = re.compile("\.TP(.*?)[(\.TP)|(\.SH)]", re.DOTALL) options_matched = re.search(options_parts_regex, options_section) print >> sys.stderr, "Command is ", CMDNAME if options_matched == None: print >> sys.stderr, self.name() + ": Unable to find options" return False while (options_matched != None): # print len(options_matched.groups()) data = options_matched.group(3) data = removeGroffFormatting(data) data = data.strip() data = data.split("\n",1) # print >> sys.stderr, data if (len(data)>1 and len(data[1].strip())>0): # and len(data[1])<400): optionName = data[0].strip() if ( optionName.find("-") == -1): print >> sys.stderr, optionName, "doesn't contains -" else: optionName = unquoteDoubleQuotes(optionName) optionName = unquoteSingleQuotes(optionName) optionDescription = data[1].strip().replace("\n"," ") # print "Option: ", optionName," Description: ", optionDescription , '\n' builtcommand(optionName, optionDescription) else: print >> sys.stderr, data print >> sys.stderr, "Unable to split option from description" # return False options_section = options_section[options_matched.end()-3:] options_matched = re.search(options_parts_regex, options_section) def name(self): return "Type2" class Type3ManParser(ManParser): def isMyType(self, manpage): options_section_matched = compileAndSearch("\.SH DESCRIPTION(.*?)", manpage) if options_section_matched == None: return False else: return True def parseManPage(self, manpage): options_section_regex = re.compile( "\.SH DESCRIPTION(.*?)(\.SH|\Z)", re.DOTALL) options_section_matched = re.search( options_section_regex, manpage) options_section = options_section_matched.group(1) # print options_section # sys.exit(1) options_parts_regex = re.compile("\.TP(.*?)\.TP", re.DOTALL) options_matched = re.search(options_parts_regex, options_section) print >> sys.stderr, "Command is ", CMDNAME if options_matched == None: print >> sys.stderr, "Unable to find options section" return False while (options_matched != None): # print len(options_matched.groups()) data = options_matched.group(1) data = removeGroffFormatting(data) data = data.strip() data = data.split("\n",1) if (len(data)>1): # and len(data[1])<400): optionName = data[0].strip() if ( optionName.find("-") == -1): print >> sys.stderr, optionName, "doesn't contains -" else: optionName = unquoteDoubleQuotes(optionName) optionName = unquoteSingleQuotes(optionName) optionDescription = data[1].strip().replace("\n"," ") # print >> sys.stderr, "Option: ", optionName," Description: ", optionDescription , '\n' builtcommand(optionName, optionDescription) else: print >> sys.stderr, "Unable to split option from description" return False options_section = options_section[options_matched.end()-3:] options_matched = re.search(options_parts_regex, options_section) def name(self): return "Type3" class Type4ManParser(ManParser): def isMyType(self, manpage): options_section_matched = compileAndSearch("\.SH FUNCTION LETTERS(.*?)", manpage) if options_section_matched == None: return False else: return True def parseManPage(self, manpage): options_section_regex = re.compile( "\.SH FUNCTION LETTERS(.*?)(\.SH|\Z)", re.DOTALL) options_section_matched = re.search( options_section_regex, manpage) options_section = options_section_matched.group(1) # print options_section # sys.exit(1) options_parts_regex = re.compile("\.TP(.*?)\.TP", re.DOTALL) options_matched = re.search(options_parts_regex, options_section) print >> sys.stderr, "Command is ", CMDNAME if options_matched == None: print >> sys.stderr, "Unable to find options section" return False while (options_matched != None): # print len(options_matched.groups()) data = options_matched.group(1) data = removeGroffFormatting(data) data = data.strip() data = data.split("\n",1) if (len(data)>1): # and len(data[1])<400): optionName = data[0].strip() if ( optionName.find("-") == -1): print >> sys.stderr, optionName, "doesn't contains -" else: optionName = unquoteDoubleQuotes(optionName) optionName = unquoteSingleQuotes(optionName) optionDescription = data[1].strip().replace("\n"," ") # print "Option: ", optionName," Description: ", optionDescription , '\n' builtcommand(optionName, optionDescription) else: print >> sys.stderr, "Unable to split option from description" return False options_section = options_section[options_matched.end()-3:] options_matched = re.search(options_parts_regex, options_section) return True def name(self): return "Type4" class TypeMacManParser(ManParser): def isMyType(self, manpage): options_section_matched = compileAndSearch("\.Sh DESCRIPTION", manpage) return options_section_matched != None def trim_groff(self, line): # Remove initial period if line.startswith('.'): line = line[1:] # Skip leading groff crud while re.match('[A-Z][a-z]\s', line): line = line[3:] return line def is_option(self, line): return line.startswith('.It Fl') def parseManPage(self, manpage): lines = manpage.splitlines() # Discard lines until we get to ".sh Description" while lines and not lines[0].startswith('.Sh DESCRIPTION'): lines.pop(0) while lines: # Pop until we get to the next option while lines and not self.is_option(lines[0]): lines.pop(0) if not lines: continue # Extract the name name = self.trim_groff(lines.pop(0)).strip() # Extract the description desc = '' while lines and not self.is_option(lines[0]): # print "*", lines[0] desc = desc + lines.pop(0) #print "name: ", name if name == '-': # Skip double -- arguments continue elif len(name) > 1: # Output the command builtcommand('--' + name, desc) elif len(name) == 1: builtcommand('-' + name, desc) def name(self): return "Darwin man parser" def parse_manpage_at_path(manpage_path): filename = os.path.basename(manpage_path) # Get the "base" command, e.g. gcc.1.gz -> gcc global CMDNAME CMDNAME = filename.split('.', 1)[0] print >> sys.stderr, "Considering " + manpage_path if manpage_path.endswith('.gz'): fd = gzip.open(manpage_path, 'r') else: fd = open(manpage_path, 'r') manpage = fd.read() fd.close() parsers = [Type1ManParser(), Type2ManParser(), Type4ManParser(), Type3ManParser(), TypeMacManParser()] parserToUse = None # Get the "base" command, e.g. gcc.1.gz -> gcc cmd_base = CMDNAME.split('.', 1)[0] ignoredcommands = ["cc", "g++", "gcc", "c++", "cpp", "emacs", "gprof", "wget", "ld", "awk"] if cmd_base in ignoredcommands: return idx = 0 for parser in parsers: if parser.isMyType(manpage): parserToUse = parser # print "Type is: ", parser.name() break idx += 1 # Clear the output list built_command_output[:] = [] if parserToUse == None: print >> sys.stderr, manpage_path, " : Not supported" else: if parserToUse.parseManPage(manpage) == False: print >> sys.stderr, "Type%d : %s is unparsable" % (idx, manpage_path) elif built_command_output: built_command_output.insert(0, "# %s: %s" % (CMDNAME, parser.name())) for line in built_command_output: pass #print line #print '' print >> sys.stderr, manpage_path, " parsed successfully" def compare_paths(a, b): """ Compare two paths by their base name, case insensitive """ return cmp(os.path.basename(a).lower(), os.path.basename(b).lower()) def parse_and_output_man_pages(paths): paths.sort(compare_paths) for manpage_path in paths: try: parse_manpage_at_path(manpage_path) except IOError: print >> sys.stderr, 'Cannot open ', manpage_path except: print >> sys.stderr, "Error parsing %s: %s" % (manpage_path, sys.exc_info()[0]) traceback.print_exc(file=sys.stdout) if __name__ == "__main__": paths = sys.argv[1:] parse_and_output_man_pages(paths) # Profiling code # import cProfile, pstats # cProfile.run('parse_and_output_man_pages(paths)', 'fooprof') # p = pstats.Stats('fooprof') # p.sort_stats('cumulative').print_stats(10)