mirror of
https://github.com/nvbn/thefuck.git
synced 2025-02-21 20:38:54 +00:00
#102 Add support of rules with side effects
This commit is contained in:
parent
b928a59672
commit
b985dfbffc
@ -74,6 +74,15 @@ def test_run_rule(capsys):
|
|||||||
main.run_rule(Rule(get_new_command=lambda *_: 'new-command'),
|
main.run_rule(Rule(get_new_command=lambda *_: 'new-command'),
|
||||||
None, None)
|
None, None)
|
||||||
assert capsys.readouterr() == ('new-command\n', '')
|
assert capsys.readouterr() == ('new-command\n', '')
|
||||||
|
# With side effect:
|
||||||
|
side_effect = Mock()
|
||||||
|
settings = Mock()
|
||||||
|
command = Mock()
|
||||||
|
main.run_rule(Rule(get_new_command=lambda *_: 'new-command',
|
||||||
|
side_effect=side_effect),
|
||||||
|
command, settings)
|
||||||
|
assert capsys.readouterr() == ('new-command\n', '')
|
||||||
|
side_effect.assert_called_once_with(command, settings)
|
||||||
with patch('thefuck.main.confirm', return_value=False):
|
with patch('thefuck.main.confirm', return_value=False):
|
||||||
main.run_rule(Rule(get_new_command=lambda *_: 'new-command'),
|
main.run_rule(Rule(get_new_command=lambda *_: 'new-command'),
|
||||||
None, None)
|
None, None)
|
||||||
@ -82,15 +91,25 @@ def test_run_rule(capsys):
|
|||||||
|
|
||||||
def test_confirm(capsys):
|
def test_confirm(capsys):
|
||||||
# When confirmation not required:
|
# When confirmation not required:
|
||||||
assert main.confirm('command', Mock(require_confirmation=False))
|
assert main.confirm('command', None, Mock(require_confirmation=False))
|
||||||
assert capsys.readouterr() == ('', 'command\n')
|
assert capsys.readouterr() == ('', 'command\n')
|
||||||
|
# With side effect and without confirmation:
|
||||||
|
assert main.confirm('command', Mock(), Mock(require_confirmation=False))
|
||||||
|
assert capsys.readouterr() == ('', 'command*\n')
|
||||||
# When confirmation required and confirmed:
|
# When confirmation required and confirmed:
|
||||||
with patch('thefuck.main.sys.stdin.read', return_value='\n'):
|
with patch('thefuck.main.sys.stdin.read', return_value='\n'):
|
||||||
assert main.confirm('command', Mock(require_confirmation=True,
|
assert main.confirm(
|
||||||
no_colors=True))
|
'command', None, Mock(require_confirmation=True,
|
||||||
|
no_colors=True))
|
||||||
assert capsys.readouterr() == ('', 'command [enter/ctrl+c]')
|
assert capsys.readouterr() == ('', 'command [enter/ctrl+c]')
|
||||||
|
# With side effect:
|
||||||
|
assert main.confirm(
|
||||||
|
'command', Mock(), Mock(require_confirmation=True,
|
||||||
|
no_colors=True))
|
||||||
|
assert capsys.readouterr() == ('', 'command* [enter/ctrl+c]')
|
||||||
# When confirmation required and ctrl+c:
|
# When confirmation required and ctrl+c:
|
||||||
with patch('thefuck.main.sys.stdin.read', side_effect=KeyboardInterrupt):
|
with patch('thefuck.main.sys.stdin.read', side_effect=KeyboardInterrupt):
|
||||||
assert not main.confirm('command', Mock(require_confirmation=True,
|
assert not main.confirm('command', None,
|
||||||
no_colors=True))
|
Mock(require_confirmation=True,
|
||||||
|
no_colors=True))
|
||||||
assert capsys.readouterr() == ('', 'command [enter/ctrl+c]Aborted\n')
|
assert capsys.readouterr() == ('', 'command [enter/ctrl+c]Aborted\n')
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
from thefuck.types import Rule, RulesNamesList, Settings
|
from thefuck.types import RulesNamesList, Settings
|
||||||
|
from tests.utils import Rule
|
||||||
|
|
||||||
|
|
||||||
def test_rules_names_list():
|
def test_rules_names_list():
|
||||||
assert RulesNamesList(['bash', 'lisp']) == ['bash', 'lisp']
|
assert RulesNamesList(['bash', 'lisp']) == ['bash', 'lisp']
|
||||||
assert RulesNamesList(['bash', 'lisp']) == RulesNamesList(['bash', 'lisp'])
|
assert RulesNamesList(['bash', 'lisp']) == RulesNamesList(['bash', 'lisp'])
|
||||||
assert Rule('lisp', None, None, False) in RulesNamesList(['lisp'])
|
assert Rule('lisp') in RulesNamesList(['lisp'])
|
||||||
assert Rule('bash', None, None, False) not in RulesNamesList(['lisp'])
|
assert Rule('bash') not in RulesNamesList(['lisp'])
|
||||||
|
|
||||||
|
|
||||||
def test_update_settings():
|
def test_update_settings():
|
||||||
|
@ -7,5 +7,7 @@ def Command(script='', stdout='', stderr=''):
|
|||||||
|
|
||||||
def Rule(name='', match=lambda *_: True,
|
def Rule(name='', match=lambda *_: True,
|
||||||
get_new_command=lambda *_: '',
|
get_new_command=lambda *_: '',
|
||||||
enabled_by_default=True):
|
enabled_by_default=True,
|
||||||
return types.Rule(name, match, get_new_command, enabled_by_default)
|
side_effect=None):
|
||||||
|
return types.Rule(name, match, get_new_command,
|
||||||
|
enabled_by_default, side_effect)
|
||||||
|
@ -26,17 +26,20 @@ def rule_failed(rule, exc_info, settings):
|
|||||||
exception('Rule {}'.format(rule.name), exc_info, settings)
|
exception('Rule {}'.format(rule.name), exc_info, settings)
|
||||||
|
|
||||||
|
|
||||||
def show_command(new_command, settings):
|
def show_command(new_command, side_effect, settings):
|
||||||
sys.stderr.write('{bold}{command}{reset}\n'.format(
|
sys.stderr.write('{bold}{command}{side_effect}{reset}\n'.format(
|
||||||
command=new_command,
|
command=new_command,
|
||||||
|
side_effect='*' if side_effect else '',
|
||||||
bold=color(colorama.Style.BRIGHT, settings),
|
bold=color(colorama.Style.BRIGHT, settings),
|
||||||
reset=color(colorama.Style.RESET_ALL, settings)))
|
reset=color(colorama.Style.RESET_ALL, settings)))
|
||||||
|
|
||||||
|
|
||||||
def confirm_command(new_command, settings):
|
def confirm_command(new_command, side_effect, settings):
|
||||||
sys.stderr.write(
|
sys.stderr.write(
|
||||||
'{bold}{command}{reset} [{green}enter{reset}/{red}ctrl+c{reset}]'.format(
|
'{bold}{command}{side_effect}{reset} '
|
||||||
|
'[{green}enter{reset}/{red}ctrl+c{reset}]'.format(
|
||||||
command=new_command,
|
command=new_command,
|
||||||
|
side_effect='*' if side_effect else '',
|
||||||
bold=color(colorama.Style.BRIGHT, settings),
|
bold=color(colorama.Style.BRIGHT, settings),
|
||||||
green=color(colorama.Fore.GREEN, settings),
|
green=color(colorama.Fore.GREEN, settings),
|
||||||
red=color(colorama.Fore.RED, settings),
|
red=color(colorama.Fore.RED, settings),
|
||||||
|
@ -24,7 +24,8 @@ def load_rule(rule):
|
|||||||
rule_module = load_source(rule.name[:-3], str(rule))
|
rule_module = load_source(rule.name[:-3], str(rule))
|
||||||
return types.Rule(rule.name[:-3], rule_module.match,
|
return types.Rule(rule.name[:-3], rule_module.match,
|
||||||
rule_module.get_new_command,
|
rule_module.get_new_command,
|
||||||
getattr(rule_module, 'enabled_by_default', True))
|
getattr(rule_module, 'enabled_by_default', True),
|
||||||
|
getattr(rule_module, 'side_effect', None))
|
||||||
|
|
||||||
|
|
||||||
def get_rules(user_dir, settings):
|
def get_rules(user_dir, settings):
|
||||||
@ -85,13 +86,13 @@ def get_matched_rule(command, rules, settings):
|
|||||||
logs.rule_failed(rule, sys.exc_info(), settings)
|
logs.rule_failed(rule, sys.exc_info(), settings)
|
||||||
|
|
||||||
|
|
||||||
def confirm(new_command, settings):
|
def confirm(new_command, side_effect, settings):
|
||||||
"""Returns `True` when running of new command confirmed."""
|
"""Returns `True` when running of new command confirmed."""
|
||||||
if not settings.require_confirmation:
|
if not settings.require_confirmation:
|
||||||
logs.show_command(new_command, settings)
|
logs.show_command(new_command, side_effect, settings)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
logs.confirm_command(new_command, settings)
|
logs.confirm_command(new_command, side_effect, settings)
|
||||||
try:
|
try:
|
||||||
sys.stdin.read(1)
|
sys.stdin.read(1)
|
||||||
return True
|
return True
|
||||||
@ -103,7 +104,9 @@ def confirm(new_command, settings):
|
|||||||
def run_rule(rule, command, settings):
|
def run_rule(rule, command, settings):
|
||||||
"""Runs command from rule for passed command."""
|
"""Runs command from rule for passed command."""
|
||||||
new_command = rule.get_new_command(command, settings)
|
new_command = rule.get_new_command(command, settings)
|
||||||
if confirm(new_command, settings):
|
if confirm(new_command, rule.side_effect, settings):
|
||||||
|
if rule.side_effect:
|
||||||
|
rule.side_effect(command, settings)
|
||||||
print(new_command)
|
print(new_command)
|
||||||
|
|
||||||
|
|
||||||
|
@ -4,11 +4,11 @@ from collections import namedtuple
|
|||||||
Command = namedtuple('Command', ('script', 'stdout', 'stderr'))
|
Command = namedtuple('Command', ('script', 'stdout', 'stderr'))
|
||||||
|
|
||||||
Rule = namedtuple('Rule', ('name', 'match', 'get_new_command',
|
Rule = namedtuple('Rule', ('name', 'match', 'get_new_command',
|
||||||
'enabled_by_default'))
|
'enabled_by_default', 'side_effect'))
|
||||||
|
|
||||||
|
|
||||||
class RulesNamesList(list):
|
class RulesNamesList(list):
|
||||||
"""Wrapper a top of list for string rules names."""
|
"""Wrapper a top of list for storing rules names."""
|
||||||
|
|
||||||
def __contains__(self, item):
|
def __contains__(self, item):
|
||||||
return super(RulesNamesList, self).__contains__(item.name)
|
return super(RulesNamesList, self).__contains__(item.name)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user