|
"Test format, coverage 99%." |
|
|
|
from idlelib import format as ft |
|
import unittest |
|
from unittest import mock |
|
from test.support import requires |
|
from tkinter import Tk, Text |
|
from idlelib.editor import EditorWindow |
|
from idlelib.idle_test.mock_idle import Editor as MockEditor |
|
|
|
|
|
class Is_Get_Test(unittest.TestCase): |
|
"""Test the is_ and get_ functions""" |
|
test_comment = '# This is a comment' |
|
test_nocomment = 'This is not a comment' |
|
trailingws_comment = '# This is a comment ' |
|
leadingws_comment = ' # This is a comment' |
|
leadingws_nocomment = ' This is not a comment' |
|
|
|
def test_is_all_white(self): |
|
self.assertTrue(ft.is_all_white('')) |
|
self.assertTrue(ft.is_all_white('\t\n\r\f\v')) |
|
self.assertFalse(ft.is_all_white(self.test_comment)) |
|
|
|
def test_get_indent(self): |
|
Equal = self.assertEqual |
|
Equal(ft.get_indent(self.test_comment), '') |
|
Equal(ft.get_indent(self.trailingws_comment), '') |
|
Equal(ft.get_indent(self.leadingws_comment), ' ') |
|
Equal(ft.get_indent(self.leadingws_nocomment), ' ') |
|
|
|
def test_get_comment_header(self): |
|
Equal = self.assertEqual |
|
|
|
Equal(ft.get_comment_header(self.test_comment), '#') |
|
Equal(ft.get_comment_header(self.trailingws_comment), '#') |
|
Equal(ft.get_comment_header(self.leadingws_comment), ' #') |
|
|
|
Equal(ft.get_comment_header(self.leadingws_nocomment), ' ') |
|
Equal(ft.get_comment_header(self.test_nocomment), '') |
|
|
|
|
|
class FindTest(unittest.TestCase): |
|
"""Test the find_paragraph function in paragraph module. |
|
|
|
Using the runcase() function, find_paragraph() is called with 'mark' set at |
|
multiple indexes before and inside the test paragraph. |
|
|
|
It appears that code with the same indentation as a quoted string is grouped |
|
as part of the same paragraph, which is probably incorrect behavior. |
|
""" |
|
|
|
@classmethod |
|
def setUpClass(cls): |
|
from idlelib.idle_test.mock_tk import Text |
|
cls.text = Text() |
|
|
|
def runcase(self, inserttext, stopline, expected): |
|
|
|
|
|
|
|
text = self.text |
|
text.insert('1.0', inserttext) |
|
for line in range(1, stopline): |
|
linelength = int(text.index("%d.end" % line).split('.')[1]) |
|
for col in (0, linelength//2, linelength): |
|
tempindex = "%d.%d" % (line, col) |
|
self.assertEqual(ft.find_paragraph(text, tempindex), expected) |
|
text.delete('1.0', 'end') |
|
|
|
def test_find_comment(self): |
|
comment = ( |
|
"# Comment block with no blank lines before\n" |
|
"# Comment line\n" |
|
"\n") |
|
self.runcase(comment, 3, ('1.0', '3.0', '#', comment[0:58])) |
|
|
|
comment = ( |
|
"\n" |
|
"# Comment block with whitespace line before and after\n" |
|
"# Comment line\n" |
|
"\n") |
|
self.runcase(comment, 4, ('2.0', '4.0', '#', comment[1:70])) |
|
|
|
comment = ( |
|
"\n" |
|
" # Indented comment block with whitespace before and after\n" |
|
" # Comment line\n" |
|
"\n") |
|
self.runcase(comment, 4, ('2.0', '4.0', ' #', comment[1:82])) |
|
|
|
comment = ( |
|
"\n" |
|
"# Single line comment\n" |
|
"\n") |
|
self.runcase(comment, 3, ('2.0', '3.0', '#', comment[1:23])) |
|
|
|
comment = ( |
|
"\n" |
|
" # Single line comment with leading whitespace\n" |
|
"\n") |
|
self.runcase(comment, 3, ('2.0', '3.0', ' #', comment[1:51])) |
|
|
|
comment = ( |
|
"\n" |
|
"# Comment immediately followed by code\n" |
|
"x = 42\n" |
|
"\n") |
|
self.runcase(comment, 3, ('2.0', '3.0', '#', comment[1:40])) |
|
|
|
comment = ( |
|
"\n" |
|
" # Indented comment immediately followed by code\n" |
|
"x = 42\n" |
|
"\n") |
|
self.runcase(comment, 3, ('2.0', '3.0', ' #', comment[1:53])) |
|
|
|
comment = ( |
|
"\n" |
|
"# Comment immediately followed by indented code\n" |
|
" x = 42\n" |
|
"\n") |
|
self.runcase(comment, 3, ('2.0', '3.0', '#', comment[1:49])) |
|
|
|
def test_find_paragraph(self): |
|
teststring = ( |
|
'"""String with no blank lines before\n' |
|
'String line\n' |
|
'"""\n' |
|
'\n') |
|
self.runcase(teststring, 4, ('1.0', '4.0', '', teststring[0:53])) |
|
|
|
teststring = ( |
|
"\n" |
|
'"""String with whitespace line before and after\n' |
|
'String line.\n' |
|
'"""\n' |
|
'\n') |
|
self.runcase(teststring, 5, ('2.0', '5.0', '', teststring[1:66])) |
|
|
|
teststring = ( |
|
'\n' |
|
' """Indented string with whitespace before and after\n' |
|
' Comment string.\n' |
|
' """\n' |
|
'\n') |
|
self.runcase(teststring, 5, ('2.0', '5.0', ' ', teststring[1:85])) |
|
|
|
teststring = ( |
|
'\n' |
|
'"""Single line string."""\n' |
|
'\n') |
|
self.runcase(teststring, 3, ('2.0', '3.0', '', teststring[1:27])) |
|
|
|
teststring = ( |
|
'\n' |
|
' """Single line string with leading whitespace."""\n' |
|
'\n') |
|
self.runcase(teststring, 3, ('2.0', '3.0', ' ', teststring[1:55])) |
|
|
|
|
|
class ReformatFunctionTest(unittest.TestCase): |
|
"""Test the reformat_paragraph function without the editor window.""" |
|
|
|
def test_reformat_paragraph(self): |
|
Equal = self.assertEqual |
|
reform = ft.reformat_paragraph |
|
hw = "O hello world" |
|
Equal(reform(' ', 1), ' ') |
|
Equal(reform("Hello world", 20), "Hello world") |
|
|
|
|
|
Equal(reform(hw, 1), "O\nhello\nworld") |
|
Equal(reform(hw, 6), "O\nhello\nworld") |
|
Equal(reform(hw, 7), "O hello\nworld") |
|
Equal(reform(hw, 12), "O hello\nworld") |
|
Equal(reform(hw, 13), "O hello world") |
|
|
|
|
|
hw = "\nO hello world" |
|
Equal(reform(hw, 1), "\nO\nhello\nworld") |
|
Equal(reform(hw, 6), "\nO\nhello\nworld") |
|
Equal(reform(hw, 7), "\nO hello\nworld") |
|
Equal(reform(hw, 12), "\nO hello\nworld") |
|
Equal(reform(hw, 13), "\nO hello world") |
|
|
|
|
|
class ReformatCommentTest(unittest.TestCase): |
|
"""Test the reformat_comment function without the editor window.""" |
|
|
|
def test_reformat_comment(self): |
|
Equal = self.assertEqual |
|
|
|
|
|
test_string = ( |
|
" \"\"\"this is a test of a reformat for a triple quoted string" |
|
" will it reformat to less than 70 characters for me?\"\"\"") |
|
result = ft.reformat_comment(test_string, 70, " ") |
|
expected = ( |
|
" \"\"\"this is a test of a reformat for a triple quoted string will it\n" |
|
" reformat to less than 70 characters for me?\"\"\"") |
|
Equal(result, expected) |
|
|
|
test_comment = ( |
|
"# this is a test of a reformat for a triple quoted string will " |
|
"it reformat to less than 70 characters for me?") |
|
result = ft.reformat_comment(test_comment, 70, "#") |
|
expected = ( |
|
"# this is a test of a reformat for a triple quoted string will it\n" |
|
"# reformat to less than 70 characters for me?") |
|
Equal(result, expected) |
|
|
|
|
|
class FormatClassTest(unittest.TestCase): |
|
def test_init_close(self): |
|
instance = ft.FormatParagraph('editor') |
|
self.assertEqual(instance.editwin, 'editor') |
|
instance.close() |
|
self.assertEqual(instance.editwin, None) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TextWrapper: |
|
def __init__(self, master): |
|
self.text = Text(master=master) |
|
def __getattr__(self, name): |
|
return getattr(self.text, name) |
|
def undo_block_start(self): pass |
|
def undo_block_stop(self): pass |
|
|
|
class Editor: |
|
def __init__(self, root): |
|
self.text = TextWrapper(root) |
|
get_selection_indices = EditorWindow. get_selection_indices |
|
|
|
class FormatEventTest(unittest.TestCase): |
|
"""Test the formatting of text inside a Text widget. |
|
|
|
This is done with FormatParagraph.format.paragraph_event, |
|
which calls functions in the module as appropriate. |
|
""" |
|
test_string = ( |
|
" '''this is a test of a reformat for a triple " |
|
"quoted string will it reformat to less than 70 " |
|
"characters for me?'''\n") |
|
multiline_test_string = ( |
|
" '''The first line is under the max width.\n" |
|
" The second line's length is way over the max width. It goes " |
|
"on and on until it is over 100 characters long.\n" |
|
" Same thing with the third line. It is also way over the max " |
|
"width, but FormatParagraph will fix it.\n" |
|
" '''\n") |
|
multiline_test_comment = ( |
|
"# The first line is under the max width.\n" |
|
"# The second line's length is way over the max width. It goes on " |
|
"and on until it is over 100 characters long.\n" |
|
"# Same thing with the third line. It is also way over the max " |
|
"width, but FormatParagraph will fix it.\n" |
|
"# The fourth line is short like the first line.") |
|
|
|
@classmethod |
|
def setUpClass(cls): |
|
requires('gui') |
|
cls.root = Tk() |
|
cls.root.withdraw() |
|
editor = Editor(root=cls.root) |
|
cls.text = editor.text.text |
|
cls.formatter = ft.FormatParagraph(editor).format_paragraph_event |
|
|
|
|
|
@classmethod |
|
def tearDownClass(cls): |
|
del cls.text, cls.formatter |
|
cls.root.update_idletasks() |
|
cls.root.destroy() |
|
del cls.root |
|
|
|
def test_short_line(self): |
|
self.text.insert('1.0', "Short line\n") |
|
self.formatter("Dummy") |
|
self.assertEqual(self.text.get('1.0', 'insert'), "Short line\n" ) |
|
self.text.delete('1.0', 'end') |
|
|
|
def test_long_line(self): |
|
text = self.text |
|
|
|
|
|
text.insert('1.0', self.test_string) |
|
text.mark_set('insert', '1.0') |
|
self.formatter('ParameterDoesNothing', limit=70) |
|
result = text.get('1.0', 'insert') |
|
|
|
expected = ( |
|
" '''this is a test of a reformat for a triple quoted string will it\n" |
|
" reformat to less than 70 characters for me?'''\n") |
|
self.assertEqual(result, expected) |
|
text.delete('1.0', 'end') |
|
|
|
|
|
text.insert('1.0', self.test_string) |
|
text.tag_add('sel', '1.11', '1.end') |
|
self.formatter('ParameterDoesNothing', limit=70) |
|
result = text.get('1.0', 'insert') |
|
|
|
expected = ( |
|
" '''this is a test of a reformat for a triple quoted string will it reformat\n" |
|
" to less than 70 characters for me?'''") |
|
self.assertEqual(result, expected) |
|
text.delete('1.0', 'end') |
|
|
|
def test_multiple_lines(self): |
|
text = self.text |
|
|
|
text.insert('1.0', self.multiline_test_string) |
|
text.tag_add('sel', '2.0', '4.0') |
|
self.formatter('ParameterDoesNothing', limit=70) |
|
result = text.get('2.0', 'insert') |
|
expected = ( |
|
" The second line's length is way over the max width. It goes on and\n" |
|
" on until it is over 100 characters long. Same thing with the third\n" |
|
" line. It is also way over the max width, but FormatParagraph will\n" |
|
" fix it.\n") |
|
self.assertEqual(result, expected) |
|
text.delete('1.0', 'end') |
|
|
|
def test_comment_block(self): |
|
text = self.text |
|
|
|
|
|
text.insert('1.0', self.multiline_test_comment) |
|
self.formatter('ParameterDoesNothing', limit=70) |
|
result = text.get('1.0', 'insert') |
|
expected = ( |
|
"# The first line is under the max width. The second line's length is\n" |
|
"# way over the max width. It goes on and on until it is over 100\n" |
|
"# characters long. Same thing with the third line. It is also way over\n" |
|
"# the max width, but FormatParagraph will fix it. The fourth line is\n" |
|
"# short like the first line.\n") |
|
self.assertEqual(result, expected) |
|
text.delete('1.0', 'end') |
|
|
|
|
|
text.insert('1.0', self.multiline_test_comment) |
|
text.tag_add('sel', '2.0', '3.0') |
|
self.formatter('ParameterDoesNothing', limit=70) |
|
result = text.get('1.0', 'insert') |
|
expected = ( |
|
"# The first line is under the max width.\n" |
|
"# The second line's length is way over the max width. It goes on and\n" |
|
"# on until it is over 100 characters long.\n") |
|
self.assertEqual(result, expected) |
|
text.delete('1.0', 'end') |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class DummyEditwin: |
|
def __init__(self, root, text): |
|
self.root = root |
|
self.text = text |
|
self.indentwidth = 4 |
|
self.tabwidth = 4 |
|
self.usetabs = False |
|
self.context_use_ps1 = True |
|
|
|
_make_blanks = EditorWindow._make_blanks |
|
get_selection_indices = EditorWindow.get_selection_indices |
|
|
|
|
|
class FormatRegionTest(unittest.TestCase): |
|
|
|
@classmethod |
|
def setUpClass(cls): |
|
requires('gui') |
|
cls.root = Tk() |
|
cls.root.withdraw() |
|
cls.text = Text(cls.root) |
|
cls.text.undo_block_start = mock.Mock() |
|
cls.text.undo_block_stop = mock.Mock() |
|
cls.editor = DummyEditwin(cls.root, cls.text) |
|
cls.formatter = ft.FormatRegion(cls.editor) |
|
|
|
@classmethod |
|
def tearDownClass(cls): |
|
del cls.text, cls.formatter, cls.editor |
|
cls.root.update_idletasks() |
|
cls.root.destroy() |
|
del cls.root |
|
|
|
def setUp(self): |
|
self.text.insert('1.0', self.code_sample) |
|
|
|
def tearDown(self): |
|
self.text.delete('1.0', 'end') |
|
|
|
code_sample = """\ |
|
# WS line needed for test. |
|
class C1: |
|
# Class comment. |
|
def __init__(self, a, b): |
|
self.a = a |
|
self.b = b |
|
|
|
def compare(self): |
|
if a > b: |
|
return a |
|
elif a < b: |
|
return b |
|
else: |
|
return None |
|
""" |
|
|
|
def test_get_region(self): |
|
get = self.formatter.get_region |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
|
|
text.tag_add('sel', '7.0', '10.0') |
|
expected_lines = ['', |
|
' def compare(self):', |
|
' if a > b:', |
|
''] |
|
eq(get(), ('7.0', '10.0', '\n'.join(expected_lines), expected_lines)) |
|
|
|
|
|
text.tag_remove('sel', '1.0', 'end') |
|
eq(get(), ('15.0', '16.0', '\n', ['', ''])) |
|
|
|
def test_set_region(self): |
|
set_ = self.formatter.set_region |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
save_bell = text.bell |
|
text.bell = mock.Mock() |
|
line6 = self.code_sample.splitlines()[5] |
|
line10 = self.code_sample.splitlines()[9] |
|
|
|
text.tag_add('sel', '6.0', '11.0') |
|
head, tail, chars, lines = self.formatter.get_region() |
|
|
|
|
|
set_(head, tail, chars, lines) |
|
text.bell.assert_called_once() |
|
eq(text.get('6.0', '11.0'), chars) |
|
eq(text.get('sel.first', 'sel.last'), chars) |
|
text.tag_remove('sel', '1.0', 'end') |
|
|
|
|
|
newstring = 'added line 1\n\n\n\n' |
|
newlines = newstring.split('\n') |
|
set_('7.0', '10.0', chars, newlines) |
|
|
|
eq(text.get('sel.first', 'sel.last'), newstring) |
|
|
|
eq(text.get('7.0', '11.0'), newstring) |
|
|
|
eq(text.get('6.0', '7.0-1c'), line6) |
|
eq(text.get('11.0', '12.0-1c'), line10) |
|
text.tag_remove('sel', '1.0', 'end') |
|
|
|
text.bell = save_bell |
|
|
|
def test_indent_region_event(self): |
|
indent = self.formatter.indent_region_event |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
text.tag_add('sel', '7.0', '10.0') |
|
indent() |
|
|
|
eq(text.get('7.0', '10.0'), ('\n def compare(self):\n if a > b:\n')) |
|
|
|
def test_dedent_region_event(self): |
|
dedent = self.formatter.dedent_region_event |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
text.tag_add('sel', '7.0', '10.0') |
|
dedent() |
|
|
|
eq(text.get('7.0', '10.0'), ('\ndef compare(self):\n if a > b:\n')) |
|
|
|
def test_comment_region_event(self): |
|
comment = self.formatter.comment_region_event |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
text.tag_add('sel', '7.0', '10.0') |
|
comment() |
|
eq(text.get('7.0', '10.0'), ('##\n## def compare(self):\n## if a > b:\n')) |
|
|
|
def test_uncomment_region_event(self): |
|
comment = self.formatter.comment_region_event |
|
uncomment = self.formatter.uncomment_region_event |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
text.tag_add('sel', '7.0', '10.0') |
|
comment() |
|
uncomment() |
|
eq(text.get('7.0', '10.0'), ('\n def compare(self):\n if a > b:\n')) |
|
|
|
|
|
text.tag_remove('sel', '1.0', 'end') |
|
text.tag_add('sel', '3.0', '4.0') |
|
uncomment() |
|
eq(text.get('3.0', '3.end'), (' # Class comment.')) |
|
|
|
self.formatter.set_region('3.0', '4.0', '', ['# Class comment.', '']) |
|
uncomment() |
|
eq(text.get('3.0', '3.end'), (' Class comment.')) |
|
|
|
@mock.patch.object(ft.FormatRegion, "_asktabwidth") |
|
def test_tabify_region_event(self, _asktabwidth): |
|
tabify = self.formatter.tabify_region_event |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
text.tag_add('sel', '7.0', '10.0') |
|
|
|
_asktabwidth.return_value = None |
|
self.assertIsNone(tabify()) |
|
|
|
_asktabwidth.return_value = 3 |
|
self.assertIsNotNone(tabify()) |
|
eq(text.get('7.0', '10.0'), ('\n\t def compare(self):\n\t\t if a > b:\n')) |
|
|
|
@mock.patch.object(ft.FormatRegion, "_asktabwidth") |
|
def test_untabify_region_event(self, _asktabwidth): |
|
untabify = self.formatter.untabify_region_event |
|
text = self.text |
|
eq = self.assertEqual |
|
|
|
text.tag_add('sel', '7.0', '10.0') |
|
|
|
_asktabwidth.return_value = None |
|
self.assertIsNone(untabify()) |
|
|
|
_asktabwidth.return_value = 2 |
|
self.formatter.tabify_region_event() |
|
_asktabwidth.return_value = 3 |
|
self.assertIsNotNone(untabify()) |
|
eq(text.get('7.0', '10.0'), ('\n def compare(self):\n if a > b:\n')) |
|
|
|
@mock.patch.object(ft, "askinteger") |
|
def test_ask_tabwidth(self, askinteger): |
|
ask = self.formatter._asktabwidth |
|
askinteger.return_value = 10 |
|
self.assertEqual(ask(), 10) |
|
|
|
|
|
class IndentsTest(unittest.TestCase): |
|
|
|
@mock.patch.object(ft, "askyesno") |
|
def test_toggle_tabs(self, askyesno): |
|
editor = DummyEditwin(None, None) |
|
indents = ft.Indents(editor) |
|
askyesno.return_value = True |
|
|
|
indents.toggle_tabs_event(None) |
|
self.assertEqual(editor.usetabs, True) |
|
self.assertEqual(editor.indentwidth, 8) |
|
|
|
indents.toggle_tabs_event(None) |
|
self.assertEqual(editor.usetabs, False) |
|
self.assertEqual(editor.indentwidth, 8) |
|
|
|
@mock.patch.object(ft, "askinteger") |
|
def test_change_indentwidth(self, askinteger): |
|
editor = DummyEditwin(None, None) |
|
indents = ft.Indents(editor) |
|
|
|
askinteger.return_value = None |
|
indents.change_indentwidth_event(None) |
|
self.assertEqual(editor.indentwidth, 4) |
|
|
|
askinteger.return_value = 3 |
|
indents.change_indentwidth_event(None) |
|
self.assertEqual(editor.indentwidth, 3) |
|
|
|
askinteger.return_value = 5 |
|
editor.usetabs = True |
|
indents.change_indentwidth_event(None) |
|
self.assertEqual(editor.indentwidth, 3) |
|
|
|
|
|
class RstripTest(unittest.TestCase): |
|
|
|
@classmethod |
|
def setUpClass(cls): |
|
requires('gui') |
|
cls.root = Tk() |
|
cls.root.withdraw() |
|
cls.text = Text(cls.root) |
|
cls.editor = MockEditor(text=cls.text) |
|
cls.do_rstrip = ft.Rstrip(cls.editor).do_rstrip |
|
|
|
@classmethod |
|
def tearDownClass(cls): |
|
del cls.text, cls.do_rstrip, cls.editor |
|
cls.root.update_idletasks() |
|
cls.root.destroy() |
|
del cls.root |
|
|
|
def tearDown(self): |
|
self.text.delete('1.0', 'end-1c') |
|
|
|
def test_rstrip_lines(self): |
|
original = ( |
|
"Line with an ending tab \n" |
|
"Line ending in 5 spaces \n" |
|
"Linewithnospaces\n" |
|
" indented line\n" |
|
" indented line with trailing space \n" |
|
" \n") |
|
stripped = ( |
|
"Line with an ending tab\n" |
|
"Line ending in 5 spaces\n" |
|
"Linewithnospaces\n" |
|
" indented line\n" |
|
" indented line with trailing space\n") |
|
|
|
self.text.insert('1.0', original) |
|
self.do_rstrip() |
|
self.assertEqual(self.text.get('1.0', 'insert'), stripped) |
|
|
|
def test_rstrip_end(self): |
|
text = self.text |
|
for code in ('', '\n', '\n\n\n'): |
|
with self.subTest(code=code): |
|
text.insert('1.0', code) |
|
self.do_rstrip() |
|
self.assertEqual(text.get('1.0','end-1c'), '') |
|
for code in ('a\n', 'a\n\n', 'a\n\n\n'): |
|
with self.subTest(code=code): |
|
text.delete('1.0', 'end-1c') |
|
text.insert('1.0', code) |
|
self.do_rstrip() |
|
self.assertEqual(text.get('1.0','end-1c'), 'a\n') |
|
|
|
|
|
if __name__ == '__main__': |
|
unittest.main(verbosity=2, exit=2) |
|
|