diff --git a/.pylintrc b/.pylintrc index 31e91a3d3..39cbf052d 100644 --- a/.pylintrc +++ b/.pylintrc @@ -43,6 +43,7 @@ function-rgx=[a-z_][a-z0-9_]{2,50}$ const-rgx=[A-Za-z_][A-Za-z0-9_]{0,30}$ method-rgx=[a-z_][A-Za-z0-9_]{1,50}$ attr-rgx=[a-z_][a-z0-9_]{0,30}$ +class-attribute-rgx=([A-Za-z_][A-Za-z0-9_]{1,30}|(__.*__))$ argument-rgx=[a-z_][a-z0-9_]{0,30}$ variable-rgx=[a-z_][a-z0-9_]{0,30}$ docstring-min-length=3 @@ -64,6 +65,7 @@ valid-metaclass-classmethod-first-arg=cls [TYPECHECK] ignored-modules=PyQt5,PyQt5.QtWebKit +ignored-classes=_CountingAttr [IMPORTS] # WORKAROUND diff --git a/README.asciidoc b/README.asciidoc index bf2231063..212d3bec0 100644 --- a/README.asciidoc +++ b/README.asciidoc @@ -116,6 +116,7 @@ The following software and libraries are required to run qutebrowser: * http://jinja.pocoo.org/[jinja2] * http://pygments.org/[pygments] * http://pyyaml.org/wiki/PyYAML[PyYAML] +* http://www.attrs.org/[attrs] The following libraries are optional: diff --git a/doc/changelog.asciidoc b/doc/changelog.asciidoc index 8c4cbcf79..302a172f7 100644 --- a/doc/changelog.asciidoc +++ b/doc/changelog.asciidoc @@ -27,6 +27,7 @@ Breaking changes - (TODO) New dependency on ruamel.yaml; dropped PyYAML dependency. - (TODO) The QtWebEngine backend is now used by default if available. - New dependency on the QtSql module and Qt sqlite support. +- New dependency on the `attrs` Python module. - New config system which ignores the old config file. - The depedency on PyOpenGL (when using QtWebEngine) got removed. Note that PyQt5.QtOpenGL is still a dependency. diff --git a/misc/requirements/requirements-qutebrowser.txt-raw b/misc/requirements/requirements-qutebrowser.txt-raw index 8aadfa8c4..c66c65beb 100644 --- a/misc/requirements/requirements-qutebrowser.txt-raw +++ b/misc/requirements/requirements-qutebrowser.txt-raw @@ -4,3 +4,4 @@ pyPEG2 PyYAML colorama cssutils +attrs diff --git a/misc/requirements/requirements-tests-git.txt b/misc/requirements/requirements-tests-git.txt index 771cf58c8..6b31140bb 100644 --- a/misc/requirements/requirements-tests-git.txt +++ b/misc/requirements/requirements-tests-git.txt @@ -40,6 +40,7 @@ git+https://github.com/pallets/jinja.git git+https://github.com/pallets/markupsafe.git hg+http://bitbucket.org/birkenfeld/pygments-main hg+https://bitbucket.org/fdik/pypeg +git+https://github.com/python-attrs/attrs.git # Fails to build: # gcc: error: ext/_yaml.c: No such file or directory diff --git a/qutebrowser/browser/browsertab.py b/qutebrowser/browser/browsertab.py index 60b5b3b31..0b21b6b05 100644 --- a/qutebrowser/browser/browsertab.py +++ b/qutebrowser/browser/browsertab.py @@ -21,6 +21,7 @@ import itertools +import attr from PyQt5.QtCore import pyqtSignal, pyqtSlot, QUrl, QObject, QSizeF, Qt from PyQt5.QtGui import QIcon from PyQt5.QtWidgets import QWidget, QApplication @@ -82,6 +83,7 @@ TerminationStatus = usertypes.enum('TerminationStatus', [ ]) +@attr.s class TabData: """A simple namespace with a fixed set of attributes. @@ -97,13 +99,12 @@ class TabData: fullscreen: Whether the tab has a video shown fullscreen currently. """ - def __init__(self): - self.keep_icon = False - self.viewing_source = False - self.inspector = None - self.override_target = None - self.pinned = False - self.fullscreen = False + keep_icon = attr.ib(False) + viewing_source = attr.ib(False) + inspector = attr.ib(None) + override_target = attr.ib(None) + pinned = attr.ib(False) + fullscreen = attr.ib(False) class AbstractAction: diff --git a/qutebrowser/browser/hints.py b/qutebrowser/browser/hints.py index 14a9cd7c2..8694a6ad5 100644 --- a/qutebrowser/browser/hints.py +++ b/qutebrowser/browser/hints.py @@ -26,6 +26,7 @@ import re import html from string import ascii_lowercase +import attr from PyQt5.QtCore import pyqtSlot, QObject, Qt, QUrl from PyQt5.QtWidgets import QLabel @@ -131,6 +132,7 @@ class HintLabel(QLabel): self.deleteLater() +@attr.s class HintContext: """Context namespace used for hinting. @@ -158,19 +160,18 @@ class HintContext: group: The group of web elements to hint. """ - def __init__(self): - self.all_labels = [] - self.labels = {} - self.target = None - self.baseurl = None - self.to_follow = None - self.rapid = False - self.add_history = False - self.filterstr = None - self.args = [] - self.tab = None - self.group = None - self.hint_mode = None + all_labels = attr.ib(attr.Factory(list)) + labels = attr.ib(attr.Factory(dict)) + target = attr.ib(None) + baseurl = attr.ib(None) + to_follow = attr.ib(None) + rapid = attr.ib(False) + add_history = attr.ib(False) + filterstr = attr.ib(None) + args = attr.ib(attr.Factory(list)) + tab = attr.ib(None) + group = attr.ib(None) + hint_mode = attr.ib(None) def get_args(self, urlstr): """Get the arguments, with {hint-url} replaced by the given URL.""" @@ -389,6 +390,7 @@ class HintManager(QObject): def _cleanup(self): """Clean up after hinting.""" + # pylint: disable=not-an-iterable for label in self._context.all_labels: label.cleanup() @@ -795,6 +797,7 @@ class HintManager(QObject): log.hints.debug("Filtering hints on {!r}".format(filterstr)) visible = [] + # pylint: disable=not-an-iterable for label in self._context.all_labels: try: if self._filter_matches(filterstr, str(label.elem)): diff --git a/qutebrowser/browser/qtnetworkdownloads.py b/qutebrowser/browser/qtnetworkdownloads.py index 22ec4c02c..921fee54d 100644 --- a/qutebrowser/browser/qtnetworkdownloads.py +++ b/qutebrowser/browser/qtnetworkdownloads.py @@ -22,8 +22,8 @@ import io import shutil import functools -import collections +import attr from PyQt5.QtCore import pyqtSlot, pyqtSignal, QTimer from PyQt5.QtNetwork import QNetworkRequest, QNetworkReply @@ -34,7 +34,11 @@ from qutebrowser.browser.webkit import http from qutebrowser.browser.webkit.network import networkmanager -_RetryInfo = collections.namedtuple('_RetryInfo', ['request', 'manager']) +@attr.s +class _RetryInfo: + + request = attr.ib() + manager = attr.ib() class DownloadItem(downloads.AbstractDownloadItem): diff --git a/qutebrowser/browser/webkit/mhtml.py b/qutebrowser/browser/webkit/mhtml.py index ed357f2bd..ccdd03dad 100644 --- a/qutebrowser/browser/webkit/mhtml.py +++ b/qutebrowser/browser/webkit/mhtml.py @@ -25,7 +25,6 @@ import io import os import re import sys -import collections import uuid import email.policy import email.generator @@ -34,15 +33,21 @@ import email.mime.multipart import email.message import quopri +import attr from PyQt5.QtCore import QUrl from qutebrowser.browser import downloads from qutebrowser.browser.webkit import webkitelem from qutebrowser.utils import log, objreg, message, usertypes, utils, urlutils -_File = collections.namedtuple('_File', - ['content', 'content_type', 'content_location', - 'transfer_encoding']) + +@attr.s +class _File: + + content = attr.ib() + content_type = attr.ib() + content_location = attr.ib() + transfer_encoding = attr.ib() _CSS_URL_PATTERNS = [re.compile(x) for x in [ @@ -174,7 +179,7 @@ class MHTMLWriter: root_content: The root content as bytes. content_location: The url of the page as str. content_type: The MIME-type of the root content as str. - _files: Mapping of location->_File namedtuple. + _files: Mapping of location->_File object. """ def __init__(self, root_content, content_location, content_type): diff --git a/qutebrowser/browser/webkit/network/networkmanager.py b/qutebrowser/browser/webkit/network/networkmanager.py index 770cb2f20..f93d47d43 100644 --- a/qutebrowser/browser/webkit/network/networkmanager.py +++ b/qutebrowser/browser/webkit/network/networkmanager.py @@ -24,6 +24,7 @@ import collections import netrc import html +import attr from PyQt5.QtCore import (pyqtSlot, pyqtSignal, QCoreApplication, QUrl, QByteArray) from PyQt5.QtNetwork import QNetworkAccessManager, QNetworkReply, QSslSocket @@ -37,10 +38,19 @@ from qutebrowser.browser.webkit.network import (webkitqutescheme, networkreply, HOSTBLOCK_ERROR_STRING = '%HOSTBLOCK%' -ProxyId = collections.namedtuple('ProxyId', 'type, hostname, port') _proxy_auth_cache = {} +@attr.s +class ProxyId: + + """Information identifying a proxy server.""" + + type = attr.ib() + hostname = attr.ib() + port = attr.ib() + + def _is_secure_cipher(cipher): """Check if a given SSL cipher (hopefully) isn't broken yet.""" tokens = [e.upper() for e in cipher.name().split('-')] diff --git a/qutebrowser/browser/webkit/rfc6266.py b/qutebrowser/browser/webkit/rfc6266.py index c3c8f7a4b..1f71b23e5 100644 --- a/qutebrowser/browser/webkit/rfc6266.py +++ b/qutebrowser/browser/webkit/rfc6266.py @@ -19,11 +19,11 @@ """pyPEG parsing for the RFC 6266 (Content-Disposition) header.""" -import collections import urllib.parse import string import re +import attr import pypeg2 as peg from qutebrowser.utils import utils @@ -210,7 +210,13 @@ class ContentDispositionValue: peg.optional(';')) -LangTagged = collections.namedtuple('LangTagged', ['string', 'langtag']) +@attr.s +class LangTagged: + + """A string with an associated language.""" + + string = attr.ib() + langtag = attr.ib() class Error(Exception): diff --git a/qutebrowser/commands/command.py b/qutebrowser/commands/command.py index 233f9d1a7..8559099f2 100644 --- a/qutebrowser/commands/command.py +++ b/qutebrowser/commands/command.py @@ -24,43 +24,32 @@ import collections import traceback import typing +import attr + from qutebrowser.commands import cmdexc, argparser -from qutebrowser.utils import (log, utils, message, docutils, objreg, - usertypes) +from qutebrowser.utils import log, message, docutils, objreg, usertypes from qutebrowser.utils import debug as debug_utils from qutebrowser.misc import objects +@attr.s class ArgInfo: """Information about an argument.""" - def __init__(self, win_id=False, count=False, hide=False, metavar=None, - flag=None, completion=None, choices=None): - if win_id and count: + win_id = attr.ib(False) + count = attr.ib(False) + hide = attr.ib(False) + metavar = attr.ib(None) + flag = attr.ib(None) + completion = attr.ib(None) + choices = attr.ib(None) + + @win_id.validator + @count.validator + def _validate_exclusive(self, _attr, _value): + if self.win_id and self.count: raise TypeError("Argument marked as both count/win_id!") - self.win_id = win_id - self.count = count - self.flag = flag - self.hide = hide - self.metavar = metavar - self.completion = completion - self.choices = choices - - def __eq__(self, other): - return (self.win_id == other.win_id and - self.count == other.count and - self.flag == other.flag and - self.hide == other.hide and - self.metavar == other.metavar and - self.completion == other.completion and - self.choices == other.choices) - - def __repr__(self): - return utils.get_repr(self, win_id=self.win_id, count=self.count, - flag=self.flag, hide=self.hide, - metavar=self.metavar, completion=self.completion, - choices=self.choices, constructor=True) class Command: diff --git a/qutebrowser/commands/runners.py b/qutebrowser/commands/runners.py index 1b81c37c6..9d480cc5d 100644 --- a/qutebrowser/commands/runners.py +++ b/qutebrowser/commands/runners.py @@ -19,10 +19,10 @@ """Module containing command managers (SearchRunner and CommandRunner).""" -import collections import traceback import re +import attr from PyQt5.QtCore import pyqtSlot, QUrl, QObject from qutebrowser.config import config @@ -31,10 +31,19 @@ from qutebrowser.utils import message, objreg, qtutils, usertypes, utils from qutebrowser.misc import split -ParseResult = collections.namedtuple('ParseResult', ['cmd', 'args', 'cmdline']) last_command = {} +@attr.s +class ParseResult: + + """The result of parsing a commandline.""" + + cmd = attr.ib() + args = attr.ib() + cmdline = attr.ib() + + def _current_url(tabbed_browser): """Convenience method to get the current url.""" try: diff --git a/qutebrowser/completion/completer.py b/qutebrowser/completion/completer.py index c6a2643fb..854231305 100644 --- a/qutebrowser/completion/completer.py +++ b/qutebrowser/completion/completer.py @@ -19,8 +19,7 @@ """Completer attached to a CompletionView.""" -import collections - +import attr from PyQt5.QtCore import pyqtSlot, QObject, QTimer from qutebrowser.config import config @@ -29,9 +28,13 @@ from qutebrowser.utils import log, utils, debug from qutebrowser.completion.models import miscmodels -# Context passed into all completion functions -CompletionInfo = collections.namedtuple('CompletionInfo', - ['config', 'keyconf']) +@attr.s +class CompletionInfo: + + """Context passed into all completion functions.""" + + config = attr.ib() + keyconf = attr.ib() class Completer(QObject): @@ -130,7 +133,9 @@ class Completer(QObject): return [], '', [] parser = runners.CommandParser() result = parser.parse(text, fallback=True, keep=True) + # pylint: disable=not-an-iterable parts = [x for x in result.cmdline if x] + # pylint: enable=not-an-iterable pos = self._cmd.cursorPosition() - len(self._cmd.prefix()) pos = min(pos, len(text)) # Qt treats 2-byte UTF-16 chars as 2 chars log.completion.debug('partitioning {} around position {}'.format(parts, diff --git a/qutebrowser/config/configdata.py b/qutebrowser/config/configdata.py index db24338b8..739086628 100644 --- a/qutebrowser/config/configdata.py +++ b/qutebrowser/config/configdata.py @@ -24,16 +24,29 @@ Module attributes: DATA: A dict of Option objects after init() has been called. """ -import collections import functools +import attr from qutebrowser.config import configtypes from qutebrowser.utils import usertypes, qtutils, utils DATA = None -Option = collections.namedtuple('Option', ['name', 'typ', 'default', - 'backends', 'raw_backends', - 'description']) + + +@attr.s +class Option: + + """Description of an Option in the config. + + Note that this is just an option which exists, with no value associated. + """ + + name = attr.ib() + typ = attr.ib() + default = attr.ib() + backends = attr.ib() + raw_backends = attr.ib() + description = attr.ib() def _raise_invalid_node(name, what, node): diff --git a/qutebrowser/config/configexc.py b/qutebrowser/config/configexc.py index a4514cd1f..4d283d21f 100644 --- a/qutebrowser/config/configexc.py +++ b/qutebrowser/config/configexc.py @@ -19,7 +19,9 @@ """Exceptions related to config parsing.""" -from qutebrowser.utils import utils, jinja +import attr + +from qutebrowser.utils import jinja class Error(Exception): @@ -74,6 +76,7 @@ class NoOptionError(Error): self.option = option +@attr.s class ConfigErrorDesc: """A description of an error happening while reading the config. @@ -84,13 +87,9 @@ class ConfigErrorDesc: traceback: The formatted traceback of the exception. """ - def __init__(self, text, exception, traceback=None): - self.text = text - self.exception = exception - self.traceback = traceback - - def __repr__(self): - return utils.get_repr(self, text=self.text, exception=self.exception) + text = attr.ib() + exception = attr.ib() + traceback = attr.ib(None) def __str__(self): return '{}: {}'.format(self.text, self.exception) diff --git a/qutebrowser/config/configtypes.py b/qutebrowser/config/configtypes.py index b92722d16..c669ff426 100644 --- a/qutebrowser/config/configtypes.py +++ b/qutebrowser/config/configtypes.py @@ -47,13 +47,13 @@ import html import codecs import os.path import itertools -import collections import warnings import datetime import functools import operator import json +import attr import yaml from PyQt5.QtCore import QUrl, Qt from PyQt5.QtGui import QColor, QFont @@ -1360,8 +1360,15 @@ class FuzzyUrl(BaseType): raise configexc.ValidationError(value, str(e)) -PaddingValues = collections.namedtuple('PaddingValues', ['top', 'bottom', - 'left', 'right']) +@attr.s +class PaddingValues: + + """Four padding values.""" + + top = attr.ib() + bottom = attr.ib() + left = attr.ib() + right = attr.ib() class Padding(Dict): diff --git a/qutebrowser/keyinput/modeman.py b/qutebrowser/keyinput/modeman.py index 504ccc916..6a6dd5459 100644 --- a/qutebrowser/keyinput/modeman.py +++ b/qutebrowser/keyinput/modeman.py @@ -21,6 +21,7 @@ import functools +import attr from PyQt5.QtCore import pyqtSlot, pyqtSignal, Qt, QObject, QEvent from PyQt5.QtWidgets import QApplication @@ -31,6 +32,7 @@ from qutebrowser.utils import usertypes, log, objreg, utils from qutebrowser.misc import objects +@attr.s(frozen=True) class KeyEvent: """A small wrapper over a QKeyEvent storing its data. @@ -44,18 +46,13 @@ class KeyEvent: text: A string (QKeyEvent::text). """ - def __init__(self, keyevent): - self.key = keyevent.key() - self.text = keyevent.text() + key = attr.ib() + text = attr.ib() - def __repr__(self): - return utils.get_repr(self, key=self.key, text=self.text) - - def __eq__(self, other): - return self.key == other.key and self.text == other.text - - def __hash__(self): - return hash((self.key, self.text)) + @classmethod + def from_event(cls, event): + """Initialize a KeyEvent from a QKeyEvent.""" + return cls(event.key(), event.text()) class NotInModeError(Exception): @@ -179,7 +176,7 @@ class ModeManager(QObject): filter_this = True if not filter_this: - self._releaseevents_to_pass.add(KeyEvent(event)) + self._releaseevents_to_pass.add(KeyEvent.from_event(event)) if curmode != usertypes.KeyMode.insert: focus_widget = QApplication.instance().focusWidget() @@ -201,7 +198,7 @@ class ModeManager(QObject): True if event should be filtered, False otherwise. """ # handle like matching KeyPress - keyevent = KeyEvent(event) + keyevent = KeyEvent.from_event(event) if keyevent in self._releaseevents_to_pass: self._releaseevents_to_pass.remove(keyevent) filter_this = False diff --git a/qutebrowser/mainwindow/prompt.py b/qutebrowser/mainwindow/prompt.py index 1ea81c75f..4242ceb8e 100644 --- a/qutebrowser/mainwindow/prompt.py +++ b/qutebrowser/mainwindow/prompt.py @@ -23,6 +23,7 @@ import os.path import html import collections +import attr import sip from PyQt5.QtCore import (pyqtSlot, pyqtSignal, Qt, QTimer, QDir, QModelIndex, QItemSelectionModel, QObject, QEventLoop) @@ -39,7 +40,13 @@ from qutebrowser.commands import cmdutils, cmdexc prompt_queue = None -AuthTuple = collections.namedtuple('AuthTuple', ['user', 'password']) +@attr.s +class AuthInfo: + + """Authentication info returned by a prompt.""" + + user = attr.ib() + password = attr.ib() class Error(Exception): @@ -750,7 +757,7 @@ class AuthenticationPrompt(_BasePrompt): "username:password, but {} was given".format( value)) username, password = value.split(':', maxsplit=1) - self.question.answer = AuthTuple(username, password) + self.question.answer = AuthInfo(username, password) return True elif self._user_lineedit.hasFocus(): # Earlier, tab was bound to :prompt-accept, so to still support @@ -758,8 +765,8 @@ class AuthenticationPrompt(_BasePrompt): self._password_lineedit.setFocus() return False else: - self.question.answer = AuthTuple(self._user_lineedit.text(), - self._password_lineedit.text()) + self.question.answer = AuthInfo(self._user_lineedit.text(), + self._password_lineedit.text()) return True def item_focus(self, which): diff --git a/qutebrowser/mainwindow/statusbar/bar.py b/qutebrowser/mainwindow/statusbar/bar.py index ca75be1ac..e8aacc935 100644 --- a/qutebrowser/mainwindow/statusbar/bar.py +++ b/qutebrowser/mainwindow/statusbar/bar.py @@ -19,6 +19,7 @@ """The main statusbar widget.""" +import attr from PyQt5.QtCore import pyqtSignal, pyqtSlot, pyqtProperty, Qt, QSize, QTimer from PyQt5.QtWidgets import QWidget, QHBoxLayout, QStackedLayout, QSizePolicy @@ -31,6 +32,7 @@ from qutebrowser.mainwindow.statusbar import (backforward, command, progress, from qutebrowser.mainwindow.statusbar import text as textwidget +@attr.s class ColorFlags: """Flags which change the appearance of the statusbar. @@ -44,13 +46,11 @@ class ColorFlags: """ CaretMode = usertypes.enum('CaretMode', ['off', 'on', 'selection']) - - def __init__(self): - self.prompt = False - self.insert = False - self.command = False - self.caret = self.CaretMode.off - self.private = False + prompt = attr.ib(False) + insert = attr.ib(False) + command = attr.ib(False) + caret = attr.ib(CaretMode.off) + private = attr.ib(False) def to_stringlist(self): """Get a string list of set flags used in the stylesheet. diff --git a/qutebrowser/mainwindow/tabbedbrowser.py b/qutebrowser/mainwindow/tabbedbrowser.py index acf966cd5..13865ba90 100644 --- a/qutebrowser/mainwindow/tabbedbrowser.py +++ b/qutebrowser/mainwindow/tabbedbrowser.py @@ -20,8 +20,8 @@ """The main tabbed browser widget.""" import functools -import collections +import attr from PyQt5.QtWidgets import QSizePolicy from PyQt5.QtCore import pyqtSignal, pyqtSlot, QTimer, QUrl from PyQt5.QtGui import QIcon @@ -34,8 +34,15 @@ from qutebrowser.utils import (log, usertypes, utils, qtutils, objreg, urlutils, message, jinja) -UndoEntry = collections.namedtuple('UndoEntry', - ['url', 'history', 'index', 'pinned']) +@attr.s +class UndoEntry: + + """Information needed for :undo.""" + + url = attr.ib() + history = attr.ib() + index = attr.ib() + pinned = attr.ib() class TabDeletedError(Exception): @@ -64,7 +71,7 @@ class TabbedBrowser(tabwidget.TabWidget): _tab_insert_idx_left: Where to insert a new tab with tabs.new_tab_position set to 'prev'. _tab_insert_idx_right: Same as above, for 'next'. - _undo_stack: List of UndoEntry namedtuples of closed tabs. + _undo_stack: List of UndoEntry objects of closed tabs. shutting_down: Whether we're currently shutting down. _local_marks: Jump markers local to each page _global_marks: Jump markers used across all pages @@ -352,16 +359,16 @@ class TabbedBrowser(tabwidget.TabWidget): use_current_tab = (only_one_tab_open and no_history and last_close_url_used) - url, history_data, idx, pinned = self._undo_stack.pop() + entry = self._undo_stack.pop() if use_current_tab: - self.openurl(url, newtab=False) + self.openurl(entry.url, newtab=False) newtab = self.widget(0) else: - newtab = self.tabopen(url, background=False, idx=idx) + newtab = self.tabopen(entry.url, background=False, idx=entry.index) - newtab.history.deserialize(history_data) - self.set_tab_pinned(newtab, pinned) + newtab.history.deserialize(entry.history) + self.set_tab_pinned(newtab, entry.pinned) @pyqtSlot('QUrl', bool) def openurl(self, url, newtab): diff --git a/qutebrowser/mainwindow/tabwidget.py b/qutebrowser/mainwindow/tabwidget.py index f86699589..111be2931 100644 --- a/qutebrowser/mainwindow/tabwidget.py +++ b/qutebrowser/mainwindow/tabwidget.py @@ -19,9 +19,9 @@ """The tab widget used for TabbedBrowser from browser.py.""" -import collections import functools +import attr from PyQt5.QtCore import (pyqtSignal, pyqtSlot, Qt, QSize, QRect, QPoint, QTimer, QUrl) from PyQt5.QtWidgets import (QTabWidget, QTabBar, QSizePolicy, QCommonStyle, @@ -592,8 +592,17 @@ class TabBar(QTabBar): tabbed_browser.wheelEvent(e) -# Used by TabBarStyle._tab_layout(). -Layouts = collections.namedtuple('Layouts', ['text', 'icon', 'indicator']) +@attr.s +class Layouts: + + """Layout information for tab. + + Used by TabBarStyle._tab_layout(). + """ + + text = attr.ib() + icon = attr.ib() + indicator = attr.ib() class TabBarStyle(QCommonStyle): @@ -765,7 +774,7 @@ class TabBarStyle(QCommonStyle): opt: QStyleOptionTab Return: - A Layout namedtuple with two QRects. + A Layout object with two QRects. """ padding = config.val.tabs.padding indicator_padding = config.val.tabs.indicator_padding diff --git a/qutebrowser/misc/crashsignal.py b/qutebrowser/misc/crashsignal.py index e51855af2..393aa26f0 100644 --- a/qutebrowser/misc/crashsignal.py +++ b/qutebrowser/misc/crashsignal.py @@ -27,13 +27,13 @@ import signal import functools import faulthandler import os.path -import collections try: # WORKAROUND for segfaults when using pdb in pytest for some reason... import readline # pylint: disable=unused-import except ImportError: pass +import attr from PyQt5.QtCore import (pyqtSlot, qInstallMessageHandler, QObject, QSocketNotifier, QTimer, QUrl) from PyQt5.QtWidgets import QApplication, QDialog @@ -43,8 +43,14 @@ from qutebrowser.misc import earlyinit, crashdialog from qutebrowser.utils import usertypes, standarddir, log, objreg, debug -ExceptionInfo = collections.namedtuple('ExceptionInfo', - 'pages, cmd_history, objects') +@attr.s +class ExceptionInfo: + + """Information stored when there was an exception.""" + + pages = attr.ib() + cmd_history = attr.ib() + objects = attr.ib() # Used by mainwindow.py to skip confirm questions on crashes @@ -172,7 +178,7 @@ class CrashHandler(QObject): """Get info needed for the exception hook/dialog. Return: - An ExceptionInfo namedtuple. + An ExceptionInfo object. """ try: pages = self._recover_pages(forgiving=True) diff --git a/qutebrowser/misc/earlyinit.py b/qutebrowser/misc/earlyinit.py index 058a66882..032dfc53b 100644 --- a/qutebrowser/misc/earlyinit.py +++ b/qutebrowser/misc/earlyinit.py @@ -261,6 +261,9 @@ def check_libraries(): "http://pyyaml.org/download/pyyaml/ (py3.4) " "or Install via pip.", pip="PyYAML"), + 'attr': + _missing_str("attrs", + pip="attrs"), 'PyQt5.QtQml': _missing_str("PyQt5.QtQml"), 'PyQt5.QtSql': _missing_str("PyQt5.QtSql"), 'PyQt5.QtOpenGL': _missing_str("PyQt5.QtOpenGL"), diff --git a/qutebrowser/utils/utils.py b/qutebrowser/utils/utils.py index 23c37a042..b60b1bc13 100644 --- a/qutebrowser/utils/utils.py +++ b/qutebrowser/utils/utils.py @@ -31,6 +31,7 @@ import contextlib import socket import shlex +import attr from PyQt5.QtCore import Qt, QUrl from PyQt5.QtGui import QKeySequence, QColor, QClipboard, QDesktopServices from PyQt5.QtWidgets import QApplication @@ -409,6 +410,7 @@ def keyevent_to_string(e): return normalize_keystr('+'.join(parts)) +@attr.s(repr=False) class KeyInfo: """Stores information about a key, like used in a QKeyEvent. @@ -419,10 +421,9 @@ class KeyInfo: text: str """ - def __init__(self, key, modifiers, text): - self.key = key - self.modifiers = modifiers - self.text = text + key = attr.ib() + modifiers = attr.ib() + text = attr.ib() def __repr__(self): if self.modifiers is None: @@ -434,10 +435,6 @@ class KeyInfo: key=debug.qenum_key(Qt, self.key), modifiers=modifiers, text=self.text) - def __eq__(self, other): - return (self.key == other.key and self.modifiers == other.modifiers and - self.text == other.text) - class KeyParseError(Exception): diff --git a/qutebrowser/utils/version.py b/qutebrowser/utils/version.py index 75ac197fa..673590208 100644 --- a/qutebrowser/utils/version.py +++ b/qutebrowser/utils/version.py @@ -29,6 +29,7 @@ import importlib import collections import pkg_resources +import attr from PyQt5.QtCore import PYQT_VERSION_STR, QLibraryInfo from PyQt5.QtNetwork import QSslSocket from PyQt5.QtWidgets import QApplication @@ -49,8 +50,15 @@ from qutebrowser.misc import objects, earlyinit, sql from qutebrowser.browser import pdfjs -DistributionInfo = collections.namedtuple( - 'DistributionInfo', ['id', 'parsed', 'version', 'pretty']) +@attr.s +class DistributionInfo: + + """Information about the running distribution.""" + + id = attr.ib() + parsed = attr.ib() + version = attr.ib() + pretty = attr.ib() Distribution = usertypes.enum( @@ -190,24 +198,25 @@ def _module_versions(): ('pygments', ['__version__']), ('yaml', ['__version__']), ('cssutils', ['__version__']), + ('attr', ['__version__']), ('PyQt5.QtWebEngineWidgets', []), ('PyQt5.QtWebKitWidgets', []), ]) - for name, attributes in modules.items(): + for modname, attributes in modules.items(): try: - module = importlib.import_module(name) + module = importlib.import_module(modname) except ImportError: - text = '{}: no'.format(name) + text = '{}: no'.format(modname) else: - for attr in attributes: + for name in attributes: try: - text = '{}: {}'.format(name, getattr(module, attr)) + text = '{}: {}'.format(modname, getattr(module, name)) except AttributeError: pass else: break else: - text = '{}: yes'.format(name) + text = '{}: yes'.format(modname) lines.append(text) return lines diff --git a/requirements.txt b/requirements.txt index b2cc93c1f..6a4739aba 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,5 +1,6 @@ # This file is automatically generated by scripts/dev/recompile_requirements.py +attrs==17.2.0 colorama==0.3.9 cssutils==1.0.2 Jinja2==2.9.6 diff --git a/scripts/dev/check_coverage.py b/scripts/dev/check_coverage.py index 0848c404b..dfd336d16 100644 --- a/scripts/dev/check_coverage.py +++ b/scripts/dev/check_coverage.py @@ -25,16 +25,26 @@ import sys import enum import os.path import subprocess -import collections - from xml.etree import ElementTree +import attr + sys.path.insert(0, os.path.join(os.path.dirname(__file__), os.pardir, os.pardir)) from scripts import utils -Message = collections.namedtuple('Message', 'typ, filename, text') + +@attr.s +class Message: + + """A message shown by coverage.py.""" + + typ = attr.ib() + filename = attr.ib() + text = attr.ib() + + MsgType = enum.Enum('MsgType', 'insufficent_coverage, perfect_file') diff --git a/scripts/dev/get_coredumpctl_traces.py b/scripts/dev/get_coredumpctl_traces.py index 8cae2a190..6d1a41a11 100644 --- a/scripts/dev/get_coredumpctl_traces.py +++ b/scripts/dev/get_coredumpctl_traces.py @@ -24,17 +24,29 @@ import os import sys import argparse import subprocess -import collections import os.path import tempfile +import attr + sys.path.insert(0, os.path.join(os.path.dirname(__file__), os.pardir, os.pardir)) from scripts import utils -Line = collections.namedtuple('Line', 'time, pid, uid, gid, sig, present, exe') +@attr.s +class Line: + + """A line in "coredumpctl list".""" + + time = attr.ib() + pid = attr.ib() + uid = attr.ib() + gid = attr.ib() + sig = attr.ib() + present = attr.ib() + exe = attr.ib() def _convert_present(data): diff --git a/scripts/dev/run_vulture.py b/scripts/dev/run_vulture.py index c747787f6..7e5713615 100755 --- a/scripts/dev/run_vulture.py +++ b/scripts/dev/run_vulture.py @@ -115,6 +115,12 @@ def whitelist_generator(): '_get_default_metavar_for_positional', '_metavar_formatter']: yield 'scripts.dev.src2asciidoc.UsageFormatter.' + attr + # attrs + yield 'qutebrowser.browser.webkit.network.networkmanager.ProxyId.hostname' + yield 'qutebrowser.command.command.ArgInfo._validate_exclusive' + yield 'scripts.get_coredumpctl_traces.Line.uid' + yield 'scripts.get_coredumpctl_traces.Line.gid' + def filter_func(item): """Check if a missing function should be filtered or not. diff --git a/setup.py b/setup.py index f594009a0..7bfd968f6 100755 --- a/setup.py +++ b/setup.py @@ -44,7 +44,7 @@ try: ['qutebrowser = qutebrowser.qutebrowser:main']}, test_suite='qutebrowser.test', zip_safe=True, - install_requires=['pypeg2', 'jinja2', 'pygments', 'PyYAML'], + install_requires=['pypeg2', 'jinja2', 'pygments', 'PyYAML', 'attrs'], **common.setupdata ) finally: diff --git a/tests/end2end/fixtures/testprocess.py b/tests/end2end/fixtures/testprocess.py index 68b30a105..e955937bc 100644 --- a/tests/end2end/fixtures/testprocess.py +++ b/tests/end2end/fixtures/testprocess.py @@ -23,6 +23,7 @@ import re import os import time +import attr import pytest import pytestqt.plugin from PyQt5.QtCore import (pyqtSlot, pyqtSignal, QProcess, QObject, @@ -58,6 +59,7 @@ class BlacklistedMessageError(Exception): """Raised when ensure_not_logged found a message.""" +@attr.s class Line: """Container for a line of data the process emits. @@ -67,12 +69,8 @@ class Line: waited_for: If Process.wait_for was used on this line already. """ - def __init__(self, data): - self.data = data - self.waited_for = False - - def __repr__(self): - return '{}({!r})'.format(self.__class__.__name__, self.data) + data = attr.ib() + waited_for = attr.ib(False) def _render_log(data, threshold=100): diff --git a/tests/end2end/fixtures/webserver.py b/tests/end2end/fixtures/webserver.py index 01887d9b9..d9d8f5eca 100644 --- a/tests/end2end/fixtures/webserver.py +++ b/tests/end2end/fixtures/webserver.py @@ -25,6 +25,7 @@ import json import os.path import http.client +import attr import pytest from PyQt5.QtCore import pyqtSignal, QUrl @@ -99,13 +100,13 @@ class Request(testprocess.Line): return NotImplemented +@attr.s(frozen=True, cmp=False, hash=True) class ExpectedRequest: """Class to compare expected requests easily.""" - def __init__(self, verb, path): - self.verb = verb - self.path = path + verb = attr.ib() + path = attr.ib() @classmethod def from_request(cls, request): @@ -118,13 +119,6 @@ class ExpectedRequest: else: return NotImplemented - def __hash__(self): - return hash(('ExpectedRequest', self.verb, self.path)) - - def __repr__(self): - return ('ExpectedRequest(verb={!r}, path={!r})' - .format(self.verb, self.path)) - class WebserverProcess(testprocess.Process): diff --git a/tests/end2end/test_dirbrowser.py b/tests/end2end/test_dirbrowser.py index 0a2c3d7b9..83fa1d4c0 100644 --- a/tests/end2end/test_dirbrowser.py +++ b/tests/end2end/test_dirbrowser.py @@ -20,8 +20,8 @@ """Test the built-in directory browser.""" import os -import collections +import attr import pytest import bs4 @@ -101,8 +101,21 @@ class DirLayout: return os.path.normpath(str(self.base)) -Parsed = collections.namedtuple('Parsed', 'path, parent, folders, files') -Item = collections.namedtuple('Item', 'path, link, text') +@attr.s +class Parsed: + + path = attr.ib() + parent = attr.ib() + folders = attr.ib() + files = attr.ib() + + +@attr.s +class Item: + + path = attr.ib() + link = attr.ib() + text = attr.ib() def parse(quteproc): @@ -182,6 +195,7 @@ def test_enter_folder_smoke(dir_layout, quteproc): @pytest.mark.parametrize('folder', DirLayout.layout_folders()) def test_enter_folder(dir_layout, quteproc, folder): + # pylint: disable=not-an-iterable quteproc.open_url(dir_layout.file_url()) quteproc.click_element_by_text(text=folder) expected_url = urlutils.file_url(dir_layout.path(folder)) diff --git a/tests/end2end/test_hints_html.py b/tests/end2end/test_hints_html.py index 7ba13f7d0..abc106505 100644 --- a/tests/end2end/test_hints_html.py +++ b/tests/end2end/test_hints_html.py @@ -22,8 +22,8 @@ import os import os.path import textwrap -import collections +import attr import yaml import pytest import bs4 @@ -36,8 +36,11 @@ def collect_tests(): return files -ParsedFile = collections.namedtuple('ParsedFile', ['target', - 'qtwebengine_todo']) +@attr.s +class ParsedFile: + + target = attr.ib() + qtwebengine_todo = attr.ib() class InvalidFile(Exception): diff --git a/tests/helpers/fixtures.py b/tests/helpers/fixtures.py index 08aea97f6..496eb486d 100644 --- a/tests/helpers/fixtures.py +++ b/tests/helpers/fixtures.py @@ -26,13 +26,13 @@ See https://pytest.org/latest/fixture.html import sys -import collections import tempfile import itertools import textwrap import unittest.mock import types +import attr import pytest import py.path # pylint: disable=no-name-in-module @@ -53,7 +53,12 @@ class WinRegistryHelper: """Helper class for win_registry.""" - FakeWindow = collections.namedtuple('FakeWindow', ['registry']) + @attr.s + class FakeWindow: + + """A fake window object for the registry.""" + + registry = attr.ib() def __init__(self): self._ids = [] @@ -161,8 +166,12 @@ def fake_web_tab(stubs, tab_registry, mode_manager, qapp): def _generate_cmdline_tests(): """Generate testcases for test_split_binding.""" - # pylint: disable=invalid-name - TestCase = collections.namedtuple('TestCase', 'cmd, valid') + @attr.s + class TestCase: + + cmd = attr.ib() + valid = attr.ib() + separators = [';;', ' ;; ', ';; ', ' ;;'] invalid = ['foo', ''] valid = ['leave-mode', 'hint all'] diff --git a/tests/helpers/messagemock.py b/tests/helpers/messagemock.py index 77116115f..6fad75281 100644 --- a/tests/helpers/messagemock.py +++ b/tests/helpers/messagemock.py @@ -20,14 +20,20 @@ """pytest helper to monkeypatch the message module.""" import logging -import collections +import attr import pytest from qutebrowser.utils import usertypes, message -Message = collections.namedtuple('Message', ['level', 'text']) +@attr.s +class Message: + + """Information about a shown message.""" + + level = attr.ib() + text = attr.ib() class MessageMock: @@ -35,8 +41,8 @@ class MessageMock: """Helper object for message_mock. Attributes: - Message: A namedtuple representing a message. - messages: A list of Message tuples. + Message: A object representing a message. + messages: A list of Message objects. """ def __init__(self): diff --git a/tests/helpers/stubs.py b/tests/helpers/stubs.py index d7dcb427f..1b386c201 100644 --- a/tests/helpers/stubs.py +++ b/tests/helpers/stubs.py @@ -23,6 +23,7 @@ from unittest import mock +import attr from PyQt5.QtCore import pyqtSignal, QPoint, QProcess, QObject from PyQt5.QtNetwork import (QNetworkRequest, QAbstractNetworkCache, QNetworkCacheMetaData) @@ -202,9 +203,9 @@ class FakeNetworkReply: def fake_qprocess(): """Factory for a QProcess mock which has the QProcess enum values.""" m = mock.Mock(spec=QProcess) - for attr in ['NormalExit', 'CrashExit', 'FailedToStart', 'Crashed', + for name in ['NormalExit', 'CrashExit', 'FailedToStart', 'Crashed', 'Timedout', 'WriteError', 'ReadError', 'UnknownError']: - setattr(m, attr, getattr(QProcess, attr)) + setattr(m, name, getattr(QProcess, name)) return m @@ -315,27 +316,26 @@ class FakeSignal: pass +@attr.s class FakeCmdUtils: """Stub for cmdutils which provides a cmd_dict.""" - def __init__(self, commands): - self.cmd_dict = commands + cmd_dict = attr.ib() +@attr.s(frozen=True) class FakeCommand: """A simple command stub which has a description.""" - def __init__(self, name='', desc='', hide=False, debug=False, - deprecated=False, completion=None, maxsplit=None): - self.desc = desc - self.name = name - self.hide = hide - self.debug = debug - self.deprecated = deprecated - self.completion = completion - self.maxsplit = maxsplit + name = attr.ib('') + desc = attr.ib('') + hide = attr.ib(False) + debug = attr.ib(False) + deprecated = attr.ib(False) + completion = attr.ib(None) + maxsplit = attr.ib(None) class FakeTimer(QObject): diff --git a/tests/unit/browser/test_signalfilter.py b/tests/unit/browser/test_signalfilter.py index b56e4ecc0..9d191e7c6 100644 --- a/tests/unit/browser/test_signalfilter.py +++ b/tests/unit/browser/test_signalfilter.py @@ -19,9 +19,9 @@ """Tests for browser.signalfilter.""" -import collections import logging +import attr import pytest from PyQt5.QtCore import pyqtSignal, pyqtSlot, QObject @@ -47,7 +47,11 @@ class Signaller(QObject): self.filtered_signal_arg = s -Objects = collections.namedtuple('Objects', 'signal_filter, signaller') +@attr.s +class Objects: + + signal_filter = attr.ib() + signaller = attr.ib() @pytest.fixture diff --git a/tests/unit/browser/webkit/network/test_filescheme.py b/tests/unit/browser/webkit/network/test_filescheme.py index 7a1bd9b19..0cc485556 100644 --- a/tests/unit/browser/webkit/network/test_filescheme.py +++ b/tests/unit/browser/webkit/network/test_filescheme.py @@ -18,8 +18,8 @@ # along with qutebrowser. If not, see . import os -import collections +import attr import pytest import bs4 from PyQt5.QtCore import QUrl @@ -109,8 +109,18 @@ def _file_url(path): class TestDirbrowserHtml: - Parsed = collections.namedtuple('Parsed', 'parent, folders, files') - Item = collections.namedtuple('Item', 'link, text') + @attr.s + class Parsed: + + parent = attr.ib() + folders = attr.ib() + files = attr.ib() + + @attr.s + class Item: + + link = attr.ib() + text = attr.ib() @pytest.fixture def parser(self): diff --git a/tests/unit/browser/webkit/test_tabhistory.py b/tests/unit/browser/webkit/test_tabhistory.py index 07b334771..67602db4c 100644 --- a/tests/unit/browser/webkit/test_tabhistory.py +++ b/tests/unit/browser/webkit/test_tabhistory.py @@ -19,8 +19,7 @@ """Tests for webelement.tabhistory.""" -import collections - +import attr from PyQt5.QtCore import QUrl, QPoint import pytest @@ -50,7 +49,11 @@ ITEMS = [ ] -Objects = collections.namedtuple('Objects', 'history, user_data') +@attr.s +class Objects: + + history = attr.ib() + user_data = attr.ib() @pytest.fixture diff --git a/tests/unit/browser/webkit/test_webkitelem.py b/tests/unit/browser/webkit/test_webkitelem.py index 9b3407821..871241a04 100644 --- a/tests/unit/browser/webkit/test_webkitelem.py +++ b/tests/unit/browser/webkit/test_webkitelem.py @@ -24,6 +24,7 @@ import collections.abc import operator import itertools +import attr import pytest from PyQt5.QtCore import QRect, QPoint, QUrl QWebElement = pytest.importorskip('PyQt5.QtWebKit').QWebElement @@ -525,7 +526,12 @@ class TestIsVisibleIframe: elem1-elem4: FakeWebElements to test. """ - Objects = collections.namedtuple('Objects', ['frame', 'iframe', 'elems']) + @attr.s + class Objects: + + frame = attr.ib() + iframe = attr.ib() + elems = attr.ib() @pytest.fixture def objects(self, stubs): @@ -550,7 +556,7 @@ class TestIsVisibleIframe: ############################## 300, 0 300, 300 - Returns an Objects namedtuple with frame/iframe/elems attributes. + Returns an Objects object with frame/iframe/elems attributes. """ frame = stubs.FakeWebFrame(QRect(0, 0, 300, 300)) iframe = stubs.FakeWebFrame(QRect(0, 10, 100, 100), parent=frame) @@ -621,7 +627,7 @@ class TestIsVisibleIframe: ############################## 300, 0 300, 300 - Returns an Objects namedtuple with frame/iframe/elems attributes. + Returns an Objects object with frame/iframe/elems attributes. """ frame = stubs.FakeWebFrame(QRect(0, 0, 300, 300)) iframe = stubs.FakeWebFrame(QRect(0, 10, 100, 100), parent=frame) diff --git a/tests/unit/config/test_configtypes.py b/tests/unit/config/test_configtypes.py index 861c052c2..122a1c20c 100644 --- a/tests/unit/config/test_configtypes.py +++ b/tests/unit/config/test_configtypes.py @@ -21,12 +21,12 @@ import re import json import math -import collections import itertools import warnings import inspect import functools +import attr import pytest import hypothesis from hypothesis import strategies @@ -54,15 +54,14 @@ class Font(QFont): @classmethod def fromdesc(cls, desc): """Get a Font based on a font description.""" - style, weight, ptsize, pxsize, family = desc f = cls() - f.setStyle(style) - f.setWeight(weight) - if ptsize is not None and ptsize != -1: - f.setPointSize(ptsize) - if pxsize is not None and ptsize != -1: - f.setPixelSize(pxsize) - f.setFamily(family) + f.setStyle(desc.style) + f.setWeight(desc.weight) + if desc.pt is not None and desc.pt != -1: + f.setPointSize(desc.pt) + if desc.px is not None and desc.pt != -1: + f.setPixelSize(desc.px) + f.setFamily(desc.family) return f @@ -1195,8 +1194,14 @@ class TestColors: klass().to_py(val) -FontDesc = collections.namedtuple('FontDesc', - ['style', 'weight', 'pt', 'px', 'family']) +@attr.s +class FontDesc: + + style = attr.ib() + weight = attr.ib() + pt = attr.ib() + px = attr.ib() + family = attr.ib() class TestFont: diff --git a/tests/unit/misc/test_ipc.py b/tests/unit/misc/test_ipc.py index a6a663a0d..2b7c17e62 100644 --- a/tests/unit/misc/test_ipc.py +++ b/tests/unit/misc/test_ipc.py @@ -22,12 +22,12 @@ import sys import os import getpass -import collections import logging import json import hashlib from unittest import mock +import attr import pytest from PyQt5.QtCore import pyqtSignal, QObject from PyQt5.QtNetwork import QLocalServer, QLocalSocket, QAbstractSocket @@ -597,8 +597,13 @@ def test_ipcserver_socket_none_error(ipc_server, caplog): class TestSendOrListen: - Args = collections.namedtuple('Args', 'no_err_windows, basedir, command, ' - 'target') + @attr.s + class Args: + + no_err_windows = attr.ib() + basedir = attr.ib() + command = attr.ib() + target = attr.ib() @pytest.fixture def args(self): @@ -623,10 +628,10 @@ class TestSendOrListen: def qlocalsocket_mock(self, mocker): m = mocker.patch('qutebrowser.misc.ipc.QLocalSocket', autospec=True) m().errorString.return_value = "Error string" - for attr in ['UnknownSocketError', 'UnconnectedState', + for name in ['UnknownSocketError', 'UnconnectedState', 'ConnectionRefusedError', 'ServerNotFoundError', 'PeerClosedError']: - setattr(m, attr, getattr(QLocalSocket, attr)) + setattr(m, name, getattr(QLocalSocket, name)) return m @pytest.mark.linux(reason="Flaky on Windows and macOS") diff --git a/tests/unit/misc/test_split.py b/tests/unit/misc/test_split.py index 179084849..eff6d2269 100644 --- a/tests/unit/misc/test_split.py +++ b/tests/unit/misc/test_split.py @@ -18,8 +18,8 @@ # along with qutebrowser. If not, see . """Tests for qutebrowser.misc.split.""" -import collections +import attr import pytest from qutebrowser.misc import split @@ -104,21 +104,26 @@ foo\ bar/foo bar/foo\ bar/ def _parse_split_test_data_str(): - """Parse the test data set into a namedtuple to use in tests. + """Parse the test data set into a TestCase object to use in tests. Returns: - A list of namedtuples with str attributes: input, keep, no_keep + A list of TestCase objects with str attributes: inp, keep, no_keep """ - tuple_class = collections.namedtuple('TestCase', 'input, keep, no_keep') + @attr.s + class TestCase: + + inp = attr.ib() + keep = attr.ib() + no_keep = attr.ib() for line in test_data_str.splitlines(): if not line: continue data = line.split('/') - item = tuple_class(input=data[0], keep=data[1].split('|'), - no_keep=data[2].split('|')) + item = TestCase(inp=data[0], keep=data[1].split('|'), + no_keep=data[2].split('|')) yield item - yield tuple_class(input='', keep=[], no_keep=[]) + yield TestCase(inp='', keep=[], no_keep=[]) class TestSplit: @@ -137,17 +142,17 @@ class TestSplit: def test_split(self, split_test_case): """Test splitting.""" - items = split.split(split_test_case.input) + items = split.split(split_test_case.inp) assert items == split_test_case.keep def test_split_keep_original(self, split_test_case): """Test if splitting with keep=True yields the original string.""" - items = split.split(split_test_case.input, keep=True) - assert ''.join(items) == split_test_case.input + items = split.split(split_test_case.inp, keep=True) + assert ''.join(items) == split_test_case.inp def test_split_keep(self, split_test_case): """Test splitting with keep=True.""" - items = split.split(split_test_case.input, keep=True) + items = split.split(split_test_case.inp, keep=True) assert items == split_test_case.no_keep diff --git a/tests/unit/utils/test_standarddir.py b/tests/unit/utils/test_standarddir.py index d652310d5..5230c9c78 100644 --- a/tests/unit/utils/test_standarddir.py +++ b/tests/unit/utils/test_standarddir.py @@ -25,10 +25,10 @@ import json import os.path import types import textwrap -import collections import logging import subprocess +import attr from PyQt5.QtCore import QStandardPaths import pytest @@ -207,9 +207,6 @@ class TestStandardDir: assert func().split(os.sep)[-elems:] == expected -DirArgTest = collections.namedtuple('DirArgTest', 'arg, expected') - - class TestArguments: """Tests the --basedir argument.""" @@ -370,10 +367,16 @@ class TestMoveWindowsAndMacOS: @pytest.fixture def files(self, tmpdir): - files = collections.namedtuple('Files', [ - 'auto_config_dir', 'config_dir', - 'local_data_dir', 'roaming_data_dir']) - return files( + + @attr.s + class Files: + + auto_config_dir = attr.ib() + config_dir = attr.ib() + local_data_dir = attr.ib() + roaming_data_dir = attr.ib() + + return Files( auto_config_dir=tmpdir / 'auto_config' / APPNAME, config_dir=tmpdir / 'config' / APPNAME, local_data_dir=tmpdir / 'data' / APPNAME, @@ -412,11 +415,17 @@ class TestMove: @pytest.fixture def dirs(self, tmpdir): - dirs = collections.namedtuple('Dirs', ['old', 'new', - 'old_file', 'new_file']) + @attr.s + class Dirs: + + old = attr.ib() + new = attr.ib() + old_file = attr.ib() + new_file = attr.ib() + old_dir = tmpdir / 'old' new_dir = tmpdir / 'new' - return dirs(old=old_dir, new=new_dir, + return Dirs(old=old_dir, new=new_dir, old_file=old_dir / 'file', new_file=new_dir / 'file') def test_no_old_dir(self, dirs, caplog): diff --git a/tests/unit/utils/test_urlutils.py b/tests/unit/utils/test_urlutils.py index a6b1242d1..875cc2967 100644 --- a/tests/unit/utils/test_urlutils.py +++ b/tests/unit/utils/test_urlutils.py @@ -20,9 +20,9 @@ """Tests for qutebrowser.utils.urlutils.""" import os.path -import collections import logging +import attr from PyQt5.QtCore import QUrl from PyQt5.QtNetwork import QNetworkProxy import pytest @@ -38,7 +38,7 @@ class FakeDNS: """Helper class for the fake_dns fixture. Class attributes: - FakeDNSAnswer: Helper class/namedtuple imitating a QHostInfo object + FakeDNSAnswer: Helper class imitating a QHostInfo object (used by fromname_mock). Attributes: @@ -48,7 +48,10 @@ class FakeDNS: when fromname_mock is called. """ - FakeDNSAnswer = collections.namedtuple('FakeDNSAnswer', ['error']) + @attr.s + class FakeDNSAnswer: + + error = attr.ib() def __init__(self): self.used = False diff --git a/tests/unit/utils/test_utils.py b/tests/unit/utils/test_utils.py index 255c88344..623f77cc7 100644 --- a/tests/unit/utils/test_utils.py +++ b/tests/unit/utils/test_utils.py @@ -25,11 +25,11 @@ import os.path import io import logging import functools -import collections import socket import re import shlex +import attr from PyQt5.QtCore import Qt, QUrl from PyQt5.QtGui import QColor, QClipboard import pytest @@ -157,7 +157,11 @@ class TestInterpolateColor: white: The Color black as a valid Color for tests. """ - Colors = collections.namedtuple('Colors', ['white', 'black']) + @attr.s + class Colors: + + white = attr.ib() + black = attr.ib() @pytest.fixture def colors(self): diff --git a/tests/unit/utils/test_version.py b/tests/unit/utils/test_version.py index 4d977540d..4983a4839 100644 --- a/tests/unit/utils/test_version.py +++ b/tests/unit/utils/test_version.py @@ -32,6 +32,7 @@ import logging import textwrap import pkg_resources +import attr import pytest import qutebrowser @@ -474,8 +475,8 @@ def test_path_info(monkeypatch, equal): 'runtime': lambda: 'RUNTIME PATH', } - for attr, val in patches.items(): - monkeypatch.setattr(version.standarddir, attr, val) + for name, val in patches.items(): + monkeypatch.setattr(version.standarddir, name, val) pathinfo = version._path_info() @@ -515,6 +516,7 @@ class ImportFake: ('pygments', True), ('yaml', True), ('cssutils', True), + ('attr', True), ('PyQt5.QtWebEngineWidgets', True), ('PyQt5.QtWebKitWidgets', True), ]) @@ -630,6 +632,7 @@ class TestModuleVersions: ('pygments', True), ('yaml', True), ('cssutils', True), + ('attr', True), ]) def test_existing_attributes(self, name, has_version): """Check if all dependencies have an expected __version__ attribute. @@ -817,17 +820,16 @@ def test_chromium_version_unpatched(qapp): assert version._chromium_version() not in ['', 'unknown', 'unavailable'] +@attr.s class VersionParams: - def __init__(self, name, git_commit=True, frozen=False, style=True, - with_webkit=True, known_distribution=True, ssl_support=True): - self.name = name - self.git_commit = git_commit - self.frozen = frozen - self.style = style - self.with_webkit = with_webkit - self.known_distribution = known_distribution - self.ssl_support = ssl_support + name = attr.ib() + git_commit = attr.ib(True) + frozen = attr.ib(False) + style = attr.ib(True) + with_webkit = attr.ib(True) + known_distribution = attr.ib(True) + ssl_support = attr.ib(True) @pytest.mark.parametrize('params', [ @@ -901,8 +903,8 @@ def test_version_output(params, stubs, monkeypatch): substitutions['ssl'] = 'SSL VERSION' if params.ssl_support else 'no' - for attr, val in patches.items(): - monkeypatch.setattr('qutebrowser.utils.version.' + attr, val) + for name, val in patches.items(): + monkeypatch.setattr('qutebrowser.utils.version.' + name, val) if params.frozen: monkeypatch.setattr(sys, 'frozen', True, raising=False)