2014-06-19 09:04:37 +02:00
|
|
|
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
|
|
|
|
|
2018-02-05 12:19:50 +01:00
|
|
|
# Copyright 2014-2018 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/>.
|
|
|
|
|
2015-01-24 14:31:58 +01:00
|
|
|
"""Tests for qutebrowser.misc.editor."""
|
2014-05-27 07:43:29 +02:00
|
|
|
|
2018-02-07 21:40:03 +01:00
|
|
|
import time
|
2014-05-27 07:43:29 +02:00
|
|
|
import os
|
|
|
|
import os.path
|
2016-09-15 13:05:53 +02:00
|
|
|
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
|
2016-09-15 13:05:53 +02:00
|
|
|
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):
|
2017-03-01 11:33:41 +01:00
|
|
|
monkeypatch.setattr(editormod.guiprocess, 'QProcess',
|
2015-08-19 09:09:09 +02:00
|
|
|
stubs.fake_qprocess())
|
|
|
|
|
|
|
|
|
2016-08-22 07:40:24 +02:00
|
|
|
@pytest.fixture
|
2017-12-29 22:08:59 +01:00
|
|
|
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
|
2016-09-15 13:05:53 +02:00
|
|
|
with caplog.at_level(logging.ERROR):
|
2017-10-03 18:54:40 +02:00
|
|
|
ed._remove_file = True
|
2016-09-15 13:05:53 +02:00
|
|
|
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."""
|
2017-07-03 18:42:41 +02:00
|
|
|
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(
|
2017-10-03 18:19:09 +02:00
|
|
|
"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."""
|
2017-07-03 18:42:41 +02:00
|
|
|
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(
|
2017-10-03 18:19:09 +02:00
|
|
|
"bin", ["foo" + editor._filename, "bar"])
|
2016-02-02 06:53:12 +01:00
|
|
|
|
2014-05-27 07:43:29 +02:00
|
|
|
|
2015-04-08 05:30:02 +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("")
|
2017-10-03 18:19:09 +02:00
|
|
|
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
|
|
|
|
2017-10-03 18:19:09 +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()
|
|
|
|
|
2015-09-11 08:32:16 +02:00
|
|
|
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("")
|
2017-10-03 18:19:09 +02:00
|
|
|
filename = editor._filename
|
2015-04-04 18:24:26 +02:00
|
|
|
assert os.path.exists(filename)
|
2015-09-11 08:32:16 +02:00
|
|
|
|
2017-10-03 19:28:41 +02:00
|
|
|
editor._proc._proc.exitStatus = lambda: QProcess.CrashExit
|
2015-09-11 08:32:16 +02:00
|
|
|
editor._proc.finished.emit(1, QProcess.NormalExit)
|
|
|
|
|
2015-10-01 14:15:50 +02:00
|
|
|
assert os.path.exists(filename)
|
|
|
|
|
|
|
|
os.remove(filename)
|
2014-05-27 07:43:29 +02:00
|
|
|
|
2015-09-11 08:32:16 +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("")
|
2017-10-03 18:19:09 +02:00
|
|
|
filename = editor._filename
|
2015-04-04 18:24:26 +02:00
|
|
|
assert os.path.exists(filename)
|
2015-10-01 14:15:50 +02:00
|
|
|
|
2017-10-03 19:28:41 +02:00
|
|
|
editor._proc._proc.exitStatus = lambda: QProcess.CrashExit
|
2015-09-11 08:32:16 +02:00
|
|
|
editor._proc.error.emit(QProcess.Crashed)
|
|
|
|
|
2015-08-19 09:09:09 +02:00
|
|
|
editor._proc.finished.emit(0, QProcess.CrashExit)
|
2015-10-01 14:15:50 +02:00
|
|
|
assert os.path.exists(filename)
|
|
|
|
|
|
|
|
os.remove(filename)
|
2014-05-27 07:43:29 +02:00
|
|
|
|
2017-12-29 22:08:59 +01: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("")
|
2017-10-03 18:19:09 +02:00
|
|
|
filename = editor._filename
|
2015-08-19 09:34:44 +02:00
|
|
|
assert os.path.exists(filename)
|
2017-12-29 22:08:59 +01:00
|
|
|
os.chmod(filename, 0o277)
|
2017-07-03 09:40:39 +02:00
|
|
|
if os.access(filename, os.R_OK):
|
|
|
|
# Docker container or similar
|
|
|
|
pytest.skip("File was still readable")
|
|
|
|
|
2016-09-15 13:05:53 +02:00
|
|
|
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: ")
|
|
|
|
|
2016-09-15 13:05:53 +02:00
|
|
|
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)
|
2017-07-03 09:40:39 +02:00
|
|
|
if os.access(str(tmpdir), os.W_OK):
|
|
|
|
# Docker container or similar
|
|
|
|
pytest.skip("File was still writable")
|
|
|
|
|
2017-03-01 11:33:41 +01:00
|
|
|
monkeypatch.setattr(editormod.tempfile, 'tempdir', str(tmpdir))
|
2016-09-15 13:05:53 +02:00
|
|
|
|
|
|
|
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("")
|
|
|
|
|
2018-03-12 13:34:50 +01:00
|
|
|
def test_backup(self, qtbot, message_mock):
|
|
|
|
editor = editormod.ExternalEditor(watch=True)
|
|
|
|
editor.edit('foo')
|
2018-03-13 12:31:48 +01:00
|
|
|
with qtbot.wait_signal(editor.file_updated):
|
2018-03-12 13:34:50 +01:00
|
|
|
_update_file(editor._filename, 'bar')
|
|
|
|
|
|
|
|
editor.backup()
|
|
|
|
|
|
|
|
msg = message_mock.getmsg(usertypes.MessageLevel.info)
|
|
|
|
prefix = 'Editor backup at '
|
|
|
|
assert msg.text.startswith(prefix)
|
|
|
|
fname = msg.text[len(prefix):]
|
|
|
|
|
2018-03-13 12:31:48 +01:00
|
|
|
with qtbot.wait_signal(editor.editing_finished):
|
2018-03-12 13:34:50 +01:00
|
|
|
editor._proc.finished.emit(0, QProcess.NormalExit)
|
|
|
|
|
2018-03-13 12:31:48 +01:00
|
|
|
with open(fname, 'r', encoding='utf-8') as f:
|
2018-03-12 13:34:50 +01:00
|
|
|
assert f.read() == 'bar'
|
|
|
|
|
2018-03-13 12:31:48 +01:00
|
|
|
def test_backup_no_content(self, qtbot, message_mock):
|
|
|
|
editor = editormod.ExternalEditor(watch=True)
|
|
|
|
editor.edit('foo')
|
|
|
|
editor.backup()
|
|
|
|
# content has not changed, so no backup should be created
|
|
|
|
assert not message_mock.messages
|
|
|
|
|
|
|
|
def test_backup(self, qtbot, message_mock, mocker):
|
|
|
|
editor = editormod.ExternalEditor(watch=True)
|
|
|
|
editor.edit('foo')
|
|
|
|
with qtbot.wait_signal(editor.file_updated):
|
|
|
|
_update_file(editor._filename, 'bar')
|
|
|
|
|
|
|
|
mocker.patch('tempfile.NamedTemporaryFile', side_effect=OSError)
|
|
|
|
editor.backup()
|
|
|
|
|
|
|
|
msg = message_mock.getmsg(usertypes.MessageLevel.error)
|
|
|
|
assert msg.text.startswith('Failed to create editor backup:')
|
|
|
|
|
2014-05-27 11:47:43 +02:00
|
|
|
|
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
|
|
|
|
2017-10-03 18:19:09 +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
|
|
|
|
2017-10-03 18:19:09 +02:00
|
|
|
with open(editor._filename, 'w', encoding='utf-8') as f:
|
2015-08-19 09:09:09 +02:00
|
|
|
f.write(edited_text)
|
2014-05-27 11:47:43 +02:00
|
|
|
|
2018-02-03 14:27:36 +01:00
|
|
|
with qtbot.wait_signal(editor.file_updated) as blocker:
|
2017-10-03 19:28:41 +02:00
|
|
|
editor._proc.finished.emit(0, QProcess.NormalExit)
|
|
|
|
|
|
|
|
assert blocker.args == [edited_text]
|
2017-10-18 20:33:14 +02:00
|
|
|
|
|
|
|
|
2018-02-07 21:40:03 +01:00
|
|
|
def _update_file(filename, contents):
|
|
|
|
"""Update the given file and make sure its mtime changed.
|
|
|
|
|
|
|
|
This might write the file multiple times, but different systems have
|
|
|
|
different mtime's, so we can't be sure how long to wait otherwise.
|
|
|
|
"""
|
|
|
|
old_mtime = new_mtime = os.stat(filename).st_mtime
|
|
|
|
while old_mtime == new_mtime:
|
|
|
|
time.sleep(0.1)
|
|
|
|
with open(filename, 'w', encoding='utf-8') as f:
|
|
|
|
f.write(contents)
|
|
|
|
new_mtime = os.stat(filename).st_mtime
|
|
|
|
|
|
|
|
|
2018-02-03 14:27:36 +01:00
|
|
|
def test_modify_watch(qtbot):
|
|
|
|
"""Test that saving triggers file_updated when watch=True."""
|
|
|
|
editor = editormod.ExternalEditor(watch=True)
|
2018-01-26 17:12:07 +01:00
|
|
|
editor.edit('foo')
|
|
|
|
|
2018-02-04 13:02:25 +01:00
|
|
|
with qtbot.wait_signal(editor.file_updated, timeout=3000) as blocker:
|
2018-02-07 21:40:03 +01:00
|
|
|
_update_file(editor._filename, 'bar')
|
2018-01-26 17:12:07 +01:00
|
|
|
assert blocker.args == ['bar']
|
|
|
|
|
|
|
|
with qtbot.wait_signal(editor.file_updated) as blocker:
|
2018-02-07 21:40:03 +01:00
|
|
|
_update_file(editor._filename, 'baz')
|
2018-01-26 17:12:07 +01:00
|
|
|
assert blocker.args == ['baz']
|
|
|
|
|
|
|
|
with qtbot.assert_not_emitted(editor.file_updated):
|
|
|
|
editor._proc.finished.emit(0, QProcess.NormalExit)
|
|
|
|
|
|
|
|
|
2018-02-08 10:18:08 +01:00
|
|
|
def test_failing_watch(qtbot, caplog, monkeypatch):
|
|
|
|
"""When watching failed, an error should be logged.
|
|
|
|
|
|
|
|
Also, updating should still work when closing the process.
|
|
|
|
"""
|
|
|
|
editor = editormod.ExternalEditor(watch=True)
|
|
|
|
monkeypatch.setattr(editor._watcher, 'addPath', lambda _path: False)
|
|
|
|
|
|
|
|
with caplog.at_level(logging.ERROR):
|
|
|
|
editor.edit('foo')
|
|
|
|
|
|
|
|
with qtbot.assert_not_emitted(editor.file_updated):
|
|
|
|
_update_file(editor._filename, 'bar')
|
|
|
|
|
|
|
|
with qtbot.wait_signal(editor.file_updated) as blocker:
|
|
|
|
editor._proc.finished.emit(0, QProcess.NormalExit)
|
|
|
|
assert blocker.args == ['bar']
|
|
|
|
|
|
|
|
message = 'Failed to watch path: {}'.format(editor._filename)
|
|
|
|
assert caplog.records[0].msg == message
|
|
|
|
|
|
|
|
|
|
|
|
def test_failing_unwatch(qtbot, caplog, monkeypatch):
|
|
|
|
"""When unwatching failed, an error should be logged."""
|
|
|
|
editor = editormod.ExternalEditor(watch=True)
|
|
|
|
monkeypatch.setattr(editor._watcher, 'addPath', lambda _path: True)
|
|
|
|
monkeypatch.setattr(editor._watcher, 'files', lambda: [editor._filename])
|
|
|
|
monkeypatch.setattr(editor._watcher, 'removePaths', lambda paths: paths)
|
|
|
|
|
|
|
|
editor.edit('foo')
|
|
|
|
|
|
|
|
with caplog.at_level(logging.ERROR):
|
|
|
|
editor._proc.finished.emit(0, QProcess.NormalExit)
|
|
|
|
|
|
|
|
message = 'Failed to unwatch paths: [{!r}]'.format(editor._filename)
|
|
|
|
assert caplog.records[-1].msg == message
|
|
|
|
|
|
|
|
|
2017-10-18 20:33:14 +02:00
|
|
|
@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)),
|
2017-11-08 14:27:09 +01:00
|
|
|
('', None, (1, 1)),
|
2017-10-18 20:33:14 +02:00
|
|
|
])
|
|
|
|
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."""
|
2017-10-18 20:33:14 +02:00
|
|
|
assert editor._calc_line_and_column(text, caret_position) == result
|