Welcome, guest | Sign In | My Account | Store | Cart
""" HelpMaker: Builds help message from comments in the code.
"""
"""
    There are two types of help messages:
        the general help,
        and the specific helps to corresponding functions/commands.

    The general help message is the first comment in the .py file,
    wrapped by the triple-double-quotes.
    The general help describes what the .py file is about, or gives a brief
    introduction of the module.

    A specific help message is wrapped by the triple-double-quotes, and has only
    one word (no space in between) in the first line of the comment.
    A specific help explains/illustrates the usage of a function/command.
    The specific helps are stored in an internal dictionary. The only-one-word
    in the first line of comment serves as the key to the help message in the
    internal dictionary.
"""

import sys
import re

class Phase:
    HAVE_NOTHING = 0
    HAVE_GENERAL_MSG = 1
    HAVE_SPECIFIC_MSG = 2

class HelpMaker:
    def __init__(self, filename):
        """ construct function
            construction function that takes the name of the .py file.
            The function screens the entire file and gets ready to
            display the comments as help messages.
        """
        try:
            with open(filename) as pyfile:
                self._comment_head = re.compile(r"""^\s*\"{3}""")
                self._key_word = re.compile(r"""^\s*\"{3}\s*\S{1,}\s*\Z""")
                self._phase = Phase.HAVE_NOTHING
                self._is_comment = False
                self._general_help = ''
                self._specific_help = dict()
                self._tmp_key = ''
                self._tmp_msg = ''
                self._tmp_index = 0

                for line in pyfile:
                    if self._phase is Phase.HAVE_NOTHING:
                        self._general_help_filter(line)
                    elif self._phase is Phase.HAVE_GENERAL_MSG:
                        self._specific_help_filter(line)
        except IOError:
            print('Could not find the file: %s'%filename)

    """ available_help
        Generate all the items with help messages
    """
    def available_help(self):
        return self._specific_help.keys()

    def show_help(self, showitem = None):
        """ show_help
            Without any input this funtion shows general common help message.
            When the showitem is given, the corresponding help message will be
            displayed, if exists.
        """
        if showitem is None:
            print(self._general_help)
        else:
            try:
                print(self._specific_help[showitem])
            except KeyError:
                print('No help on %s available'%showitem)

    def _general_help_filter(self, line):
        """ helper function that finds out the general help message
        """
        if self._comment_head.search(line) is not None:
            self._tmp_indent = line.find('"') + 3
            self._is_comment = not self._is_comment

        if self._is_comment is True:
            self._general_help += line[self._tmp_indent:]

        if self._general_help is not '' and self._is_comment is False:
            self._phase = Phase.HAVE_GENERAL_MSG

    def _specific_help_filter(self, line):
        """ helper function that finds out the speific help message
        """
        if self._key_word.search(line) is not None:
            self._tmp_indent = line.find('"') + 3
            self._tmp_key = line[self._tmp_indent:].strip()
            self._is_comment = True
            return

        if self._is_comment is True:
            if self._comment_head.search(line) is not None:
                self._specific_help[self._tmp_key] = self._tmp_msg
                self._tmp_msg = ''
                self._is_comment = False
            else:
                self._tmp_msg += line[self._tmp_indent:]

if __name__ == '__main__':
    hm = HelpMaker(sys.argv[0])
    print('General help of HelpMaker:')
    hm.show_help()

    print("Functions with help message:")
    for item in hm.available_help():
        print(item)
    print('')
    print('Specific help of HelpMaker:show_help():')
    hm.show_help('show_help')
    print('Specific help of a nonexist function:')
    hm.show_help('nonsense')

History