qutebrowser/tests/unit/misc/test_editor.py

215 lines
7.0 KiB
Python
Raw Normal View History

2014-06-19 09:04:37 +02:00
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
2017-05-09 21:37:03 +02:00
# Copyright 2014-2017 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
2014-05-27 07:43:29 +02:00
#
# This file is part of qutebrowser.
#
# qutebrowser 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.
#
# qutebrowser 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 qutebrowser. If not, see <http://www.gnu.org/licenses/>.
"""Tests for qutebrowser.misc.editor."""
2014-05-27 07:43:29 +02:00
import os
import os.path
import logging
2014-05-27 07:43:29 +02:00
from PyQt5.QtCore import QProcess
2015-04-04 18:24:26 +02:00
import pytest
2014-05-27 07:43:29 +02:00
2015-08-19 09:09:09 +02:00
from qutebrowser.misc import editor as editormod
from qutebrowser.utils import usertypes
2015-08-19 09:09:09 +02:00
@pytest.fixture(autouse=True)
2016-09-15 11:56:46 +02:00
def patch_things(config_stub, monkeypatch, stubs):
monkeypatch.setattr(editormod.guiprocess, 'QProcess',
2015-08-19 09:09:09 +02:00
stubs.fake_qprocess())
@pytest.fixture
def editor(caplog, qtbot):
2016-09-14 20:52:32 +02:00
ed = editormod.ExternalEditor()
2015-08-19 09:09:09 +02:00
yield ed
with caplog.at_level(logging.ERROR):
2017-10-03 18:54:40 +02:00
ed._remove_file = True
ed._cleanup()
2014-05-27 07:43:29 +02:00
2015-04-04 18:24:26 +02:00
class TestArg:
2014-05-27 11:17:27 +02:00
2014-05-27 13:06:13 +02:00
"""Test argument handling.
Attributes:
editor: The ExternalEditor instance to test.
"""
2014-05-27 11:17:27 +02:00
2015-08-19 09:09:09 +02:00
def test_placeholder(self, config_stub, editor):
2014-05-27 11:17:27 +02:00
"""Test starting editor with placeholder argument."""
config_stub.val.editor.command = ['bin', 'foo', '{}', 'bar']
2015-08-19 09:09:09 +02:00
editor.edit("")
editor._proc._proc.start.assert_called_with(
"bin", ["foo", editor._filename, "bar"])
2014-05-27 11:30:57 +02:00
2016-02-02 06:53:12 +01:00
def test_placeholder_inline(self, config_stub, editor):
"""Test starting editor with placeholder arg inside of another arg."""
config_stub.val.editor.command = ['bin', 'foo{}', 'bar']
2016-02-02 06:53:12 +01:00
editor.edit("")
editor._proc._proc.start.assert_called_with(
"bin", ["foo" + editor._filename, "bar"])
2016-02-02 06:53:12 +01:00
2014-05-27 07:43:29 +02:00
class TestFileHandling:
2015-04-05 20:30:31 +02:00
2015-08-19 09:09:09 +02:00
"""Test creation/deletion of tempfile."""
2014-05-27 13:06:13 +02:00
2015-08-19 09:09:09 +02:00
def test_ok(self, editor):
2015-03-31 20:49:29 +02:00
"""Test file handling when closing with an exit status == 0."""
2015-08-19 09:09:09 +02:00
editor.edit("")
filename = editor._filename
2015-04-04 18:24:26 +02:00
assert os.path.exists(filename)
2015-08-19 09:34:44 +02:00
assert os.path.basename(filename).startswith('qutebrowser-editor-')
2015-08-19 09:09:09 +02:00
editor._proc.finished.emit(0, QProcess.NormalExit)
2015-04-04 18:24:26 +02:00
assert not os.path.exists(filename)
2014-05-27 07:43:29 +02:00
def test_existing_file(self, editor, tmpdir):
"""Test editing an existing file."""
path = tmpdir / 'foo.txt'
path.ensure()
editor.edit_file(str(path))
editor._proc.finished.emit(0, QProcess.NormalExit)
assert path.exists()
def test_error(self, editor):
2015-03-31 20:49:29 +02:00
"""Test file handling when closing with an exit status != 0."""
2015-08-19 09:09:09 +02:00
editor.edit("")
filename = editor._filename
2015-04-04 18:24:26 +02:00
assert os.path.exists(filename)
2017-10-03 19:28:41 +02:00
editor._proc._proc.exitStatus = lambda: QProcess.CrashExit
editor._proc.finished.emit(1, QProcess.NormalExit)
assert os.path.exists(filename)
os.remove(filename)
2014-05-27 07:43:29 +02:00
def test_crash(self, editor):
2014-05-27 07:43:29 +02:00
"""Test file handling when closing with a crash."""
2015-08-19 09:09:09 +02:00
editor.edit("")
filename = editor._filename
2015-04-04 18:24:26 +02:00
assert os.path.exists(filename)
2017-10-03 19:28:41 +02:00
editor._proc._proc.exitStatus = lambda: QProcess.CrashExit
editor._proc.error.emit(QProcess.Crashed)
2015-08-19 09:09:09 +02:00
editor._proc.finished.emit(0, QProcess.CrashExit)
assert os.path.exists(filename)
os.remove(filename)
2014-05-27 07:43:29 +02:00
def test_unreadable(self, message_mock, editor, caplog, qtbot):
2015-08-19 09:34:44 +02:00
"""Test file handling when closing with an unreadable file."""
editor.edit("")
filename = editor._filename
2015-08-19 09:34:44 +02:00
assert os.path.exists(filename)
os.chmod(filename, 0o277)
if os.access(filename, os.R_OK):
# Docker container or similar
pytest.skip("File was still readable")
with caplog.at_level(logging.ERROR):
editor._proc.finished.emit(0, QProcess.NormalExit)
2015-08-19 09:34:44 +02:00
assert not os.path.exists(filename)
2016-09-15 11:56:46 +02:00
msg = message_mock.getmsg(usertypes.MessageLevel.error)
2015-08-19 09:34:44 +02:00
assert msg.text.startswith("Failed to read back edited file: ")
def test_unwritable(self, monkeypatch, message_mock, editor, tmpdir,
caplog):
2015-08-19 09:34:44 +02:00
"""Test file handling when the initial file is not writable."""
tmpdir.chmod(0)
if os.access(str(tmpdir), os.W_OK):
# Docker container or similar
pytest.skip("File was still writable")
monkeypatch.setattr(editormod.tempfile, 'tempdir', str(tmpdir))
with caplog.at_level(logging.ERROR):
editor.edit("")
2016-09-15 11:56:46 +02:00
msg = message_mock.getmsg(usertypes.MessageLevel.error)
2015-08-19 09:34:44 +02:00
assert msg.text.startswith("Failed to create initial file: ")
assert editor._proc is None
def test_double_edit(self, editor):
editor.edit("")
with pytest.raises(ValueError):
editor.edit("")
2015-08-19 09:09:09 +02:00
@pytest.mark.parametrize('initial_text, edited_text', [
('', 'Hello'),
('Hello', 'World'),
('Hällö Wörld', 'Überprüfung'),
('\u2603', '\u2601') # Unicode snowman -> cloud
])
2017-10-03 19:28:41 +02:00
def test_modify(qtbot, editor, initial_text, edited_text):
2015-08-19 09:09:09 +02:00
"""Test if inputs get modified correctly."""
editor.edit(initial_text)
2015-04-05 20:30:31 +02:00
with open(editor._filename, 'r', encoding='utf-8') as f:
2015-08-19 09:09:09 +02:00
assert f.read() == initial_text
2014-05-27 13:06:13 +02:00
with qtbot.wait_signal(editor.file_updated) as blocker:
with open(editor._filename, 'w', encoding='utf-8') as f:
f.write(edited_text)
with qtbot.assert_not_emitted(editor.file_updated):
editor._proc.finished.emit(0, QProcess.NormalExit)
2017-10-03 19:28:41 +02:00
assert blocker.args == [edited_text]
def test_modify_multiple(qtbot, editor):
"""Test that multiple saves all trigger file_updated."""
editor.edit('foo')
with qtbot.wait_signal(editor.file_updated) as blocker:
with open(editor._filename, 'w', encoding='utf-8') as f:
f.write('bar')
assert blocker.args == ['bar']
with qtbot.wait_signal(editor.file_updated) as blocker:
with open(editor._filename, 'w', encoding='utf-8') as f:
f.write('baz')
assert blocker.args == ['baz']
with qtbot.assert_not_emitted(editor.file_updated):
editor._proc.finished.emit(0, QProcess.NormalExit)
@pytest.mark.parametrize('text, caret_position, result', [
('', 0, (1, 1)),
('a', 0, (1, 1)),
('a\nb', 1, (1, 2)),
('a\nb', 2, (2, 1)),
('a\nb', 3, (2, 2)),
('a\nbb\nccc', 4, (2, 3)),
('a\nbb\nccc', 5, (3, 1)),
('a\nbb\nccc', 8, (3, 4)),
('', None, (1, 1)),
])
def test_calculation(editor, text, caret_position, result):
2017-10-25 21:18:53 +02:00
"""Test calculation for line and column given text and caret_position."""
assert editor._calc_line_and_column(text, caret_position) == result