446 lines
15 KiB
Python
446 lines
15 KiB
Python
|
'''Run human tests of Idle's window, dialog, and popup widgets.
|
||
|
|
||
|
run(*tests)
|
||
|
Create a master Tk window. Within that, run each callable in tests
|
||
|
after finding the matching test spec in this file. If tests is empty,
|
||
|
run an htest for each spec dict in this file after finding the matching
|
||
|
callable in the module named in the spec. Close the window to skip or
|
||
|
end the test.
|
||
|
|
||
|
In a tested module, let X be a global name bound to a callable (class
|
||
|
or function) whose .__name__ attribute is also X (the usual situation).
|
||
|
The first parameter of X must be 'parent'. When called, the parent
|
||
|
argument will be the root window. X must create a child Toplevel
|
||
|
window (or subclass thereof). The Toplevel may be a test widget or
|
||
|
dialog, in which case the callable is the corresponding class. Or the
|
||
|
Toplevel may contain the widget to be tested or set up a context in
|
||
|
which a test widget is invoked. In this latter case, the callable is a
|
||
|
wrapper function that sets up the Toplevel and other objects. Wrapper
|
||
|
function names, such as _editor_window', should start with '_'.
|
||
|
|
||
|
|
||
|
End the module with
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
<unittest, if there is one>
|
||
|
from idlelib.idle_test.htest import run
|
||
|
run(X)
|
||
|
|
||
|
To have wrapper functions and test invocation code ignored by coveragepy
|
||
|
reports, put '# htest #' on the def statement header line.
|
||
|
|
||
|
def _wrapper(parent): # htest #
|
||
|
|
||
|
Also make sure that the 'if __name__' line matches the above. Then have
|
||
|
make sure that .coveragerc includes the following.
|
||
|
|
||
|
[report]
|
||
|
exclude_lines =
|
||
|
.*# htest #
|
||
|
if __name__ == .__main__.:
|
||
|
|
||
|
(The "." instead of "'" is intentional and necessary.)
|
||
|
|
||
|
|
||
|
To run any X, this file must contain a matching instance of the
|
||
|
following template, with X.__name__ prepended to '_spec'.
|
||
|
When all tests are run, the prefix is use to get X.
|
||
|
|
||
|
_spec = {
|
||
|
'file': '',
|
||
|
'kwds': {'title': ''},
|
||
|
'msg': ""
|
||
|
}
|
||
|
|
||
|
file (no .py): run() imports file.py.
|
||
|
kwds: augmented with {'parent':root} and passed to X as **kwds.
|
||
|
title: an example kwd; some widgets need this, delete if not.
|
||
|
msg: master window hints about testing the widget.
|
||
|
|
||
|
|
||
|
Modules and classes not being tested at the moment:
|
||
|
pyshell.PyShellEditorWindow
|
||
|
debugger.Debugger
|
||
|
autocomplete_w.AutoCompleteWindow
|
||
|
outwin.OutputWindow (indirectly being tested with grep test)
|
||
|
'''
|
||
|
|
||
|
import idlelib.pyshell # Set Windows DPI awareness before Tk().
|
||
|
from importlib import import_module
|
||
|
import textwrap
|
||
|
import tkinter as tk
|
||
|
from tkinter.ttk import Scrollbar
|
||
|
tk.NoDefaultRoot()
|
||
|
|
||
|
AboutDialog_spec = {
|
||
|
'file': 'help_about',
|
||
|
'kwds': {'title': 'help_about test',
|
||
|
'_htest': True,
|
||
|
},
|
||
|
'msg': "Test every button. Ensure Python, TK and IDLE versions "
|
||
|
"are correctly displayed.\n [Close] to exit.",
|
||
|
}
|
||
|
|
||
|
# TODO implement ^\; adding '<Control-Key-\\>' to function does not work.
|
||
|
_calltip_window_spec = {
|
||
|
'file': 'calltip_w',
|
||
|
'kwds': {},
|
||
|
'msg': "Typing '(' should display a calltip.\n"
|
||
|
"Typing ') should hide the calltip.\n"
|
||
|
"So should moving cursor out of argument area.\n"
|
||
|
"Force-open-calltip does not work here.\n"
|
||
|
}
|
||
|
|
||
|
_module_browser_spec = {
|
||
|
'file': 'browser',
|
||
|
'kwds': {},
|
||
|
'msg': "Inspect names of module, class(with superclass if "
|
||
|
"applicable), methods and functions.\nToggle nested items.\n"
|
||
|
"Double clicking on items prints a traceback for an exception "
|
||
|
"that is ignored."
|
||
|
}
|
||
|
|
||
|
_color_delegator_spec = {
|
||
|
'file': 'colorizer',
|
||
|
'kwds': {},
|
||
|
'msg': "The text is sample Python code.\n"
|
||
|
"Ensure components like comments, keywords, builtins,\n"
|
||
|
"string, definitions, and break are correctly colored.\n"
|
||
|
"The default color scheme is in idlelib/config-highlight.def"
|
||
|
}
|
||
|
|
||
|
CustomRun_spec = {
|
||
|
'file': 'query',
|
||
|
'kwds': {'title': 'Customize query.py Run',
|
||
|
'_htest': True},
|
||
|
'msg': "Enter with <Return> or [Run]. Print valid entry to Shell\n"
|
||
|
"Arguments are parsed into a list\n"
|
||
|
"Mode is currently restart True or False\n"
|
||
|
"Close dialog with valid entry, <Escape>, [Cancel], [X]"
|
||
|
}
|
||
|
|
||
|
ConfigDialog_spec = {
|
||
|
'file': 'configdialog',
|
||
|
'kwds': {'title': 'ConfigDialogTest',
|
||
|
'_htest': True,},
|
||
|
'msg': "IDLE preferences dialog.\n"
|
||
|
"In the 'Fonts/Tabs' tab, changing font face, should update the "
|
||
|
"font face of the text in the area below it.\nIn the "
|
||
|
"'Highlighting' tab, try different color schemes. Clicking "
|
||
|
"items in the sample program should update the choices above it."
|
||
|
"\nIn the 'Keys', 'General' and 'Extensions' tabs, test settings "
|
||
|
"of interest."
|
||
|
"\n[Ok] to close the dialog.[Apply] to apply the settings and "
|
||
|
"and [Cancel] to revert all changes.\nRe-run the test to ensure "
|
||
|
"changes made have persisted."
|
||
|
}
|
||
|
|
||
|
# TODO Improve message
|
||
|
_dyn_option_menu_spec = {
|
||
|
'file': 'dynoption',
|
||
|
'kwds': {},
|
||
|
'msg': "Select one of the many options in the 'old option set'.\n"
|
||
|
"Click the button to change the option set.\n"
|
||
|
"Select one of the many options in the 'new option set'."
|
||
|
}
|
||
|
|
||
|
# TODO edit wrapper
|
||
|
_editor_window_spec = {
|
||
|
'file': 'editor',
|
||
|
'kwds': {},
|
||
|
'msg': "Test editor functions of interest.\n"
|
||
|
"Best to close editor first."
|
||
|
}
|
||
|
|
||
|
GetKeysDialog_spec = {
|
||
|
'file': 'config_key',
|
||
|
'kwds': {'title': 'Test keybindings',
|
||
|
'action': 'find-again',
|
||
|
'current_key_sequences': [['<Control-Key-g>', '<Key-F3>', '<Control-Key-G>']],
|
||
|
'_htest': True,
|
||
|
},
|
||
|
'msg': "Test for different key modifier sequences.\n"
|
||
|
"<nothing> is invalid.\n"
|
||
|
"No modifier key is invalid.\n"
|
||
|
"Shift key with [a-z],[0-9], function key, move key, tab, space "
|
||
|
"is invalid.\nNo validity checking if advanced key binding "
|
||
|
"entry is used."
|
||
|
}
|
||
|
|
||
|
_grep_dialog_spec = {
|
||
|
'file': 'grep',
|
||
|
'kwds': {},
|
||
|
'msg': "Click the 'Show GrepDialog' button.\n"
|
||
|
"Test the various 'Find-in-files' functions.\n"
|
||
|
"The results should be displayed in a new '*Output*' window.\n"
|
||
|
"'Right-click'->'Go to file/line' anywhere in the search results "
|
||
|
"should open that file \nin a new EditorWindow."
|
||
|
}
|
||
|
|
||
|
HelpSource_spec = {
|
||
|
'file': 'query',
|
||
|
'kwds': {'title': 'Help name and source',
|
||
|
'menuitem': 'test',
|
||
|
'filepath': __file__,
|
||
|
'used_names': {'abc'},
|
||
|
'_htest': True},
|
||
|
'msg': "Enter menu item name and help file path\n"
|
||
|
"'', > than 30 chars, and 'abc' are invalid menu item names.\n"
|
||
|
"'' and file does not exist are invalid path items.\n"
|
||
|
"Any url ('www...', 'http...') is accepted.\n"
|
||
|
"Test Browse with and without path, as cannot unittest.\n"
|
||
|
"[Ok] or <Return> prints valid entry to shell\n"
|
||
|
"[Cancel] or <Escape> prints None to shell"
|
||
|
}
|
||
|
|
||
|
_io_binding_spec = {
|
||
|
'file': 'iomenu',
|
||
|
'kwds': {},
|
||
|
'msg': "Test the following bindings.\n"
|
||
|
"<Control-o> to open file from dialog.\n"
|
||
|
"Edit the file.\n"
|
||
|
"<Control-p> to print the file.\n"
|
||
|
"<Control-s> to save the file.\n"
|
||
|
"<Alt-s> to save-as another file.\n"
|
||
|
"<Control-c> to save-copy-as another file.\n"
|
||
|
"Check that changes were saved by opening the file elsewhere."
|
||
|
}
|
||
|
|
||
|
_linenumbers_drag_scrolling_spec = {
|
||
|
'file': 'sidebar',
|
||
|
'kwds': {},
|
||
|
'msg': textwrap.dedent("""\
|
||
|
1. Click on the line numbers and drag down below the edge of the
|
||
|
window, moving the mouse a bit and then leaving it there for a while.
|
||
|
The text and line numbers should gradually scroll down, with the
|
||
|
selection updated continuously.
|
||
|
|
||
|
2. With the lines still selected, click on a line number above the
|
||
|
selected lines. Only the line whose number was clicked should be
|
||
|
selected.
|
||
|
|
||
|
3. Repeat step #1, dragging to above the window. The text and line
|
||
|
numbers should gradually scroll up, with the selection updated
|
||
|
continuously.
|
||
|
|
||
|
4. Repeat step #2, clicking a line number below the selection."""),
|
||
|
}
|
||
|
|
||
|
_multi_call_spec = {
|
||
|
'file': 'multicall',
|
||
|
'kwds': {},
|
||
|
'msg': "The following actions should trigger a print to console or IDLE"
|
||
|
" Shell.\nEntering and leaving the text area, key entry, "
|
||
|
"<Control-Key>,\n<Alt-Key-a>, <Control-Key-a>, "
|
||
|
"<Alt-Control-Key-a>, \n<Control-Button-1>, <Alt-Button-1> and "
|
||
|
"focusing out of the window\nare sequences to be tested."
|
||
|
}
|
||
|
|
||
|
_multistatus_bar_spec = {
|
||
|
'file': 'statusbar',
|
||
|
'kwds': {},
|
||
|
'msg': "Ensure presence of multi-status bar below text area.\n"
|
||
|
"Click 'Update Status' to change the multi-status text"
|
||
|
}
|
||
|
|
||
|
_object_browser_spec = {
|
||
|
'file': 'debugobj',
|
||
|
'kwds': {},
|
||
|
'msg': "Double click on items upto the lowest level.\n"
|
||
|
"Attributes of the objects and related information "
|
||
|
"will be displayed side-by-side at each level."
|
||
|
}
|
||
|
|
||
|
_path_browser_spec = {
|
||
|
'file': 'pathbrowser',
|
||
|
'kwds': {},
|
||
|
'msg': "Test for correct display of all paths in sys.path.\n"
|
||
|
"Toggle nested items upto the lowest level.\n"
|
||
|
"Double clicking on an item prints a traceback\n"
|
||
|
"for an exception that is ignored."
|
||
|
}
|
||
|
|
||
|
_percolator_spec = {
|
||
|
'file': 'percolator',
|
||
|
'kwds': {},
|
||
|
'msg': "There are two tracers which can be toggled using a checkbox.\n"
|
||
|
"Toggling a tracer 'on' by checking it should print tracer "
|
||
|
"output to the console or to the IDLE shell.\n"
|
||
|
"If both the tracers are 'on', the output from the tracer which "
|
||
|
"was switched 'on' later, should be printed first\n"
|
||
|
"Test for actions like text entry, and removal."
|
||
|
}
|
||
|
|
||
|
Query_spec = {
|
||
|
'file': 'query',
|
||
|
'kwds': {'title': 'Query',
|
||
|
'message': 'Enter something',
|
||
|
'text0': 'Go',
|
||
|
'_htest': True},
|
||
|
'msg': "Enter with <Return> or [Ok]. Print valid entry to Shell\n"
|
||
|
"Blank line, after stripping, is ignored\n"
|
||
|
"Close dialog with valid entry, <Escape>, [Cancel], [X]"
|
||
|
}
|
||
|
|
||
|
|
||
|
_replace_dialog_spec = {
|
||
|
'file': 'replace',
|
||
|
'kwds': {},
|
||
|
'msg': "Click the 'Replace' button.\n"
|
||
|
"Test various replace options in the 'Replace dialog'.\n"
|
||
|
"Click [Close] or [X] to close the 'Replace Dialog'."
|
||
|
}
|
||
|
|
||
|
_search_dialog_spec = {
|
||
|
'file': 'search',
|
||
|
'kwds': {},
|
||
|
'msg': "Click the 'Search' button.\n"
|
||
|
"Test various search options in the 'Search dialog'.\n"
|
||
|
"Click [Close] or [X] to close the 'Search Dialog'."
|
||
|
}
|
||
|
|
||
|
_searchbase_spec = {
|
||
|
'file': 'searchbase',
|
||
|
'kwds': {},
|
||
|
'msg': "Check the appearance of the base search dialog\n"
|
||
|
"Its only action is to close."
|
||
|
}
|
||
|
|
||
|
_scrolled_list_spec = {
|
||
|
'file': 'scrolledlist',
|
||
|
'kwds': {},
|
||
|
'msg': "You should see a scrollable list of items\n"
|
||
|
"Selecting (clicking) or double clicking an item "
|
||
|
"prints the name to the console or Idle shell.\n"
|
||
|
"Right clicking an item will display a popup."
|
||
|
}
|
||
|
|
||
|
show_idlehelp_spec = {
|
||
|
'file': 'help',
|
||
|
'kwds': {},
|
||
|
'msg': "If the help text displays, this works.\n"
|
||
|
"Text is selectable. Window is scrollable."
|
||
|
}
|
||
|
|
||
|
_stack_viewer_spec = {
|
||
|
'file': 'stackviewer',
|
||
|
'kwds': {},
|
||
|
'msg': "A stacktrace for a NameError exception.\n"
|
||
|
"Expand 'idlelib ...' and '<locals>'.\n"
|
||
|
"Check that exc_value, exc_tb, and exc_type are correct.\n"
|
||
|
}
|
||
|
|
||
|
_tooltip_spec = {
|
||
|
'file': 'tooltip',
|
||
|
'kwds': {},
|
||
|
'msg': "Place mouse cursor over both the buttons\n"
|
||
|
"A tooltip should appear with some text."
|
||
|
}
|
||
|
|
||
|
_tree_widget_spec = {
|
||
|
'file': 'tree',
|
||
|
'kwds': {},
|
||
|
'msg': "The canvas is scrollable.\n"
|
||
|
"Click on folders upto to the lowest level."
|
||
|
}
|
||
|
|
||
|
_undo_delegator_spec = {
|
||
|
'file': 'undo',
|
||
|
'kwds': {},
|
||
|
'msg': "Click [Undo] to undo any action.\n"
|
||
|
"Click [Redo] to redo any action.\n"
|
||
|
"Click [Dump] to dump the current state "
|
||
|
"by printing to the console or the IDLE shell.\n"
|
||
|
}
|
||
|
|
||
|
ViewWindow_spec = {
|
||
|
'file': 'textview',
|
||
|
'kwds': {'title': 'Test textview',
|
||
|
'contents': 'The quick brown fox jumps over the lazy dog.\n'*35,
|
||
|
'_htest': True},
|
||
|
'msg': "Test for read-only property of text.\n"
|
||
|
"Select text, scroll window, close"
|
||
|
}
|
||
|
|
||
|
_widget_redirector_spec = {
|
||
|
'file': 'redirector',
|
||
|
'kwds': {},
|
||
|
'msg': "Every text insert should be printed to the console "
|
||
|
"or the IDLE shell."
|
||
|
}
|
||
|
|
||
|
def run(*tests):
|
||
|
root = tk.Tk()
|
||
|
root.title('IDLE htest')
|
||
|
root.resizable(0, 0)
|
||
|
|
||
|
# a scrollable Label like constant width text widget.
|
||
|
frameLabel = tk.Frame(root, padx=10)
|
||
|
frameLabel.pack()
|
||
|
text = tk.Text(frameLabel, wrap='word')
|
||
|
text.configure(bg=root.cget('bg'), relief='flat', height=4, width=70)
|
||
|
scrollbar = Scrollbar(frameLabel, command=text.yview)
|
||
|
text.config(yscrollcommand=scrollbar.set)
|
||
|
scrollbar.pack(side='right', fill='y', expand=False)
|
||
|
text.pack(side='left', fill='both', expand=True)
|
||
|
|
||
|
test_list = [] # List of tuples of the form (spec, callable widget)
|
||
|
if tests:
|
||
|
for test in tests:
|
||
|
test_spec = globals()[test.__name__ + '_spec']
|
||
|
test_spec['name'] = test.__name__
|
||
|
test_list.append((test_spec, test))
|
||
|
else:
|
||
|
for k, d in globals().items():
|
||
|
if k.endswith('_spec'):
|
||
|
test_name = k[:-5]
|
||
|
test_spec = d
|
||
|
test_spec['name'] = test_name
|
||
|
mod = import_module('idlelib.' + test_spec['file'])
|
||
|
test = getattr(mod, test_name)
|
||
|
test_list.append((test_spec, test))
|
||
|
|
||
|
test_name = tk.StringVar(root)
|
||
|
callable_object = None
|
||
|
test_kwds = None
|
||
|
|
||
|
def next_test():
|
||
|
|
||
|
nonlocal test_name, callable_object, test_kwds
|
||
|
if len(test_list) == 1:
|
||
|
next_button.pack_forget()
|
||
|
test_spec, callable_object = test_list.pop()
|
||
|
test_kwds = test_spec['kwds']
|
||
|
test_kwds['parent'] = root
|
||
|
test_name.set('Test ' + test_spec['name'])
|
||
|
|
||
|
text.configure(state='normal') # enable text editing
|
||
|
text.delete('1.0','end')
|
||
|
text.insert("1.0",test_spec['msg'])
|
||
|
text.configure(state='disabled') # preserve read-only property
|
||
|
|
||
|
def run_test(_=None):
|
||
|
widget = callable_object(**test_kwds)
|
||
|
try:
|
||
|
print(widget.result)
|
||
|
except AttributeError:
|
||
|
pass
|
||
|
|
||
|
def close(_=None):
|
||
|
root.destroy()
|
||
|
|
||
|
button = tk.Button(root, textvariable=test_name,
|
||
|
default='active', command=run_test)
|
||
|
next_button = tk.Button(root, text="Next", command=next_test)
|
||
|
button.pack()
|
||
|
next_button.pack()
|
||
|
next_button.focus_set()
|
||
|
root.bind('<Key-Return>', run_test)
|
||
|
root.bind('<Key-Escape>', close)
|
||
|
|
||
|
next_test()
|
||
|
root.mainloop()
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
run()
|